feat(LOY-268): Feature branch for profiling consent work * feat: Add feature branch for profile and consent work * Merged in feat/LOY-268-profile-consent-banner-comp (pull request #2908) Feat/LOY-358 profile consent banner component * feat: Add feature branch for profile and consent work * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component Approved-by: Chuma Mcphoy (We Ahead) * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * Merged in feat/profile-consent-contentstack (pull request #2921) Feat(LOY-389): Profile consent in Contentstack * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * chore(LOY-348): add profiling consent as CS entry * chore(LOY-348): add banner as dynamic content Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-347-Profile-consent-modal-phase-1 (pull request #2901) Feat(LOY-347): Profiling Consent Modal (phase 1) * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Icons * feat(LOY-347): Implement ability to open modal from banner * fix(LOY-347): minor fixes * fix(LOY-347): replace old spa icon * fix(LOY-347): re-add env vars * fix(LOY-347): File renaming and cleanup * chore(LOY-347): Update readme * fix(LOY-347): use correct space var * fix(LOY-347): Add TODO comment for adding link to accordion Approved-by: Matilda Landström * Merged in fix/LOY-386-profiling-consent-modal-contentstack (pull request #2930) Fix(LOY-386): profiling consent modal contentstack * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Icons * feat(LOY-347): Implement ability to open modal from banner * fix(LOY-347): minor fixes * fix(LOY-347): replace old spa icon * fix(LOY-347): re-add env vars * fix(LOY-347): File renaming and cleanup * fix(LOY-386): Use contentstack content for profile consent modal * fix(LOY-386): beneift cards schema transform * chore(LOY-386): remove usememo * fix(LOY-386): fix modalcontent check * fix(LOY-386): remove uneeded vars Approved-by: Matilda Landström * Merged in feat/LOY-412-profiling-consent-in-signup (pull request #2976) Feat(LOY-412): profiling consent in signup * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * Merged in feat/profile-consent-contentstack (pull request #2921) Feat(LOY-389): Profile consent in Contentstack * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * chore(LOY-348): add profiling consent as CS entry * chore(LOY-348): add banner as dynamic content Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-347-Profile-consent-modal-phase-1 (pull request #2901) Feat(LOY-347): Profiling Consent Modal (phase 1) * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Ico… * Merged in fix/lokalise-ids (pull request #3013) fix: add ids to translations in Profiling Consent * fix: add ids to translations Approved-by: Erik Tiekstra Approved-by: Chuma Mcphoy (We Ahead) * Merged in LOY-436-my-pages-profiling-consent (pull request #3011) LOY-436: Profiling Consent on My Profile, no api Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-418-profiling-consent-ui-text-update (pull request #3080) Feat/LOY-418: Profiling consent ui and text update * chore(LOY-418): update /consent buttons * chore(LOY-418): update legal texts Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-268-profiling-consent-api (pull request #3088) Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-413-Signup-API-Profiling-Consent (pull request #3105) Feat/LOY-413 Signup API Profiling Consent * feat(LOY-413): signup profiling consent * chore(LOY-413): remove todo * fix(LOY-413): only pass in profilingConsent if true * fix(LOY-413): proper spread of profilingConsent in signup input Approved-by: Christel Westerberg * Merged in fix/LOY-413-use-v2-for-signup-call (pull request #3112) fix(LOY-413): use v2 endpoint for profile POST in signup * fix(LOY-413): use v2 endpoint for profile POST in signup Approved-by: Erik Tiekstra * Merged in feat/LOY-268-profiling-consent-improvements (pull request #3094) Feat/LOY-268: Profiling consent improvements * Merged in feat/profile-consent-contentstack (pull request #2921) Feat(LOY-389): Profile consent in Contentstack * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * chore(LOY-348): add profiling consent as CS entry * chore(LOY-348): add banner as dynamic content Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-347-Profile-consent-modal-phase-1 (pull request #2901) Feat(LOY-347): Profiling Consent Modal (phase 1) * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Icons * feat(LOY-347): Implement ability to open modal from banner * fix(LOY-347): min… * Merged in fix/update-graphql (pull request #3130) fix: update graphql * fix: update graphql Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-414-prof-consent-tracking (pull request #3127) Feat/LOY-414 profile consent tracking + credit card ui update * chore(LOY-414): create track link function * chore(LOY-414): add cta tracking * chore(LOY-414): add profileConsent to userInfo datalayer * chore(LOY-414): update credit card ui * chore(LOY-414): update tracking specs * chore(LOY-414): add pageView tracking to modal Approved-by: Chuma Mcphoy (We Ahead) * fix: remove old flag * Merged in fix/LOY-268-prof-consent-button-fix (pull request #3162) fix(LOY-268): add button as link * fix(LOY-268): add button as link Approved-by: Chuma Mcphoy (We Ahead) Approved-by: Matilda Landström
316 lines
8.8 KiB
TypeScript
316 lines
8.8 KiB
TypeScript
import { z } from "zod"
|
|
|
|
import { countriesMap } from "../../constants/countries"
|
|
import { imageSchema } from "../../routers/hotels/schemas/image"
|
|
import { getFriendsMembership } from "./helpers"
|
|
|
|
const scandicFriendsTier = z.enum(["L1", "L2", "L3", "L4", "L5", "L6", "L7"])
|
|
const sasEurobonusTier = z.enum(["EBB", "EBS", "EBG", "EBD", "EBP"])
|
|
|
|
const commonMembershipSchema = z.object({
|
|
membershipNumber: z.string(),
|
|
tierExpires: z.string().nullish().default(null),
|
|
memberSince: z.string().nullish(),
|
|
})
|
|
|
|
// This prevents validation errors if the API returns an unhandled membership type
|
|
const otherMembershipSchema = z
|
|
.object({
|
|
// This ensures that `type` won't widen into "string", losing the literal types, when used in a union
|
|
type: z.string().refine((val): val is string & {} => true),
|
|
})
|
|
.merge(commonMembershipSchema)
|
|
|
|
export const sasMembershipSchema = z
|
|
.object({
|
|
type: z.literal("SAS_EB"),
|
|
tier: sasEurobonusTier,
|
|
nextTier: sasEurobonusTier.nullish(),
|
|
spendablePoints: z.number().nullish(),
|
|
boostedByScandic: z.boolean().nullish(),
|
|
boostedTier: sasEurobonusTier.nullish(),
|
|
boostedTierExpires: z.string().nullish().default(null),
|
|
})
|
|
.merge(commonMembershipSchema)
|
|
.transform((response) => {
|
|
return {
|
|
...response,
|
|
tierExpires:
|
|
// SAS API returns 1900-01-01 for non-expiring tiers
|
|
response.tierExpires === "1900-01-01" ? null : response.tierExpires,
|
|
}
|
|
})
|
|
|
|
export const friendsMembershipSchema = z
|
|
.object({
|
|
type: z.literal("SCANDIC_NATIVE"),
|
|
tier: scandicFriendsTier,
|
|
nextTier: scandicFriendsTier.nullish(),
|
|
pointsToNextTier: z.number().nullish(),
|
|
tierPoints: z.number().nullish(),
|
|
nightsToTopTier: z.number().nullish(),
|
|
})
|
|
.merge(commonMembershipSchema)
|
|
|
|
export const membershipSchema = z.union([
|
|
friendsMembershipSchema,
|
|
sasMembershipSchema,
|
|
otherMembershipSchema,
|
|
])
|
|
|
|
const pointExpirationSchema = z.object({
|
|
points: z.number().int(),
|
|
expires: z.string(),
|
|
})
|
|
|
|
export const employmentDetailsSchema = z
|
|
.object({
|
|
employeeId: z.string(),
|
|
location: z.string(),
|
|
country: z.string(),
|
|
retired: z.boolean(),
|
|
})
|
|
.optional()
|
|
|
|
export const userLoyaltySchema = z.object({
|
|
memberships: z.array(membershipSchema),
|
|
points: z.object({
|
|
spendable: z.number().int(),
|
|
earned: z.number().int().optional(),
|
|
spent: z.number().int().optional(),
|
|
}),
|
|
tier: scandicFriendsTier,
|
|
tierExpires: z.string(),
|
|
tierBoostedBy: z.string().nullish(),
|
|
pointExpirations: z.array(pointExpirationSchema).optional(),
|
|
})
|
|
|
|
export const getUserSchema = z
|
|
.object({
|
|
data: z.object({
|
|
attributes: z.object({
|
|
dateOfBirth: z.string().optional().default("1900-01-01"),
|
|
email: z.string().email(),
|
|
firstName: z.string(),
|
|
language: z
|
|
.string()
|
|
// Preserve Profile v1 formatting for now so it matches ApiLang enum
|
|
.transform((s) => s.charAt(0).toUpperCase() + s.slice(1))
|
|
.optional(),
|
|
lastName: z.string(),
|
|
phoneNumber: z.string().optional(),
|
|
profileId: z.string().optional(),
|
|
membershipNumber: z.string(),
|
|
address: z
|
|
.object({
|
|
city: z.string().optional(),
|
|
country: z.string().optional(),
|
|
countryCode: z.nativeEnum(countriesMap).optional(),
|
|
streetAddress: z.string().optional(),
|
|
zipCode: z.string().optional(),
|
|
})
|
|
.optional()
|
|
.nullable(),
|
|
loyalty: userLoyaltySchema.optional(),
|
|
employmentDetails: employmentDetailsSchema,
|
|
profilingConsent: z.boolean().optional(),
|
|
profilingConsentUpdate: z.string().optional(),
|
|
promotions: z.array(z.string()).nullish(),
|
|
}),
|
|
type: z.string(),
|
|
}),
|
|
})
|
|
.transform((apiResponse) => {
|
|
return {
|
|
...apiResponse.data.attributes,
|
|
membership: apiResponse.data.attributes.loyalty
|
|
? getFriendsMembership(apiResponse.data.attributes.loyalty)
|
|
: null,
|
|
name: `${apiResponse.data.attributes.firstName} ${apiResponse.data.attributes.lastName}`,
|
|
}
|
|
})
|
|
|
|
export const getBasicUserSchema = z.object({
|
|
dateOfBirth: z.string().optional().default("1900-01-01"),
|
|
firstName: z.string(),
|
|
language: z
|
|
.string()
|
|
.transform((s) => s.charAt(0).toUpperCase() + s.slice(1))
|
|
.optional(),
|
|
lastName: z.string(),
|
|
phoneNumber: z.string().optional(),
|
|
profileId: z.string().optional(),
|
|
membershipNumber: z.string(),
|
|
tier: scandicFriendsTier,
|
|
address: z
|
|
.object({
|
|
city: z.string().optional(),
|
|
country: z.string().optional(),
|
|
countryCode: z.nativeEnum(countriesMap).optional(),
|
|
streetAddress: z.string().optional(),
|
|
zipCode: z.string().optional(),
|
|
})
|
|
.optional()
|
|
.nullable(),
|
|
})
|
|
|
|
export const creditCardSchema = z
|
|
.object({
|
|
attribute: z.object({
|
|
cardName: z.string().optional(),
|
|
alias: z.string(),
|
|
truncatedNumber: z.string().transform((s) => s.slice(-4)),
|
|
expirationDate: z.string(),
|
|
cardType: z
|
|
.string()
|
|
.transform((s) => s.charAt(0).toLowerCase() + s.slice(1)),
|
|
}),
|
|
id: z.string(),
|
|
type: z.string(),
|
|
})
|
|
.transform((apiResponse) => {
|
|
return {
|
|
id: apiResponse.id,
|
|
type: apiResponse.attribute.cardType,
|
|
truncatedNumber: apiResponse.attribute.truncatedNumber,
|
|
alias: apiResponse.attribute.alias,
|
|
expirationDate: apiResponse.attribute.expirationDate,
|
|
cardType: apiResponse.attribute.cardType,
|
|
}
|
|
})
|
|
|
|
export const creditCardsSchema = z.object({
|
|
data: z.array(creditCardSchema),
|
|
})
|
|
|
|
// Schema is the same for upcoming and previous stays endpoints
|
|
export const getStaysSchema = z.object({
|
|
data: z.array(
|
|
z.object({
|
|
attributes: z.object({
|
|
hotelOperaId: z.string(),
|
|
hotelInformation: z.object({
|
|
hotelContent: z.object({
|
|
images: imageSchema,
|
|
}),
|
|
hotelName: z.string(),
|
|
cityName: z.string().nullable(),
|
|
}),
|
|
confirmationNumber: z.string(),
|
|
checkinDate: z.string(),
|
|
checkoutDate: z.string(),
|
|
isWebAppOrigin: z.boolean(),
|
|
bookingUrl: z.string().default(""),
|
|
}),
|
|
relationships: z.object({
|
|
hotel: z.object({
|
|
links: z.object({
|
|
related: z.string().nullable().optional(),
|
|
}),
|
|
data: z.object({
|
|
id: z.string(),
|
|
type: z.string(),
|
|
}),
|
|
}),
|
|
}),
|
|
type: z.string(),
|
|
id: z.string(),
|
|
links: z.object({
|
|
self: z.object({
|
|
href: z.string(),
|
|
meta: z.object({
|
|
method: z.string(),
|
|
}),
|
|
}),
|
|
}),
|
|
})
|
|
),
|
|
links: z
|
|
.object({
|
|
self: z.string(),
|
|
offset: z.number(),
|
|
limit: z.number(),
|
|
totalCount: z.number(),
|
|
})
|
|
.optional()
|
|
.nullable(),
|
|
})
|
|
|
|
type GetStaysData = z.infer<typeof getStaysSchema>
|
|
|
|
export type Stay = GetStaysData["data"][number]
|
|
|
|
export const getFriendTransactionsSchema = z.object({
|
|
data: z.array(
|
|
z.object({
|
|
attributes: z.object({
|
|
awardPoints: z.number().default(0),
|
|
checkinDate: z.string().default(""),
|
|
checkoutDate: z.string().default(""),
|
|
confirmationNumber: z.string().default(""),
|
|
hotelOperaId: z.string().default(""),
|
|
nights: z.number().default(1),
|
|
pointsCalculated: z.boolean().default(true),
|
|
transactionDate: z.string().default(""),
|
|
bookingUrl: z.string().default(""),
|
|
hotelInformation: z
|
|
.object({
|
|
city: z.string().default(""),
|
|
name: z.string().default(""),
|
|
hotelContent: z.object({
|
|
images: imageSchema,
|
|
}),
|
|
})
|
|
.optional(),
|
|
}),
|
|
relationships: z.object({
|
|
booking: z.object({
|
|
data: z.object({
|
|
id: z.string().default(""),
|
|
type: z.string().default(""),
|
|
}),
|
|
links: z.object({
|
|
related: z.string().default(""),
|
|
}),
|
|
}),
|
|
hotel: z
|
|
.object({
|
|
data: z.object({
|
|
id: z.string().default(""),
|
|
type: z.string().default(""),
|
|
}),
|
|
links: z.object({
|
|
related: z.string().default(""),
|
|
}),
|
|
})
|
|
.optional(),
|
|
}),
|
|
type: z.string().default(""),
|
|
})
|
|
),
|
|
links: z
|
|
.object({
|
|
self: z.string(),
|
|
})
|
|
.nullable(),
|
|
})
|
|
|
|
type GetFriendTransactionsData = z.infer<typeof getFriendTransactionsSchema>
|
|
|
|
export type FriendTransaction = GetFriendTransactionsData["data"][number]
|
|
|
|
export const initiateSaveCardSchema = z.object({
|
|
data: z.object({
|
|
attribute: z.object({
|
|
transactionId: z.string(),
|
|
link: z.string(),
|
|
mobileToken: z.string().optional(),
|
|
}),
|
|
type: z.string(),
|
|
}),
|
|
})
|
|
|
|
export const subscriberIdSchema = z.object({
|
|
subscriberId: z.string(),
|
|
})
|