Files
web/apps/scandic-web/stores/destination-data/index.ts
Anton Gunnarsson 002d093af4 Merged in feat/sw-2863-move-contentstack-router-to-trpc-package (pull request #2389)
feat(SW-2863): Move contentstack router to trpc package

* Add exports to packages and lint rule to prevent relative imports

* Add env to trpc package

* Add eslint to trpc package

* Apply lint rules

* Use direct imports from trpc package

* Add lint-staged config to trpc

* Move lang enum to common

* Restructure trpc package folder structure

* WIP first step

* update internal imports in trpc

* Fix most errors in scandic-web

Just 100 left...

* Move Props type out of trpc

* Fix CategorizedFilters types

* Move more schemas in hotel router

* Fix deps

* fix getNonContentstackUrls

* Fix import error

* Fix entry error handling

* Fix generateMetadata metrics

* Fix alertType enum

* Fix duplicated types

* lint:fix

* Merge branch 'master' into feat/sw-2863-move-contentstack-router-to-trpc-package

* Fix broken imports

* Merge branch 'master' into feat/sw-2863-move-contentstack-router-to-trpc-package


Approved-by: Linus Flood
2025-06-26 07:53:01 +00:00

201 lines
6.2 KiB
TypeScript

import { produce } from "immer"
import { useContext } from "react"
import { create, useStore } from "zustand"
import { getSortedCities } from "@scandic-hotels/trpc/utils/getSortedCities"
import { DestinationDataContext } from "@/contexts/DestinationData"
import {
trackFilterChangeEvent,
trackSortingChangeEvent,
} from "@/utils/tracking/destinationPage"
import {
getBasePathNameWithoutFilters,
getFilteredCities,
getFilteredHotels,
getSortedHotels,
isValidSortOption,
} from "./helper"
import type { Filter } from "@scandic-hotels/trpc/types/destinationFilterAndSort"
import type {
DestinationDataState,
InitialState,
} from "@/types/stores/destination-data"
export function createDestinationDataStore({
allCities,
allHotels,
allFilters,
filterFromUrl,
pathname,
sortItems,
searchParams,
}: InitialState) {
const defaultSort =
sortItems.find((s) => s.isDefault)?.value ?? sortItems[0].value
const allFilterSlugs = Object.values(allFilters).flatMap((filter: Filter[]) =>
filter.map((f) => f.slug)
)
const activeFilters: string[] = filterFromUrl ? [filterFromUrl] : []
const basePathnameWithoutFilters = getBasePathNameWithoutFilters(
pathname,
allFilterSlugs
)
if (basePathnameWithoutFilters !== pathname) {
const pathParts = pathname.split("/")
const lastPathPart = pathParts[pathParts.length - 1]
activeFilters.push(lastPathPart)
}
let activeSort = defaultSort
if (searchParams) {
const sortParam = searchParams.get("sort")
if (sortParam && isValidSortOption(sortParam, sortItems)) {
activeSort = sortParam
}
}
const filteredHotels = getFilteredHotels(allHotels, activeFilters)
const activeHotels = getSortedHotels(filteredHotels, activeSort)
const filteredCities = getFilteredCities(filteredHotels, allCities)
const activeCities = getSortedCities(filteredCities, activeSort)
return create<DestinationDataState>((set) => ({
actions: {
updateActiveFiltersAndSort(filters, sort) {
return set(
produce((state: DestinationDataState) => {
const newSort =
sort && isValidSortOption(sort, state.sortItems)
? sort
: state.defaultSort
const filteredHotels = getFilteredHotels(state.allHotels, filters)
const sortedHotels = getSortedHotels(filteredHotels, newSort)
const filteredCities = state.allHotels.length
? getFilteredCities(filteredHotels, state.allCities)
: []
const sortedCities = getSortedCities(filteredCities, newSort)
// Tracking
if (newSort !== state.activeSort) {
trackSortingChangeEvent(newSort)
}
if (
JSON.stringify(filters) !== JSON.stringify(state.activeFilters)
) {
const facilityFiltersUsed = filters.filter((f) =>
state.allFilters.facilityFilters
.map((ff) => ff.slug)
.includes(f)
)
const surroundingsFiltersUsed = filters.filter((f) =>
state.allFilters.surroundingsFilters
.map((sf) => sf.slug)
.includes(f)
)
trackFilterChangeEvent(
facilityFiltersUsed,
surroundingsFiltersUsed
)
}
state.activeSort = newSort
state.activeFilters = filters
state.activeHotels = sortedHotels
state.activeCities = sortedCities
state.pendingFilters = filters
state.pendingSort = newSort
state.pendingHotelCount = filteredHotels.length
state.pendingCityCount = filteredCities.length
state.isLoading = false
})
)
},
setIsLoading(isLoading) {
return set(
produce((state: DestinationDataState) => {
state.isLoading = isLoading
})
)
},
setPendingSort(sort) {
return set(
produce((state: DestinationDataState) => {
state.pendingSort = sort
})
)
},
togglePendingFilter(filter) {
return set(
produce((state: DestinationDataState) => {
const isActive = state.pendingFilters.includes(filter)
const filters = isActive
? state.pendingFilters.filter((f) => f !== filter)
: [...state.pendingFilters, filter]
const pendingHotels = getFilteredHotels(state.allHotels, filters)
const pendingCities = state.allHotels.length
? getFilteredCities(pendingHotels, state.allCities)
: []
state.pendingFilters = filters
state.pendingHotelCount = pendingHotels.length
state.pendingCityCount = pendingCities.length
})
)
},
clearPendingFilters() {
return set(
produce((state: DestinationDataState) => {
state.pendingFilters = []
state.pendingHotelCount = state.allHotels.length
state.pendingCityCount = state.allCities.length
})
)
},
resetPendingValues() {
return set(
produce((state: DestinationDataState) => {
state.pendingFilters = state.activeFilters
state.pendingSort = state.activeSort
state.pendingHotelCount = state.activeHotels.length
state.pendingCityCount = state.activeCities.length
})
)
},
},
allHotels,
activeHotels: activeHotels,
pendingHotelCount: activeHotels.length,
allCities,
activeCities: activeCities,
pendingCityCount: activeCities.length,
activeSort,
pendingSort: activeSort,
defaultSort,
activeFilters,
pendingFilters: activeFilters,
allFilters,
allFilterSlugs,
basePathnameWithoutFilters,
sortItems,
isLoading: false,
}))
}
export function useDestinationDataStore<T>(
selector: (store: DestinationDataState) => T
) {
const store = useContext(DestinationDataContext)
if (!store) {
throw new Error("useHotelDataStore must be used within HotelDataProvider")
}
return useStore(store, selector)
}