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:
Erik Tiekstra
2025-09-24 10:40:58 +00:00
parent f3b6c4a089
commit af4f544b8a
15 changed files with 141 additions and 61 deletions

View File

@@ -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}
> >

View File

@@ -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}
> >

View File

@@ -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,

View File

@@ -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)

View File

@@ -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

View File

@@ -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
} }

View File

@@ -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
} }

View File

@@ -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(),

View File

@@ -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(),

View File

@@ -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,
}) })

View File

@@ -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
) )

View File

@@ -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,
}
}

View File

@@ -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,
}))

View File

@@ -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[]
}

View File

@@ -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