189 lines
6.5 KiB
TypeScript
189 lines
6.5 KiB
TypeScript
import { useEffect, useState } from "react"
|
|
import { useIntl } from "react-intl"
|
|
|
|
import { dt } from "@/lib/dt"
|
|
import { useRateSelectionStore } from "@/stores/select-rate/rate-selection"
|
|
|
|
import SignupPromoDesktop from "@/components/HotelReservation/SignupPromo/Desktop"
|
|
import SignupPromoMobile from "@/components/HotelReservation/SignupPromo/Mobile"
|
|
import Button from "@/components/TempDesignSystem/Button"
|
|
import Body from "@/components/TempDesignSystem/Text/Body"
|
|
import Caption from "@/components/TempDesignSystem/Text/Caption"
|
|
import Footnote from "@/components/TempDesignSystem/Text/Footnote"
|
|
import Subtitle from "@/components/TempDesignSystem/Text/Subtitle"
|
|
import { formatPrice } from "@/utils/numberFormatting"
|
|
|
|
import styles from "./rateSummary.module.css"
|
|
|
|
import type { RateSummaryProps } from "@/types/components/hotelReservation/selectRate/rateSummary"
|
|
import { RoomPackageCodeEnum } from "@/types/components/hotelReservation/selectRate/roomFilter"
|
|
|
|
export default function RateSummary({
|
|
isUserLoggedIn,
|
|
packages,
|
|
roomsAvailability,
|
|
}: RateSummaryProps) {
|
|
const intl = useIntl()
|
|
const [isVisible, setIsVisible] = useState(false)
|
|
|
|
const { getSelectedRateSummary } = useRateSelectionStore()
|
|
|
|
useEffect(() => {
|
|
const timer = setTimeout(() => setIsVisible(true), 0)
|
|
return () => clearTimeout(timer)
|
|
}, [])
|
|
|
|
const selectedRateSummary = getSelectedRateSummary()
|
|
|
|
if (selectedRateSummary.length === 0) return null
|
|
|
|
const {
|
|
member,
|
|
public: publicRate,
|
|
features,
|
|
roomType,
|
|
priceName,
|
|
priceTerm,
|
|
} = selectedRateSummary[0] // TODO: Support multiple rooms
|
|
|
|
const isPetRoomSelected = features.some(
|
|
(feature) => feature.code === RoomPackageCodeEnum.PET_ROOM
|
|
)
|
|
|
|
const petRoomPackage = packages?.find(
|
|
(pkg) => pkg.code === RoomPackageCodeEnum.PET_ROOM
|
|
)
|
|
|
|
const petRoomLocalPrice =
|
|
isPetRoomSelected && petRoomPackage?.localPrice.totalPrice
|
|
? Number(petRoomPackage?.localPrice.totalPrice)
|
|
: 0
|
|
const petRoomRequestedPrice =
|
|
isPetRoomSelected && petRoomPackage?.requestedPrice.totalPrice
|
|
? Number(petRoomPackage?.requestedPrice.totalPrice)
|
|
: 0
|
|
|
|
const priceToShow = isUserLoggedIn && member ? member : publicRate
|
|
|
|
const totalPriceToShow = {
|
|
localPrice: {
|
|
currency: priceToShow.localPrice.currency,
|
|
price: priceToShow.localPrice.pricePerStay + petRoomLocalPrice,
|
|
},
|
|
requestedPrice: !priceToShow.requestedPrice
|
|
? undefined
|
|
: {
|
|
currency: priceToShow.requestedPrice.currency,
|
|
price:
|
|
priceToShow.requestedPrice.pricePerStay + petRoomRequestedPrice,
|
|
},
|
|
}
|
|
|
|
const checkInDate = new Date(roomsAvailability.checkInDate)
|
|
const checkOutDate = new Date(roomsAvailability.checkOutDate)
|
|
const nights = dt(checkOutDate).diff(dt(checkInDate), "days")
|
|
|
|
const showMemberDiscountBanner = member && !isUserLoggedIn
|
|
|
|
const summaryPriceText = `${intl.formatMessage(
|
|
{ id: "{totalNights, plural, one {# night} other {# nights}}" },
|
|
{ totalNights: nights }
|
|
)}, ${intl.formatMessage(
|
|
{ id: "{totalAdults, plural, one {# adult} other {# adults}}" },
|
|
{ totalAdults: roomsAvailability.occupancy?.adults }
|
|
)}${
|
|
roomsAvailability.occupancy?.children?.length
|
|
? `, ${intl.formatMessage(
|
|
{ id: "{totalChildren, plural, one {# child} other {# children}}" },
|
|
{ totalChildren: roomsAvailability.occupancy.children.length }
|
|
)}`
|
|
: ""
|
|
}`
|
|
|
|
return (
|
|
<div className={styles.summary} data-visible={isVisible}>
|
|
{showMemberDiscountBanner && <SignupPromoMobile />}
|
|
<div className={styles.content}>
|
|
<div className={styles.summaryText}>
|
|
<Subtitle color="uiTextHighContrast">{roomType}</Subtitle>
|
|
<Body color="uiTextMediumContrast">{`${priceName}, ${priceTerm}`}</Body>
|
|
</div>
|
|
<div className={styles.summaryPriceContainer}>
|
|
{showMemberDiscountBanner && (
|
|
<div className={styles.promoContainer}>
|
|
<SignupPromoDesktop
|
|
memberPrice={{
|
|
amount: member.localPrice.pricePerStay + petRoomLocalPrice,
|
|
currency: member.localPrice.currency,
|
|
}}
|
|
/>
|
|
</div>
|
|
)}
|
|
<div className={styles.summaryPriceTextDesktop}>
|
|
<Body>
|
|
{intl.formatMessage<React.ReactNode>(
|
|
{ id: "<b>Total price</b> (incl VAT)" },
|
|
{ b: (str) => <b>{str}</b> }
|
|
)}
|
|
</Body>
|
|
<Caption color="uiTextMediumContrast">{summaryPriceText}</Caption>
|
|
</div>
|
|
<div className={styles.summaryPrice}>
|
|
<div className={styles.summaryPriceTextDesktop}>
|
|
<Subtitle
|
|
color={isUserLoggedIn ? "red" : "uiTextHighContrast"}
|
|
textAlign="right"
|
|
>
|
|
{formatPrice(
|
|
intl,
|
|
totalPriceToShow.localPrice.price,
|
|
totalPriceToShow.localPrice.currency
|
|
)}
|
|
</Subtitle>
|
|
{totalPriceToShow?.requestedPrice ? (
|
|
<Body color="uiTextMediumContrast">
|
|
{intl.formatMessage(
|
|
{ id: "Approx. {value}" },
|
|
{
|
|
value: formatPrice(
|
|
intl,
|
|
totalPriceToShow.requestedPrice.price,
|
|
totalPriceToShow.requestedPrice.currency
|
|
),
|
|
}
|
|
)}
|
|
</Body>
|
|
) : null}
|
|
</div>
|
|
<div className={styles.summaryPriceTextMobile}>
|
|
<Caption color="uiTextHighContrast">
|
|
{intl.formatMessage({ id: "Total price" })}
|
|
</Caption>
|
|
<Subtitle color={isUserLoggedIn ? "red" : "uiTextHighContrast"}>
|
|
{formatPrice(
|
|
intl,
|
|
totalPriceToShow.localPrice.price,
|
|
totalPriceToShow.localPrice.currency
|
|
)}
|
|
</Subtitle>
|
|
<Footnote
|
|
color="uiTextMediumContrast"
|
|
className={styles.summaryPriceTextMobile}
|
|
>
|
|
{summaryPriceText}
|
|
</Footnote>
|
|
</div>
|
|
<Button
|
|
type="submit"
|
|
theme="base"
|
|
className={styles.continueButton}
|
|
>
|
|
{intl.formatMessage({ id: "Continue" })}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|