Merged in fix/SW-1353-multiroom-tracking (pull request #1870)

fix(SW-1353): enter details tracking fixes

* fix(SW-1353): tracking fixes

* refactor: move code

* fix: use hasPublicPrice

* fix: update calculation for roomPrice and totalPrice

* fix: only include bedtype if it is "preselected"


Approved-by: Michael Zetterberg
This commit is contained in:
Tobias Johansson
2025-04-29 06:29:57 +00:00
parent 52c99e8767
commit 70095043f8
4 changed files with 161 additions and 29 deletions

View File

@@ -96,7 +96,7 @@ export default async function DetailsPage({
hotel.merchantInformationData.alternatePaymentOptions = []
}
const { hotelsTrackingData, pageTrackingData } = getTracking(
const { hotelsTrackingData, pageTrackingData, ancillaries } = getTracking(
booking,
hotel,
rooms,
@@ -170,7 +170,11 @@ export default async function DetailsPage({
</aside>
</div>
</main>
<TrackingSDK hotelInfo={hotelsTrackingData} pageData={pageTrackingData} />
<TrackingSDK
hotelInfo={hotelsTrackingData}
pageData={pageTrackingData}
ancillaries={ancillaries}
/>
</EnterDetailsProvider>
)
}

View File

@@ -2,18 +2,25 @@ import { differenceInCalendarDays, format, isWeekend } from "date-fns"
import { REDEMPTION } from "@/constants/booking"
import { sumPackages } from "@/components/HotelReservation/utils"
import { getSpecialRoomType } from "@/utils/specialRoomType"
import { ChildBedMapEnum } from "@/types/components/bookingWidget/enums"
import { RoomPackageCodeEnum } from "@/types/components/hotelReservation/selectRate/roomFilter"
import type { SelectRateSearchParams } from "@/types/components/hotelReservation/selectRate/selectRate"
import {
TrackingChannelEnum,
type TrackingSDKAncillaries,
type TrackingSDKHotelInfo,
type TrackingSDKPageData,
} from "@/types/components/tracking"
import { CurrencyEnum } from "@/types/enums/currency"
import type { Hotel } from "@/types/hotel"
import type { Room } from "@/types/providers/details/room"
import type {
PriceProduct,
Product,
} from "@/types/trpc/routers/hotel/roomAvailability"
import type { Lang } from "@/constants/languages"
import type { SelectHotelParams } from "@/utils/url"
@@ -47,6 +54,10 @@ export function getTracking(
.join("|"),
analyticsRateCode: rooms.map((room) => room.rate).join("|"),
arrivalDate: format(arrivalDate, "yyyy-MM-dd"),
// only include bedtype here if it is "preselected" meanining there is only 1
bedType: rooms
.map((r) => (r?.bedTypes?.length === 1 ? r.bedTypes[0].description : "-"))
.join("|"),
// Comma separated booking code values in "code,code,n/a" format for multiroom and "code" or "n/a" for singleroom
// n/a is used whenever code is Not applicable as defined by Tracking team
bookingCode: rooms
@@ -72,6 +83,22 @@ export function getTracking(
.join("|"),
country: hotel?.address.country,
departureDate: format(departureDate, "yyyy-MM-dd"),
discount: rooms
.map((room, idx) => {
const isMainRoom = idx === 0
if (
hasMemberPrice(room.roomRate) &&
isMainRoom &&
isMember &&
hasPublicPrice(room.roomRate)
) {
const memberPrice = room.roomRate.member.localPrice.pricePerStay
const publicPrice = room.roomRate.public.localPrice.pricePerStay
return publicPrice - memberPrice
}
return 0
})
.join(","),
duration: differenceInCalendarDays(departureDate, arrivalDate),
hotelID: hotel?.operaId,
leadTime: differenceInCalendarDays(arrivalDate, new Date()),
@@ -80,17 +107,28 @@ export function getTracking(
.map((room) => room.childrenInRoom?.length ?? 0)
.join(","),
noOfRooms: booking.rooms.length,
...(rooms.length > 1 && {
multiroomRateIdentity: rooms.every((room) => {
const firstRoom = rooms[0]
if (
hasPublicPrice(firstRoom.roomRate) &&
hasPublicPrice(room.roomRate)
) {
return (
firstRoom.roomRate.public?.localPrice.pricePerNight ===
room.roomRate.public?.localPrice.pricePerNight
)
}
})
? "same rate"
: "mixed rate",
}),
rateCode: rooms
.map((room, idx) => {
const isMainRoom = idx === 0
if (
"member" in room.roomRate &&
room.roomRate.member &&
isMember &&
isMainRoom
) {
if (hasMemberPrice(room.roomRate) && isMember && isMainRoom) {
return room.roomRate.member.rateCode
} else if ("public" in room.roomRate && room.roomRate.public) {
} else if (hasPublicPrice(room.roomRate)) {
return room.roomRate.public.rateCode
} else if ("corporateCheque" in room.roomRate) {
return room.roomRate.corporateCheque.rateCode
@@ -108,25 +146,29 @@ export function getTracking(
rateCodeName: rooms.map((room) => room.rateDefinitionTitle).join(","),
rateCodeType: rooms.map((room) => room.rateType.toLowerCase()).join(","),
region: hotel?.address.city,
revenueCurrencyCode: rooms
.map((room) => {
if ("corporateCheque" in room.roomRate) {
return CurrencyEnum.CC
} else if ("redemption" in room.roomRate) {
return CurrencyEnum.POINTS
} else if ("voucher" in room.roomRate) {
return CurrencyEnum.Voucher
} else if ("public" in room.roomRate && room.roomRate.public) {
return room.roomRate.public.localPrice.currency
} else if ("member" in room.roomRate && room.roomRate.member) {
return room.roomRate.member.localPrice.currency
}
return CurrencyEnum.Unknown
})
.join(","),
revenueCurrencyCode: [
...new Set(
rooms.map((room) => {
if ("corporateCheque" in room.roomRate) {
return CurrencyEnum.CC
} else if ("redemption" in room.roomRate) {
return CurrencyEnum.POINTS
} else if ("voucher" in room.roomRate) {
return CurrencyEnum.Voucher
} else if (hasPublicPrice(room.roomRate)) {
return room.roomRate.public.localPrice.currency
} else if (hasMemberPrice(room.roomRate)) {
return room.roomRate.member.localPrice.currency
}
return CurrencyEnum.Unknown
})
),
].join(","),
rewardNight: booking.searchType === REDEMPTION ? "yes" : "no",
rewardNightAvailability:
booking.searchType === REDEMPTION ? "true" : "false",
roomPrice: calcTotalRoomPrice(rooms, isMember),
totalPrice: calcTotalPrice(rooms, isMember),
searchTerm: city,
searchType: "hotel",
specialRoomType: rooms
@@ -134,8 +176,91 @@ export function getTracking(
.join(","),
}
const roomsWithPetRoom = rooms.filter(hasPetRoom)
const ancillaries: TrackingSDKAncillaries = roomsWithPetRoom
.slice(0, 1) // should only be one item
.map((room) => {
const petRoomPackage = room.packages.find(
(p) => p.code === RoomPackageCodeEnum.PET_ROOM
)
return {
hotelId: hotel.operaId,
productId: petRoomPackage?.code!, // property is guaranteed at this point
productUnits: roomsWithPetRoom.length,
productPoints: 0,
productPrice: petRoomPackage?.localPrice.totalPrice ?? 0,
productType: "room preference",
productName: "pet room",
productCategory: "",
}
})
return {
hotelsTrackingData,
pageTrackingData,
ancillaries,
}
}
function hasPublicPrice(
roomRate: Product
): roomRate is PriceProduct & { public: NonNullable<PriceProduct["public"]> } {
if ("public" in roomRate && roomRate.public) {
return true
}
return false
}
function hasMemberPrice(
roomRate: Product
): roomRate is PriceProduct & { member: NonNullable<PriceProduct["member"]> } {
if ("member" in roomRate && roomRate.member) {
return true
}
return false
}
function hasPetRoom(
room: Room
): room is Room & { packages: NonNullable<Room["packages"]> } {
if (!room.packages) {
return false
}
return room.packages.some((p) => p.code === RoomPackageCodeEnum.PET_ROOM)
}
function calcTotalPrice(rooms: Room[], isMember: boolean) {
const totalRoomPrice = calcTotalRoomPrice(rooms, isMember)
const totalPackageSum = rooms.reduce((total, room) => {
const packageSum = sumPackages(room.packages)
return (total += packageSum.price ?? 0)
}, 0)
return totalRoomPrice + totalPackageSum
}
function calcTotalRoomPrice(rooms: Room[], isMember: boolean) {
return rooms.reduce((total, room, idx) => {
// When it comes special rates, only redemption has additional price and that should be added
// other special rates like voucher, corporateCheck should be added as 0 according to Priyanka
if ("redemption" in room.roomRate) {
return (
room.roomRate.redemption.requestedPrice?.additionalPricePerStay ?? 0
)
} else if (
"corporateCheck" in room.roomRate ||
"voucher" in room.roomRate
) {
return 0
}
const isMainRoom = idx === 0
if (hasMemberPrice(room.roomRate) && isMember && isMainRoom) {
total += room.roomRate.member.localPrice.pricePerStay
} else if (hasPublicPrice(room.roomRate)) {
total += room.roomRate.public.localPrice.pricePerStay
}
return total
}, 0)
}

View File

@@ -146,17 +146,19 @@ export function getTracking(
.toLowerCase(),
//rateCodeType: , //TODO: Add when available in API. "regular, promotion, corporate etx",
region: hotel?.address.city,
revenueCurrencyCode: rooms.map((r) => r.currencyCode).join(","),
revenueCurrencyCode: [...new Set(rooms.map((r) => r.currencyCode))].join(
","
),
rewardNight: booking.roomPoints > 0 ? "yes" : "no",
rewardNightAvailability: booking.roomPoints > 0 ? "true" : "false",
points: booking.roomPoints > 0 ? booking.roomPoints : undefined,
roomPrice: rooms.map((r) => r.roomPrice).join(","),
roomPrice: rooms.reduce((total, room) => total + room.roomPrice, 0),
roomTypeCode: rooms.map((r) => r.roomTypeCode ?? "-").join(","),
searchType: "hotel",
specialRoomType: rooms
.map((room) => getSpecialRoomType(room.packages))
.join(","),
totalPrice: rooms.map((r) => r.totalPrice).join(","),
totalPrice: rooms.reduce((total, room) => total + room.totalPrice, 0),
lateArrivalGuarantee: booking.rateDefinition.mustBeGuaranteed
? "mandatory"
: isFlexBooking

View File

@@ -70,11 +70,12 @@ export type TrackingSDKHotelInfo = {
childBedPreference?: string
country?: string // Country of the hotel
departureDate?: string
discount?: number
discount?: number | string
duration?: number // Number of nights to stay
hotelID?: string
leadTime?: number // Number of days from booking date until arrivalDate
lowestRoomPrice?: number
multiroomRateIdentity?: string
//modifyValues?: string // <price:<value>,roomtype:value>,bed:<value,<breakfast:value>
noOfAdults?: number | string // multiroom support, "2,1,3"
noOfChildren?: number | string // multiroom support, "2,1,3"