Files
web/packages/booking-flow/lib/components/SelectRate/RoomsContainer/index.tsx
Bianca Widstam 2dd08bb5d0 Merged in feat/BOOK-747-alert-extra-cost (pull request #3455)
feat(BOOK-747): show extra cost alert if reward night or voucher

* feat(BOOK-747): show extra cost alert if reward night or voucher

* feat(BOOK-747): use enum

* feat(BOOK-747): refactor

* feat(BOOK-747): add underline to trigger text


Approved-by: Anton Gunnarsson
2026-01-20 11:51:24 +00:00

176 lines
4.8 KiB
TypeScript

"use client"
import { TRPCClientError } from "@trpc/client"
import { useEffect } from "react"
import { useIntl } from "react-intl"
import { AlertTypeEnum } from "@scandic-hotels/common/constants/alert"
import { Alert } from "@scandic-hotels/design-system/Alert"
import { trackNoAvailability } from "@scandic-hotels/tracking/NoAvailabilityTracking"
import { SEARCH_TYPE_REDEMPTION } from "@scandic-hotels/trpc/constants/booking"
import { AvailabilityEnum } from "@scandic-hotels/trpc/enums/selectHotel"
import { useSelectRateContext } from "../../../contexts/SelectRate/SelectRateContext"
import useLang from "../../../hooks/useLang"
import { mapPackageToLabel } from "../../../utils/getRoomFeatureDescription"
import { trackLowestRoomPrice, trackRoomsLoaded } from "../Tracking/tracking"
import ExtraCostAlert from "./ExtraCostAlert"
import { RateSummary } from "./RateSummary"
import Rooms from "./Rooms"
import { RoomsContainerSkeleton } from "./RoomsContainerSkeleton"
import styles from "./index.module.css"
import type { HotelData } from "@scandic-hotels/trpc/types/hotel"
interface RoomsContainerProps
extends
Pick<HotelData, "roomCategories">,
Pick<HotelData["hotel"], "hotelType" | "vat"> {}
export function RoomsContainer({}: RoomsContainerProps) {
const intl = useIntl()
const lang = useLang()
const {
hotel,
availability: { error, isFetching, isError, data: availabilityData },
input: { hasError: hasInputError, errorCode, data: inputData },
getLowestRoomPrice,
} = useSelectRateContext()
const { price: lowestRoomPrice, currency: lowestRoomPriceCurrency } =
getLowestRoomPrice() ?? {}
const dataAvailable = availabilityData?.length
const { hotelId, fromDate, toDate, bookingCode, searchType, rooms } =
inputData?.booking ?? {}
const specialRoomType = rooms
?.map((room) => {
const packages = room.packages
?.map((pkg) => {
mapPackageToLabel(pkg)
})
.join(",")
return packages ?? ""
})
.join("|")
useEffect(() => {
if (!dataAvailable) return
if (!hotelId || !fromDate || !toDate || !lowestRoomPrice) return
trackLowestRoomPrice({
hotelId: hotelId,
arrivalDate: fromDate,
departureDate: toDate,
lowestPrice: lowestRoomPrice.toString(),
currency: lowestRoomPriceCurrency,
})
}, [
dataAvailable,
hotelId,
fromDate,
toDate,
lowestRoomPrice,
lowestRoomPriceCurrency,
])
useEffect(() => {
if (isFetching) return
trackRoomsLoaded()
}, [isFetching])
useEffect(() => {
if (isFetching || !availabilityData || !toDate || !fromDate) return
const shouldTrackNoAvailability = availabilityData.some((room) => {
if (!room || "error" in room) return false
return room.roomConfigurations.every(
(roomConfig) => roomConfig.status === AvailabilityEnum.NotAvailable
)
})
if (shouldTrackNoAvailability) {
trackNoAvailability({
specialRoomType,
searchTerm: hotel.data?.additionalData?.name || "",
lang,
fromDate,
toDate,
noOfRooms: rooms?.length,
searchType: "hotel",
hotelId,
rewardNight: searchType === SEARCH_TYPE_REDEMPTION ? "yes" : "no",
bookingCode,
pageId: "select-rate",
pageName: "hotelreservation|select-rate",
pageType: "bookingroomsandratespage",
siteSections: "hotelreservation|select-rate",
domain: typeof window !== "undefined" ? window.location.host : "",
})
}
}, [
isFetching,
availabilityData,
bookingCode,
hotelId,
fromDate,
toDate,
specialRoomType,
hotel.data?.additionalData?.name,
rooms?.length,
searchType,
lang,
])
if (isFetching) {
return <RoomsContainerSkeleton />
}
if (isError || hasInputError) {
const errorMessage = getErrorMessage(error ?? errorCode, intl)
return (
<div className={styles.errorContainer}>
<Alert type={AlertTypeEnum.Alarm} heading={errorMessage} />
</div>
)
}
return (
<>
<ExtraCostAlert />
<Rooms />
<RateSummary />
</>
)
}
function getErrorMessage(error: unknown, intl: ReturnType<typeof useIntl>) {
let errorCode = ""
if (error instanceof TRPCClientError) {
errorCode = error.data?.zodError?.formErrors?.at(0)
} else if (typeof error == "string") {
errorCode = error
}
switch (errorCode) {
case "FROMDATE_INVALID":
case "TODATE_INVALID":
case "TODATE_MUST_BE_AFTER_FROMDATE":
case "FROMDATE_CANNOT_BE_IN_THE_PAST": {
return intl.formatMessage({
id: "selectRate.invalidDatesErrorMessage",
defaultMessage: "Invalid dates",
})
}
default:
return intl.formatMessage({
id: "errorMessage.somethingWentWrong",
defaultMessage: "Something went wrong!",
})
}
}