feat(BOOK-55): Listen to SEO filter slugs when navigating to such page
Approved-by: Chuma Mcphoy (We Ahead) Approved-by: Matilda Landström
This commit is contained in:
@@ -64,7 +64,8 @@ export default async function DestinationCityPage({
|
|||||||
} = destinationCityPage
|
} = destinationCityPage
|
||||||
|
|
||||||
const allHotels = await getHotelsByCityIdentifier(cityIdentifier)
|
const allHotels = await getHotelsByCityIdentifier(cityIdentifier)
|
||||||
const allFilters = getFiltersFromHotels(allHotels, lang)
|
const hotelFilters = getFiltersFromHotels(allHotels, lang)
|
||||||
|
|
||||||
const sortItems: HotelSortItem[] = [
|
const sortItems: HotelSortItem[] = [
|
||||||
{
|
{
|
||||||
label: intl.formatMessage({
|
label: intl.formatMessage({
|
||||||
@@ -92,7 +93,8 @@ export default async function DestinationCityPage({
|
|||||||
<Suspense fallback={<DestinationCityPageSkeleton />}>
|
<Suspense fallback={<DestinationCityPageSkeleton />}>
|
||||||
<DestinationDataProvider
|
<DestinationDataProvider
|
||||||
allHotels={allHotels}
|
allHotels={allHotels}
|
||||||
allFilters={allFilters}
|
hotelFilters={hotelFilters}
|
||||||
|
seoFilters={seo_filters}
|
||||||
sortItems={sortItems}
|
sortItems={sortItems}
|
||||||
pathname={pathname}
|
pathname={pathname}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ export default async function DestinationCountryPage({
|
|||||||
getHotelsByCountry(destination_settings.country),
|
getHotelsByCountry(destination_settings.country),
|
||||||
getDestinationCityPagesByCountry(destination_settings.country),
|
getDestinationCityPagesByCountry(destination_settings.country),
|
||||||
])
|
])
|
||||||
const allFilters = getFiltersFromHotels(allHotels, lang)
|
const hotelFilters = getFiltersFromHotels(allHotels, lang)
|
||||||
|
|
||||||
const sortItems: HotelSortItem[] = [
|
const sortItems: HotelSortItem[] = [
|
||||||
{
|
{
|
||||||
@@ -91,7 +91,8 @@ export default async function DestinationCountryPage({
|
|||||||
<DestinationDataProvider
|
<DestinationDataProvider
|
||||||
allHotels={allHotels}
|
allHotels={allHotels}
|
||||||
allCities={allCities}
|
allCities={allCities}
|
||||||
allFilters={allFilters}
|
hotelFilters={hotelFilters}
|
||||||
|
seoFilters={seo_filters}
|
||||||
sortItems={sortItems}
|
sortItems={sortItems}
|
||||||
pathname={pathname}
|
pathname={pathname}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -14,7 +14,8 @@ import type { DestinationDataProviderProps } from "@/types/providers/destination
|
|||||||
export default function DestinationDataProvider({
|
export default function DestinationDataProvider({
|
||||||
allCities = [],
|
allCities = [],
|
||||||
allHotels,
|
allHotels,
|
||||||
allFilters,
|
hotelFilters,
|
||||||
|
seoFilters,
|
||||||
sortItems,
|
sortItems,
|
||||||
pathname,
|
pathname,
|
||||||
children,
|
children,
|
||||||
@@ -26,7 +27,8 @@ export default function DestinationDataProvider({
|
|||||||
storeRef.current = createDestinationDataStore({
|
storeRef.current = createDestinationDataStore({
|
||||||
allCities,
|
allCities,
|
||||||
allHotels,
|
allHotels,
|
||||||
allFilters,
|
hotelFilters,
|
||||||
|
seoFilters,
|
||||||
pathname,
|
pathname,
|
||||||
sortItems,
|
sortItems,
|
||||||
searchParams,
|
searchParams,
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { produce } from "immer"
|
|||||||
import { useContext } from "react"
|
import { useContext } from "react"
|
||||||
import { create, useStore } from "zustand"
|
import { create, useStore } from "zustand"
|
||||||
|
|
||||||
|
import { mergeHotelFiltersAndSeoFilters } from "@scandic-hotels/trpc/utils/getFiltersFromHotels"
|
||||||
import { getSortedCities } from "@scandic-hotels/trpc/utils/getSortedCities"
|
import { getSortedCities } from "@scandic-hotels/trpc/utils/getSortedCities"
|
||||||
|
|
||||||
import { DestinationDataContext } from "@/contexts/DestinationData"
|
import { DestinationDataContext } from "@/contexts/DestinationData"
|
||||||
@@ -28,15 +29,17 @@ import type {
|
|||||||
export function createDestinationDataStore({
|
export function createDestinationDataStore({
|
||||||
allCities,
|
allCities,
|
||||||
allHotels,
|
allHotels,
|
||||||
allFilters,
|
hotelFilters,
|
||||||
|
seoFilters,
|
||||||
pathname,
|
pathname,
|
||||||
sortItems,
|
sortItems,
|
||||||
searchParams,
|
searchParams,
|
||||||
}: InitialState) {
|
}: InitialState) {
|
||||||
const defaultSort =
|
const defaultSort =
|
||||||
sortItems.find((s) => s.isDefault)?.value ?? sortItems[0].value
|
sortItems.find((s) => s.isDefault)?.value ?? sortItems[0].value
|
||||||
const flattenedFilters = Object.values(allFilters).flat<HotelFilter[]>()
|
const allFilters = mergeHotelFiltersAndSeoFilters(hotelFilters, seoFilters)
|
||||||
const allFilterSlugs = flattenedFilters.map((filter) => filter.slug)
|
const allFlattenedFilters = Object.values(allFilters).flat<HotelFilter[]>()
|
||||||
|
const allFilterSlugs = allFlattenedFilters.map((filter) => filter.slug)
|
||||||
const activeFilters: HotelFilter[] = []
|
const activeFilters: HotelFilter[] = []
|
||||||
let filterFromUrl: HotelFilter | null = null
|
let filterFromUrl: HotelFilter | null = null
|
||||||
|
|
||||||
@@ -47,7 +50,7 @@ export function createDestinationDataStore({
|
|||||||
if (basePathnameWithoutFilters !== pathname) {
|
if (basePathnameWithoutFilters !== pathname) {
|
||||||
const pathParts = pathname.split("/")
|
const pathParts = pathname.split("/")
|
||||||
filterFromUrl =
|
filterFromUrl =
|
||||||
flattenedFilters.find(
|
allFlattenedFilters.find(
|
||||||
(filter) => filter.slug === pathParts[pathParts.length - 1]
|
(filter) => filter.slug === pathParts[pathParts.length - 1]
|
||||||
) ?? null
|
) ?? null
|
||||||
if (filterFromUrl) {
|
if (filterFromUrl) {
|
||||||
@@ -76,11 +79,11 @@ export function createDestinationDataStore({
|
|||||||
sort && isValidSortOption(sort, state.sortItems)
|
sort && isValidSortOption(sort, state.sortItems)
|
||||||
? sort
|
? sort
|
||||||
: state.defaultSort
|
: state.defaultSort
|
||||||
const filters = flattenedFilters.filter((filter) =>
|
const filters = allFlattenedFilters.filter((filter) =>
|
||||||
filterSlugs.includes(filter.slug)
|
filterSlugs.includes(filter.slug)
|
||||||
)
|
)
|
||||||
const filterFromUrl =
|
const filterFromUrl =
|
||||||
flattenedFilters.find(
|
allFlattenedFilters.find(
|
||||||
(filter) => filter.slug === filterSlugFromUrl
|
(filter) => filter.slug === filterSlugFromUrl
|
||||||
) ?? null
|
) ?? null
|
||||||
const filteredHotels = getFilteredHotels(state.allHotels, filters)
|
const filteredHotels = getFilteredHotels(state.allHotels, filters)
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import type { DestinationCityListItem } from "@scandic-hotels/trpc/types/destinationCityPage"
|
import type { DestinationCityListItem } from "@scandic-hotels/trpc/types/destinationCityPage"
|
||||||
|
import type { SEOFilters } from "@scandic-hotels/trpc/types/destinationsData"
|
||||||
import type {
|
import type {
|
||||||
CategorizedHotelFilters,
|
CategorizedHotelFilters,
|
||||||
HotelListingHotelData,
|
HotelListingHotelData,
|
||||||
@@ -8,7 +9,8 @@ import type {
|
|||||||
export interface DestinationDataProviderProps extends React.PropsWithChildren {
|
export interface DestinationDataProviderProps extends React.PropsWithChildren {
|
||||||
allHotels: HotelListingHotelData[]
|
allHotels: HotelListingHotelData[]
|
||||||
allCities?: DestinationCityListItem[]
|
allCities?: DestinationCityListItem[]
|
||||||
allFilters: CategorizedHotelFilters
|
hotelFilters: CategorizedHotelFilters
|
||||||
|
seoFilters: SEOFilters | null
|
||||||
filterFromUrl?: string
|
filterFromUrl?: string
|
||||||
sortItems: HotelSortItem[]
|
sortItems: HotelSortItem[]
|
||||||
pathname: string
|
pathname: string
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import type { DestinationCityListItem } from "@scandic-hotels/trpc/types/destinationCityPage"
|
import type { DestinationCityListItem } from "@scandic-hotels/trpc/types/destinationCityPage"
|
||||||
|
import type { SEOFilters } from "@scandic-hotels/trpc/types/destinationsData"
|
||||||
import type {
|
import type {
|
||||||
CategorizedHotelFilters,
|
CategorizedHotelFilters,
|
||||||
HotelFilter,
|
HotelFilter,
|
||||||
@@ -42,10 +43,9 @@ export interface DestinationDataState {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface InitialState
|
export interface InitialState
|
||||||
extends Pick<
|
extends Pick<DestinationDataState, "allHotels" | "allCities" | "sortItems"> {
|
||||||
DestinationDataState,
|
|
||||||
"allHotels" | "allCities" | "sortItems" | "allFilters"
|
|
||||||
> {
|
|
||||||
pathname: string
|
pathname: string
|
||||||
searchParams: ReadonlyURLSearchParams
|
searchParams: ReadonlyURLSearchParams
|
||||||
|
seoFilters: SEOFilters | null
|
||||||
|
hotelFilters: CategorizedHotelFilters
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
#import "../../Fragments/Metadata.graphql"
|
#import "../../Fragments/Metadata.graphql"
|
||||||
#import "../../Fragments/System.graphql"
|
#import "../../Fragments/System.graphql"
|
||||||
|
#import "../../Fragments/HotelFilter.graphql"
|
||||||
|
|
||||||
query GetDestinationCityPageMetadata($locale: String!, $uid: String!) {
|
query GetDestinationCityPageMetadata($locale: String!, $uid: String!) {
|
||||||
destination_city_page(locale: $locale, uid: $uid) {
|
destination_city_page(locale: $locale, uid: $uid) {
|
||||||
@@ -22,6 +23,15 @@ query GetDestinationCityPageMetadata($locale: String!, $uid: String!) {
|
|||||||
images {
|
images {
|
||||||
image
|
image
|
||||||
}
|
}
|
||||||
|
seo_filters {
|
||||||
|
filterConnection {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
...HotelFilter
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
system {
|
system {
|
||||||
...System
|
...System
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
import { contentRefsSchema, contentSchema } from "../schemas/blocks/content"
|
import { contentRefsSchema, contentSchema } from "../schemas/blocks/content"
|
||||||
import {
|
import {
|
||||||
destinationFiltersRefsSchema,
|
destinationFiltersRefsSchema,
|
||||||
destinationFiltersSchema,
|
transformedDestinationFiltersSchema,
|
||||||
} from "../schemas/destinationFilters"
|
} from "../schemas/destinationFilters"
|
||||||
import { mapLocationSchema } from "../schemas/mapLocation"
|
import { mapLocationSchema } from "../schemas/mapLocation"
|
||||||
import {
|
import {
|
||||||
@@ -163,7 +163,7 @@ export const destinationCityPageSchema = z.object({
|
|||||||
})
|
})
|
||||||
.nullish(),
|
.nullish(),
|
||||||
blocks: discriminatedUnionArray(blocksSchema.options).nullable(),
|
blocks: discriminatedUnionArray(blocksSchema.options).nullable(),
|
||||||
seo_filters: destinationFiltersSchema,
|
seo_filters: transformedDestinationFiltersSchema,
|
||||||
system: systemSchema.merge(
|
system: systemSchema.merge(
|
||||||
z.object({
|
z.object({
|
||||||
created_at: z.string(),
|
created_at: z.string(),
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
import { contentRefsSchema, contentSchema } from "../schemas/blocks/content"
|
import { contentRefsSchema, contentSchema } from "../schemas/blocks/content"
|
||||||
import {
|
import {
|
||||||
destinationFiltersRefsSchema,
|
destinationFiltersRefsSchema,
|
||||||
destinationFiltersSchema,
|
transformedDestinationFiltersSchema,
|
||||||
} from "../schemas/destinationFilters"
|
} from "../schemas/destinationFilters"
|
||||||
import { mapLocationSchema } from "../schemas/mapLocation"
|
import { mapLocationSchema } from "../schemas/mapLocation"
|
||||||
import {
|
import {
|
||||||
@@ -92,7 +92,7 @@ export const destinationCountryPageSchema = z.object({
|
|||||||
})
|
})
|
||||||
.nullish(),
|
.nullish(),
|
||||||
blocks: discriminatedUnionArray(blocksSchema.options).nullable(),
|
blocks: discriminatedUnionArray(blocksSchema.options).nullable(),
|
||||||
seo_filters: destinationFiltersSchema,
|
seo_filters: transformedDestinationFiltersSchema,
|
||||||
system: systemSchema.merge(
|
system: systemSchema.merge(
|
||||||
z.object({
|
z.object({
|
||||||
created_at: z.string(),
|
created_at: z.string(),
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import { Country } from "../../../types/country"
|
|||||||
import { RTETypeEnum } from "../../../types/RTEenums"
|
import { RTETypeEnum } from "../../../types/RTEenums"
|
||||||
import { additionalDataAttributesSchema } from "../../hotels/schemas/hotel/include/additionalData"
|
import { additionalDataAttributesSchema } from "../../hotels/schemas/hotel/include/additionalData"
|
||||||
import { imageSchema } from "../../hotels/schemas/image"
|
import { imageSchema } from "../../hotels/schemas/image"
|
||||||
|
import { destinationFiltersSchema } from "../schemas/destinationFilters"
|
||||||
import { systemSchema } from "../schemas/system"
|
import { systemSchema } from "../schemas/system"
|
||||||
|
|
||||||
import type { Lang } from "@scandic-hotels/common/constants/language"
|
import type { Lang } from "@scandic-hotels/common/constants/language"
|
||||||
@@ -166,6 +167,7 @@ export const rawMetadataSchema = z.object({
|
|||||||
cities: z.array(z.string()).nullish(),
|
cities: z.array(z.string()).nullish(),
|
||||||
})
|
})
|
||||||
.nullish(),
|
.nullish(),
|
||||||
|
seo_filters: destinationFiltersSchema,
|
||||||
system: systemSchema,
|
system: systemSchema,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,9 @@
|
|||||||
import { ApiCountry } from "../../../types/country"
|
import { ApiCountry } from "../../../types/country"
|
||||||
import { HotelSortOption } from "../../../types/hotel"
|
import { HotelSortOption } from "../../../types/hotel"
|
||||||
import { getFiltersFromHotels } from "../../../utils/getFiltersFromHotels"
|
import {
|
||||||
|
getFiltersFromHotels,
|
||||||
|
mergeHotelFiltersAndSeoFilters,
|
||||||
|
} from "../../../utils/getFiltersFromHotels"
|
||||||
import { getSortedCities } from "../../../utils/getSortedCities"
|
import { getSortedCities } from "../../../utils/getSortedCities"
|
||||||
import {
|
import {
|
||||||
getCityByCityIdentifier,
|
getCityByCityIdentifier,
|
||||||
@@ -9,6 +12,7 @@ import {
|
|||||||
getHotelsByHotelIds,
|
getHotelsByHotelIds,
|
||||||
} from "../../hotels/utils"
|
} from "../../hotels/utils"
|
||||||
import { getCityPages } from "../destinationCountryPage/utils"
|
import { getCityPages } from "../destinationCountryPage/utils"
|
||||||
|
import { transformDestinationFiltersResponse } from "../schemas/destinationFilters"
|
||||||
|
|
||||||
import type { Lang } from "@scandic-hotels/common/constants/language"
|
import type { Lang } from "@scandic-hotels/common/constants/language"
|
||||||
|
|
||||||
@@ -24,6 +28,7 @@ export async function getCityData(
|
|||||||
lang: Lang
|
lang: Lang
|
||||||
) {
|
) {
|
||||||
const destinationSettings = data.destination_settings
|
const destinationSettings = data.destination_settings
|
||||||
|
const seoFilters = transformDestinationFiltersResponse(data.seo_filters)
|
||||||
const filter = input.filterFromUrl
|
const filter = input.filterFromUrl
|
||||||
|
|
||||||
if (destinationSettings) {
|
if (destinationSettings) {
|
||||||
@@ -61,7 +66,11 @@ export async function getCityData(
|
|||||||
|
|
||||||
let filterType
|
let filterType
|
||||||
if (filter) {
|
if (filter) {
|
||||||
const allFilters = getFiltersFromHotels(hotels, lang)
|
const hotelFilters = getFiltersFromHotels(hotels, lang)
|
||||||
|
const allFilters = mergeHotelFiltersAndSeoFilters(
|
||||||
|
hotelFilters,
|
||||||
|
seoFilters
|
||||||
|
)
|
||||||
const facilityFilter = allFilters.facilityFilters.find(
|
const facilityFilter = allFilters.facilityFilters.find(
|
||||||
(f) => f.slug === filter
|
(f) => f.slug === filter
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { z } from "zod"
|
import { z } from "zod"
|
||||||
|
|
||||||
|
import { FacilityEnum } from "@scandic-hotels/common/constants/facilities"
|
||||||
import { isDefined } from "@scandic-hotels/common/utils/isDefined"
|
import { isDefined } from "@scandic-hotels/common/utils/isDefined"
|
||||||
|
|
||||||
import { hotelFilterSchema } from "./hotelFilter"
|
|
||||||
import { systemSchema } from "./system"
|
import { systemSchema } from "./system"
|
||||||
|
|
||||||
export const destinationFiltersSchema = z
|
export const destinationFiltersSchema = z
|
||||||
@@ -11,32 +11,25 @@ export const destinationFiltersSchema = z
|
|||||||
filterConnection: z.object({
|
filterConnection: z.object({
|
||||||
edges: z.array(
|
edges: z.array(
|
||||||
z.object({
|
z.object({
|
||||||
node: hotelFilterSchema,
|
node: z.object({
|
||||||
|
title: z.string(),
|
||||||
|
facility_id: z
|
||||||
|
.nativeEnum(FacilityEnum)
|
||||||
|
.catch(FacilityEnum.UNKNOWN),
|
||||||
|
category: z.string(),
|
||||||
|
slug: z.string(),
|
||||||
|
}),
|
||||||
})
|
})
|
||||||
),
|
),
|
||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
.nullish()
|
.nullish()
|
||||||
.transform((data) => {
|
|
||||||
const filters = data
|
|
||||||
?.map(({ filterConnection }) => filterConnection.edges[0]?.node)
|
|
||||||
.filter(isDefined)
|
|
||||||
|
|
||||||
if (!data || !filters?.length) {
|
export const transformedDestinationFiltersSchema =
|
||||||
return null
|
destinationFiltersSchema.transform((data) =>
|
||||||
}
|
transformDestinationFiltersResponse(data)
|
||||||
|
)
|
||||||
const facilityFilters = filters.filter((f) => f.filterType === "facility")
|
|
||||||
const surroundingsFilters = filters.filter(
|
|
||||||
(f) => f.filterType === "surroundings"
|
|
||||||
)
|
|
||||||
|
|
||||||
return {
|
|
||||||
facilityFilters,
|
|
||||||
surroundingsFilters,
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
export const destinationFiltersRefsSchema = z
|
export const destinationFiltersRefsSchema = z
|
||||||
.array(
|
.array(
|
||||||
@@ -53,3 +46,35 @@ export const destinationFiltersRefsSchema = z
|
|||||||
})
|
})
|
||||||
)
|
)
|
||||||
.nullish()
|
.nullish()
|
||||||
|
|
||||||
|
export function transformDestinationFiltersResponse(
|
||||||
|
data: typeof destinationFiltersSchema._type
|
||||||
|
) {
|
||||||
|
const filters = data
|
||||||
|
?.map(({ filterConnection }) => filterConnection.edges[0]?.node)
|
||||||
|
.filter(isDefined)
|
||||||
|
|
||||||
|
if (!data || !filters?.length) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
const transformedFilters = filters.map((filter) => ({
|
||||||
|
id: filter.facility_id,
|
||||||
|
name: filter.title,
|
||||||
|
filterType: filter.category,
|
||||||
|
slug: filter.slug,
|
||||||
|
sortOrder: 0,
|
||||||
|
}))
|
||||||
|
|
||||||
|
const facilityFilters = transformedFilters.filter(
|
||||||
|
(f) => f.filterType === "facility"
|
||||||
|
)
|
||||||
|
const surroundingsFilters = transformedFilters.filter(
|
||||||
|
(f) => f.filterType === "surroundings"
|
||||||
|
)
|
||||||
|
|
||||||
|
return {
|
||||||
|
facilityFilters,
|
||||||
|
surroundingsFilters,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,18 +0,0 @@
|
|||||||
import { z } from "zod"
|
|
||||||
|
|
||||||
import { FacilityEnum } from "@scandic-hotels/common/constants/facilities"
|
|
||||||
|
|
||||||
export const hotelFilterSchema = z
|
|
||||||
.object({
|
|
||||||
title: z.string(),
|
|
||||||
facility_id: z.nativeEnum(FacilityEnum).catch(FacilityEnum.UNKNOWN),
|
|
||||||
category: z.string(),
|
|
||||||
slug: z.string(),
|
|
||||||
})
|
|
||||||
.transform((data) => ({
|
|
||||||
id: data.facility_id,
|
|
||||||
name: data.title,
|
|
||||||
filterType: data.category,
|
|
||||||
slug: data.slug,
|
|
||||||
sortOrder: 0,
|
|
||||||
}))
|
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import type { HotelFilter } from "./hotel"
|
||||||
|
|
||||||
export type City = {
|
export type City = {
|
||||||
id: string
|
id: string
|
||||||
name: string
|
name: string
|
||||||
@@ -14,3 +16,8 @@ export type DestinationCountry = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export type DestinationsData = DestinationCountry[]
|
export type DestinationsData = DestinationCountry[]
|
||||||
|
|
||||||
|
export type SEOFilters = {
|
||||||
|
facilityFilters: HotelFilter[]
|
||||||
|
surroundingsFilters: HotelFilter[]
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
|
import type { FacilityEnum } from "@scandic-hotels/common/constants/facilities"
|
||||||
import type { Lang } from "@scandic-hotels/common/constants/language"
|
import type { Lang } from "@scandic-hotels/common/constants/language"
|
||||||
|
|
||||||
|
import type { SEOFilters } from "../types/destinationsData"
|
||||||
import type {
|
import type {
|
||||||
CategorizedHotelFilters,
|
CategorizedHotelFilters,
|
||||||
HotelFilter,
|
HotelFilter,
|
||||||
@@ -33,6 +35,39 @@ function sortFilters(filters: HotelFilter[]): HotelFilter[] {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Merges hotel and SEO filters, removing duplicates (by id).
|
||||||
|
// In case of duplicates, the SEO filter takes precedence.
|
||||||
|
function mergeAndDeduplicate(
|
||||||
|
hotelFilters: HotelFilter[],
|
||||||
|
seoFilters: HotelFilter[]
|
||||||
|
): HotelFilter[] {
|
||||||
|
const map = new Map<FacilityEnum, HotelFilter>()
|
||||||
|
hotelFilters.forEach((filter) => map.set(filter.id, filter))
|
||||||
|
seoFilters.forEach((filter) => map.set(filter.id, filter))
|
||||||
|
return Array.from(map.values())
|
||||||
|
}
|
||||||
|
|
||||||
|
export function mergeHotelFiltersAndSeoFilters(
|
||||||
|
hotelFilters: CategorizedHotelFilters,
|
||||||
|
seoFilters: SEOFilters | null
|
||||||
|
): CategorizedHotelFilters {
|
||||||
|
if (!seoFilters) {
|
||||||
|
return hotelFilters
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...hotelFilters,
|
||||||
|
facilityFilters: mergeAndDeduplicate(
|
||||||
|
hotelFilters.facilityFilters,
|
||||||
|
seoFilters.facilityFilters
|
||||||
|
),
|
||||||
|
surroundingsFilters: mergeAndDeduplicate(
|
||||||
|
hotelFilters.surroundingsFilters,
|
||||||
|
seoFilters.surroundingsFilters
|
||||||
|
),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export function getFiltersFromHotels(
|
export function getFiltersFromHotels(
|
||||||
hotels: HotelListingHotelData[],
|
hotels: HotelListingHotelData[],
|
||||||
lang: Lang
|
lang: Lang
|
||||||
|
|||||||
Reference in New Issue
Block a user