feat: enhance stays with api data
This commit is contained in:
@@ -4,7 +4,7 @@ import Title from "@/components/Title"
|
|||||||
|
|
||||||
import styles from "./header.module.css"
|
import styles from "./header.module.css"
|
||||||
|
|
||||||
import type { HeaderProps } from "@/types/components/myPages/myStays/title"
|
import type { HeaderProps } from "@/types/components/myPages/stays/title"
|
||||||
|
|
||||||
export default function Header({ title, subtitle, link }: HeaderProps) {
|
export default function Header({ title, subtitle, link }: HeaderProps) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import StayList from "../StayList"
|
|||||||
import EmptyPreviousStaysBlock from "./EmptyPreviousStays"
|
import EmptyPreviousStaysBlock from "./EmptyPreviousStays"
|
||||||
|
|
||||||
import { AccountPageComponentProps } from "@/types/components/myPages/myPage/accountPage"
|
import { AccountPageComponentProps } from "@/types/components/myPages/myPage/accountPage"
|
||||||
import type { Page } from "@/types/components/myPages/myStays/page"
|
import type { Page } from "@/types/components/myPages/stays/page"
|
||||||
|
|
||||||
export default function PreviousStays({
|
export default function PreviousStays({
|
||||||
lang,
|
lang,
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import Button from "@/components/TempDesignSystem/Button"
|
|||||||
|
|
||||||
import styles from "./button.module.css"
|
import styles from "./button.module.css"
|
||||||
|
|
||||||
import type { ShowMoreButtonParams } from "@/types/components/myPages/myStays/button"
|
import type { ShowMoreButtonParams } from "@/types/components/myPages/stays/button"
|
||||||
|
|
||||||
export default function ShowMoreButton({
|
export default function ShowMoreButton({
|
||||||
disabled,
|
disabled,
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ export default async function SoonestStays({
|
|||||||
subtitle,
|
subtitle,
|
||||||
link,
|
link,
|
||||||
}: AccountPageComponentProps) {
|
}: AccountPageComponentProps) {
|
||||||
const stays = await serverClient().user.stays.soonestUpcoming()
|
const { data: stays } = await serverClient().user.stays.upcoming({ limit: 3 })
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<MaxWidth className={styles.container} tag="section">
|
<MaxWidth className={styles.container} tag="section">
|
||||||
|
|||||||
@@ -5,15 +5,15 @@ import Title from "@/components/Title"
|
|||||||
|
|
||||||
import styles from "./stay.module.css"
|
import styles from "./stay.module.css"
|
||||||
|
|
||||||
import type { StayCardProps } from "@/types/components/myPages/myStays/stayCard"
|
import type { StayCardProps } from "@/types/components/myPages/stays/stayCard"
|
||||||
|
|
||||||
export default function StayCard({ stay, lang }: StayCardProps) {
|
export default function StayCard({ stay, lang }: StayCardProps) {
|
||||||
const { dateArrive, dateDepart, guests, hotel } = stay
|
const { checkinDate, checkoutDate, hotelInformation } = stay.attributes
|
||||||
|
|
||||||
const arrival = dt(dateArrive).locale(lang)
|
const arrival = dt(checkinDate).locale(lang)
|
||||||
const arrivalDate = arrival.format("DD MMM")
|
const arrivalDate = arrival.format("DD MMM")
|
||||||
const arrivalDateTime = arrival.format("YYYY-MM-DD")
|
const arrivalDateTime = arrival.format("YYYY-MM-DD")
|
||||||
const depart = dt(dateDepart).locale(lang)
|
const depart = dt(checkoutDate).locale(lang)
|
||||||
const departDate = depart.format("DD MMM YYYY")
|
const departDate = depart.format("DD MMM YYYY")
|
||||||
const departDateTime = depart.format("YYYY-MM-DD")
|
const departDateTime = depart.format("YYYY-MM-DD")
|
||||||
|
|
||||||
@@ -27,7 +27,7 @@ export default function StayCard({ stay, lang }: StayCardProps) {
|
|||||||
uppercase
|
uppercase
|
||||||
className={styles.hotel}
|
className={styles.hotel}
|
||||||
>
|
>
|
||||||
{hotel}
|
{hotelInformation.hotelName}
|
||||||
</Title>
|
</Title>
|
||||||
<section className={styles.container}>
|
<section className={styles.container}>
|
||||||
<div className={styles.date}>
|
<div className={styles.date}>
|
||||||
@@ -43,17 +43,6 @@ export default function StayCard({ stay, lang }: StayCardProps) {
|
|||||||
<time dateTime={departDateTime}>{departDate}</time>
|
<time dateTime={departDateTime}>{departDate}</time>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div className={styles.guests}>
|
|
||||||
<Image
|
|
||||||
alt="Guests Icon"
|
|
||||||
height={20}
|
|
||||||
src="/_static/icons/person.svg"
|
|
||||||
width={20}
|
|
||||||
/>
|
|
||||||
<span>
|
|
||||||
{guests} guest{guests > 1 ? "s" : ""}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</section>
|
</section>
|
||||||
</footer>
|
</footer>
|
||||||
</article>
|
</article>
|
||||||
|
|||||||
@@ -2,13 +2,17 @@ import StayCard from "../StayCard"
|
|||||||
|
|
||||||
import styles from "./stayList.module.css"
|
import styles from "./stayList.module.css"
|
||||||
|
|
||||||
import { StayListProps } from "@/types/components/myPages/myStays/stayList"
|
import { StayListProps } from "@/types/components/myPages/stays/stayList"
|
||||||
|
|
||||||
export default function StayList({ lang, stays }: StayListProps) {
|
export default function StayList({ lang, stays }: StayListProps) {
|
||||||
return (
|
return (
|
||||||
<section className={styles.stays}>
|
<section className={styles.stays}>
|
||||||
{stays.map((stay) => (
|
{stays.map((stay) => (
|
||||||
<StayCard key={stay.uid} stay={stay} lang={lang} />
|
<StayCard
|
||||||
|
key={stay.attributes.confirmationNumber}
|
||||||
|
stay={stay}
|
||||||
|
lang={lang}
|
||||||
|
/>
|
||||||
))}
|
))}
|
||||||
</section>
|
</section>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
"use client"
|
"use client"
|
||||||
|
|
||||||
import { Lang } from "@/constants/languages"
|
|
||||||
import { _ } from "@/lib/translation"
|
import { _ } from "@/lib/translation"
|
||||||
import { trpc } from "@/lib/trpc/client"
|
import { trpc } from "@/lib/trpc/client"
|
||||||
|
|
||||||
@@ -12,7 +11,7 @@ import StayList from "../StayList"
|
|||||||
import EmptyUpcomingStaysBlock from "./EmptyUpcomingStays"
|
import EmptyUpcomingStaysBlock from "./EmptyUpcomingStays"
|
||||||
|
|
||||||
import { AccountPageComponentProps } from "@/types/components/myPages/myPage/accountPage"
|
import { AccountPageComponentProps } from "@/types/components/myPages/myPage/accountPage"
|
||||||
import type { Page } from "@/types/components/myPages/myStays/page"
|
import type { Page } from "@/types/components/myPages/stays/page"
|
||||||
|
|
||||||
export default function UpcomingStays({
|
export default function UpcomingStays({
|
||||||
lang,
|
lang,
|
||||||
@@ -22,14 +21,16 @@ export default function UpcomingStays({
|
|||||||
}: AccountPageComponentProps) {
|
}: AccountPageComponentProps) {
|
||||||
const { data, hasNextPage, isFetching, fetchNextPage } =
|
const { data, hasNextPage, isFetching, fetchNextPage } =
|
||||||
trpc.user.stays.upcoming.useInfiniteQuery(
|
trpc.user.stays.upcoming.useInfiniteQuery(
|
||||||
{},
|
{ limit: 6 },
|
||||||
{
|
{
|
||||||
getNextPageParam: (lastPage: Page) => lastPage.nextCursor,
|
getNextPageParam: (lastPage: Page) => lastPage.nextCursor,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
function loadMoreData() {
|
function loadMoreData() {
|
||||||
fetchNextPage()
|
if (hasNextPage) {
|
||||||
|
fetchNextPage()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -5,6 +5,10 @@ export namespace endpoints {
|
|||||||
export const enum v0 {
|
export const enum v0 {
|
||||||
profile = "profile/v0/Profile",
|
profile = "profile/v0/Profile",
|
||||||
}
|
}
|
||||||
|
export const enum v1 {
|
||||||
|
upcomingStays = "booking/v1/Stays/future",
|
||||||
|
previousStays = "booking/v1/Stays/past",
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Endpoint = endpoints.v0
|
export type Endpoint = endpoints.v0 | endpoints.v1
|
||||||
|
|||||||
@@ -20,10 +20,14 @@ const defaultOptions: RequestInit = {
|
|||||||
|
|
||||||
export async function get(
|
export async function get(
|
||||||
endpoint: Endpoint,
|
endpoint: Endpoint,
|
||||||
options: RequestOptionsWithOutBody
|
options: RequestOptionsWithOutBody,
|
||||||
|
params?: URLSearchParams
|
||||||
) {
|
) {
|
||||||
defaultOptions.method = "GET"
|
defaultOptions.method = "GET"
|
||||||
return fetch(`${env.API_BASEURL}/${endpoint}`, merge(defaultOptions, options))
|
const url = new URL(
|
||||||
|
`${env.API_BASEURL}/${endpoint}${params ? `?${params.toString()}` : ""}`
|
||||||
|
)
|
||||||
|
return fetch(url, merge(defaultOptions, options))
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function patch(
|
export async function patch(
|
||||||
|
|||||||
@@ -2,15 +2,13 @@ import { z } from "zod"
|
|||||||
|
|
||||||
export const staysInput = z
|
export const staysInput = z
|
||||||
.object({
|
.object({
|
||||||
perPage: z.number().min(0).default(6),
|
limit: z.number().min(0).default(6),
|
||||||
page: z.number().min(0).default(0),
|
|
||||||
cursor: z.number().nullish(),
|
cursor: z.number().nullish(),
|
||||||
})
|
})
|
||||||
.default({})
|
.default({})
|
||||||
|
|
||||||
|
export const soonestUpcomingStaysInput = z
|
||||||
export const soonestUpcomingStaysInput = z
|
.object({
|
||||||
.object({
|
limit: z.number().int().positive(),
|
||||||
limit: z.number().int().positive(),
|
})
|
||||||
})
|
.default({ limit: 3 })
|
||||||
.default({ limit: 3 })
|
|
||||||
|
|||||||
@@ -23,3 +23,69 @@ export const getUserSchema = z.object({
|
|||||||
phoneNumber: z.string(),
|
phoneNumber: z.string(),
|
||||||
profileId: z.string(),
|
profileId: z.string(),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// 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: z.object({
|
||||||
|
metaData: z.object({
|
||||||
|
title: z.string(),
|
||||||
|
altText: z.string(),
|
||||||
|
altText_En: z.string(),
|
||||||
|
copyRight: z.string(),
|
||||||
|
}),
|
||||||
|
imageSizes: z.object({
|
||||||
|
tiny: z.string(),
|
||||||
|
small: z.string(),
|
||||||
|
medium: z.string(),
|
||||||
|
large: z.string(),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
hotelName: z.string(),
|
||||||
|
cityName: z.string(),
|
||||||
|
}),
|
||||||
|
confirmationNumber: z.string(),
|
||||||
|
checkinDate: z.string(),
|
||||||
|
checkoutDate: z.string(),
|
||||||
|
isWebAppOrigin: z.boolean(),
|
||||||
|
}),
|
||||||
|
relationships: z.object({
|
||||||
|
hotel: z.object({
|
||||||
|
links: z.object({
|
||||||
|
related: z.string(),
|
||||||
|
}),
|
||||||
|
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(),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
|
||||||
|
type GetStaysData = z.infer<typeof getStaysSchema>
|
||||||
|
|
||||||
|
export type Stay = GetStaysData["data"][number]
|
||||||
|
|||||||
@@ -9,15 +9,9 @@ import {
|
|||||||
} from "@/server/errors/trpc"
|
} from "@/server/errors/trpc"
|
||||||
import { protectedProcedure, router } from "@/server/trpc"
|
import { protectedProcedure, router } from "@/server/trpc"
|
||||||
|
|
||||||
import { soonestUpcomingStaysInput, staysInput } from "./input"
|
import { staysInput } from "./input"
|
||||||
import { getUserSchema } from "./output"
|
import { getStaysSchema, getUserSchema } from "./output"
|
||||||
import {
|
import { benefits, extendedUser, nextLevelPerks } from "./temp"
|
||||||
benefits,
|
|
||||||
extendedUser,
|
|
||||||
nextLevelPerks,
|
|
||||||
previousStays,
|
|
||||||
upcomingStays,
|
|
||||||
} from "./temp"
|
|
||||||
|
|
||||||
function fakingRequest<T>(payload: T): Promise<T> {
|
function fakingRequest<T>(payload: T): Promise<T> {
|
||||||
return new Promise((resolve) => {
|
return new Promise((resolve) => {
|
||||||
@@ -84,66 +78,130 @@ export const userQueryRouter = router({
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
stays: router({
|
stays: router({
|
||||||
soonestUpcoming: protectedProcedure
|
|
||||||
.input(soonestUpcomingStaysInput)
|
|
||||||
.query(async ({ input }) => {
|
|
||||||
return upcomingStays.slice(0, input.limit)
|
|
||||||
}),
|
|
||||||
|
|
||||||
previous: protectedProcedure.input(staysInput).query(async (opts) => {
|
previous: protectedProcedure.input(staysInput).query(async (opts) => {
|
||||||
const { perPage, page, cursor } = opts.input
|
try {
|
||||||
let nextCursor: typeof cursor | undefined = undefined
|
const { limit: perPage, cursor } = opts.input
|
||||||
const nrPages = Math.ceil(previousStays.length / perPage)
|
|
||||||
|
|
||||||
let stays, nextPage
|
const params = new URLSearchParams()
|
||||||
if (cursor) {
|
params.set("limit", perPage.toString())
|
||||||
stays = previousStays.slice(cursor, perPage + cursor + 1)
|
|
||||||
nextPage = cursor / perPage + 1
|
|
||||||
} else {
|
|
||||||
stays = previousStays.slice(
|
|
||||||
page * perPage,
|
|
||||||
page * perPage + perPage + 1
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
if (cursor) {
|
||||||
(nextPage && nextPage < nrPages && stays.length == perPage + 1) ||
|
params.set("offset", cursor.toString())
|
||||||
(!nextPage && nrPages > 1)
|
|
||||||
) {
|
|
||||||
const nextItem = stays.pop()
|
|
||||||
if (nextItem) {
|
|
||||||
nextCursor = previousStays.indexOf(nextItem)
|
|
||||||
}
|
}
|
||||||
} // TODO: Make request to get user data from Scandic API
|
|
||||||
return { data: stays, nextCursor }
|
const apiResponse = await api.get(
|
||||||
|
api.endpoints.v1.previousStays,
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${opts.ctx.session.token.access_token}`,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params
|
||||||
|
)
|
||||||
|
|
||||||
|
if (!apiResponse.ok) {
|
||||||
|
switch (apiResponse.status) {
|
||||||
|
case 400:
|
||||||
|
throw badRequestError()
|
||||||
|
case 401:
|
||||||
|
throw unauthorizedError()
|
||||||
|
case 403:
|
||||||
|
throw forbiddenError()
|
||||||
|
default:
|
||||||
|
throw internalServerError()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const apiJson = await apiResponse.json()
|
||||||
|
if (!apiJson.data?.length) {
|
||||||
|
throw internalServerError()
|
||||||
|
}
|
||||||
|
|
||||||
|
const verifiedData = getStaysSchema.safeParse(apiJson)
|
||||||
|
|
||||||
|
if (!verifiedData.success) {
|
||||||
|
console.info(`Get Previous Stays - Verified Data Error`)
|
||||||
|
console.error(verifiedData.error)
|
||||||
|
throw badRequestError()
|
||||||
|
}
|
||||||
|
|
||||||
|
const nextCursor =
|
||||||
|
verifiedData.data.links.offset < verifiedData.data.links.totalCount
|
||||||
|
? verifiedData.data.links.offset
|
||||||
|
: undefined
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: verifiedData.data.data,
|
||||||
|
nextCursor,
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.info(`Get Previous Stays Error`)
|
||||||
|
console.error(error)
|
||||||
|
throw internalServerError()
|
||||||
|
}
|
||||||
}),
|
}),
|
||||||
|
|
||||||
upcoming: protectedProcedure.input(staysInput).query(async (opts) => {
|
upcoming: protectedProcedure.input(staysInput).query(async (opts) => {
|
||||||
const { perPage, page, cursor } = opts.input
|
try {
|
||||||
let nextCursor: typeof cursor | undefined = undefined
|
const { limit: perPage, cursor } = opts.input
|
||||||
const nrPages = Math.ceil(upcomingStays.length / perPage)
|
|
||||||
|
|
||||||
let stays, nextPage
|
const params = new URLSearchParams()
|
||||||
if (cursor) {
|
params.set("limit", perPage.toString())
|
||||||
stays = upcomingStays.slice(cursor, perPage + cursor + 1)
|
|
||||||
nextPage = cursor / perPage + 1
|
|
||||||
} else {
|
|
||||||
stays = upcomingStays.slice(
|
|
||||||
page * perPage,
|
|
||||||
page * perPage + perPage + 1
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
if (cursor) {
|
||||||
(nextPage && nextPage < nrPages && stays.length == perPage + 1) ||
|
params.set("offset", cursor.toString())
|
||||||
(!nextPage && nrPages > 1)
|
|
||||||
) {
|
|
||||||
const nextItem = stays.pop()
|
|
||||||
if (nextItem) {
|
|
||||||
nextCursor = upcomingStays.indexOf(nextItem)
|
|
||||||
}
|
}
|
||||||
} // TODO: Make request to get user data from Scandic API
|
|
||||||
return { data: stays, nextCursor }
|
const apiResponse = await api.get(
|
||||||
|
api.endpoints.v1.upcomingStays,
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${opts.ctx.session.token.access_token}`,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
params
|
||||||
|
)
|
||||||
|
|
||||||
|
if (!apiResponse.ok) {
|
||||||
|
switch (apiResponse.status) {
|
||||||
|
case 400:
|
||||||
|
throw badRequestError()
|
||||||
|
case 401:
|
||||||
|
throw unauthorizedError()
|
||||||
|
case 403:
|
||||||
|
throw forbiddenError()
|
||||||
|
default:
|
||||||
|
throw internalServerError()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const apiJson = await apiResponse.json()
|
||||||
|
if (!apiJson.data?.length) {
|
||||||
|
throw internalServerError()
|
||||||
|
}
|
||||||
|
|
||||||
|
const verifiedData = getStaysSchema.safeParse(apiJson)
|
||||||
|
|
||||||
|
if (!verifiedData.success) {
|
||||||
|
console.info(`Get Upcoming Stays - Verified Data Error`)
|
||||||
|
console.error(verifiedData.error)
|
||||||
|
throw badRequestError()
|
||||||
|
}
|
||||||
|
|
||||||
|
const nextCursor =
|
||||||
|
verifiedData.data.links.offset < verifiedData.data.links.totalCount
|
||||||
|
? verifiedData.data.links.offset
|
||||||
|
: undefined
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: verifiedData.data.data,
|
||||||
|
nextCursor,
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.info(`Get Upcoming Stays Error`)
|
||||||
|
console.error(error)
|
||||||
|
throw internalServerError()
|
||||||
|
}
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,9 +0,0 @@
|
|||||||
import { UUID } from "crypto"
|
|
||||||
|
|
||||||
export type Stay = {
|
|
||||||
uid: UUID
|
|
||||||
dateArrive: Date
|
|
||||||
dateDepart: Date
|
|
||||||
guests: number
|
|
||||||
hotel: string
|
|
||||||
}
|
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { Stay } from "../myPage/stay"
|
import { Stay } from "@/server/routers/user/output"
|
||||||
|
|
||||||
export type Page = {
|
export type Page = {
|
||||||
data: Stay[]
|
data: Stay[]
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { Lang } from "@/constants/languages"
|
import type { Lang } from "@/constants/languages"
|
||||||
import type { Stay } from "../myPage/stay"
|
import type { Stay } from "@/server/routers/user/output"
|
||||||
|
|
||||||
export type StayCardProps = {
|
export type StayCardProps = {
|
||||||
lang: Lang
|
lang: Lang
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { Lang } from "@/constants/languages"
|
import type { Lang } from "@/constants/languages"
|
||||||
import type { Stay } from "../myPage/stay"
|
import type { Stay } from "@/server/routers/user/output"
|
||||||
|
|
||||||
export type StayListProps = {
|
export type StayListProps = {
|
||||||
stays: Stay[]
|
stays: Stay[]
|
||||||
Reference in New Issue
Block a user