feat: add data handling to EarnAndBurn
This commit is contained in:
@@ -39,7 +39,7 @@ function DynamicComponent({ component, props }: AccountPageContentProps) {
|
|||||||
case DynamicContentComponents.expiring_points:
|
case DynamicContentComponents.expiring_points:
|
||||||
return <ExpiringPoints />
|
return <ExpiringPoints />
|
||||||
case DynamicContentComponents.earn_and_burn:
|
case DynamicContentComponents.earn_and_burn:
|
||||||
return <EarnAndBurn />
|
return <EarnAndBurn lang={props.lang} />
|
||||||
default:
|
default:
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { _ } from "@/lib/translation"
|
|||||||
|
|
||||||
import styles from "./currentPointsBalance.module.css"
|
import styles from "./currentPointsBalance.module.css"
|
||||||
|
|
||||||
const CurrentPointsBalance = () => {
|
function CurrentPointsBalance() {
|
||||||
const points = 30000
|
const points = 30000
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,44 +1,101 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import { dt } from "@/lib/dt"
|
||||||
import { _ } from "@/lib/translation"
|
import { _ } from "@/lib/translation"
|
||||||
|
import { trpc } from "@/lib/trpc/client"
|
||||||
|
|
||||||
import Button from "@/components/TempDesignSystem/Button"
|
import Button from "@/components/TempDesignSystem/Button"
|
||||||
|
|
||||||
import styles from "./earnAndBurn.module.css"
|
import styles from "./earnAndBurn.module.css"
|
||||||
|
|
||||||
const EarnAndBurn = () => {
|
import {
|
||||||
|
EarnAndBurnProps,
|
||||||
|
Page,
|
||||||
|
RowProps,
|
||||||
|
} from "@/types/components/myPages/myPage/earnAndBurn"
|
||||||
|
|
||||||
|
const tableHeadings = [
|
||||||
|
_("Arrival date"),
|
||||||
|
_("Description"),
|
||||||
|
_("Booking number"),
|
||||||
|
_("Transaction date"),
|
||||||
|
_("Points"),
|
||||||
|
]
|
||||||
|
|
||||||
|
function EarnAndBurn({ lang }: EarnAndBurnProps) {
|
||||||
|
const { data, hasNextPage, isFetching, fetchNextPage } =
|
||||||
|
trpc.user.transaction.friendTransactions.useInfiniteQuery(
|
||||||
|
{ limit: 5 },
|
||||||
|
{
|
||||||
|
getNextPageParam: (lastPage: Page) => lastPage.nextCursor,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
function loadMoreData() {
|
||||||
|
if (hasNextPage) {
|
||||||
|
fetchNextPage()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const transactions = data?.pages.flatMap((page) => page.data) ?? []
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={styles.container}>
|
<div className={styles.container}>
|
||||||
<table className={styles.table}>
|
<table className={styles.table}>
|
||||||
<thead className={styles.thead}>
|
<thead className={styles.thead}>
|
||||||
<tr>
|
<tr>
|
||||||
<th className={styles.th}>{_("Arrival date")}</th>
|
{tableHeadings.map((heading) => (
|
||||||
<th className={styles.th}>{_("Description")}</th>
|
<th key={heading} className={styles.th}>
|
||||||
<th className={styles.th}>{_("Booking number")}</th>
|
{heading}
|
||||||
<th className={styles.th}>{_("Transaction date")}</th>
|
</th>
|
||||||
<th className={styles.th}>{_("Points")}</th>
|
))}
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr className={styles.tr}>
|
{transactions.map((transaction) => (
|
||||||
<td className={styles.td}>23 May 2023</td>
|
<Row
|
||||||
<td className={styles.td}>Scandic Mölndal, Mölndal 2 nights</td>
|
lang={lang}
|
||||||
<td className={styles.td}>5123456576</td>
|
key={transaction.confirmationNumber}
|
||||||
<td className={styles.td}>26 May 2023</td>
|
transaction={transaction}
|
||||||
<td className={styles.td}>30000</td>
|
/>
|
||||||
</tr>
|
))}
|
||||||
<tr className={styles.tr}>
|
|
||||||
<td className={styles.td}>23 May 2023</td>
|
|
||||||
<td className={styles.td}>Scandic Mölndal, Mölndal 2 nights</td>
|
|
||||||
<td className={styles.td}>5123456576</td>
|
|
||||||
<td className={styles.td}>26 May 2023</td>
|
|
||||||
<td className={styles.td}>-30000</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
<Button intent="primary" bgcolor="white" type="button">
|
<Button
|
||||||
|
disabled={isFetching}
|
||||||
|
intent="primary"
|
||||||
|
bgcolor="white"
|
||||||
|
type="button"
|
||||||
|
onClick={loadMoreData}
|
||||||
|
>
|
||||||
{_("See more transactions")}
|
{_("See more transactions")}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function Row({ transaction, lang }: RowProps) {
|
||||||
|
const description = `${_(transaction.hotelName)}, ${transaction.city} ${transaction.nights} ${_("nights")}`
|
||||||
|
const arrival = dt(transaction.checkInDate).locale(lang).format("DD MMM YYYY")
|
||||||
|
const departure = dt(transaction.checkOutDate)
|
||||||
|
.locale(lang)
|
||||||
|
.format("DD MMM YYYY")
|
||||||
|
const values = [
|
||||||
|
arrival,
|
||||||
|
description,
|
||||||
|
transaction.confirmationNumber,
|
||||||
|
departure,
|
||||||
|
transaction.awardPoints,
|
||||||
|
]
|
||||||
|
return (
|
||||||
|
<tr className={styles.tr}>
|
||||||
|
{values.map((value, idx) => (
|
||||||
|
<td key={`value-${idx}`} className={styles.td}>
|
||||||
|
{value}
|
||||||
|
</td>
|
||||||
|
))}
|
||||||
|
</tr>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
export default EarnAndBurn
|
export default EarnAndBurn
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { _ } from "@/lib/translation"
|
|||||||
|
|
||||||
import styles from "./expiringPoints.module.css"
|
import styles from "./expiringPoints.module.css"
|
||||||
|
|
||||||
export const ExpiringPoints = () => {
|
export function ExpiringPoints() {
|
||||||
return (
|
return (
|
||||||
<table className={styles.table}>
|
<table className={styles.table}>
|
||||||
<thead className={styles.thead}>
|
<thead className={styles.thead}>
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ export namespace endpoints {
|
|||||||
profile = "profile/v1/Profile",
|
profile = "profile/v1/Profile",
|
||||||
upcomingStays = "booking/v1/Stays/future",
|
upcomingStays = "booking/v1/Stays/future",
|
||||||
previousStays = "booking/v1/Stays/past",
|
previousStays = "booking/v1/Stays/past",
|
||||||
|
friendTransactions = "profile/v1/Transaction/friendTransactions",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -12,3 +12,8 @@ export const soonestUpcomingStaysInput = z
|
|||||||
limit: z.number().int().positive(),
|
limit: z.number().int().positive(),
|
||||||
})
|
})
|
||||||
.default({ limit: 3 })
|
.default({ limit: 3 })
|
||||||
|
|
||||||
|
export const friendTransactionsInput = z.object({
|
||||||
|
cursor: z.number().int().nullish(),
|
||||||
|
limit: z.number().min(0).default(5),
|
||||||
|
})
|
||||||
|
|||||||
@@ -93,3 +93,72 @@ export const getStaysSchema = z.object({
|
|||||||
type GetStaysData = z.infer<typeof getStaysSchema>
|
type GetStaysData = z.infer<typeof getStaysSchema>
|
||||||
|
|
||||||
export type Stay = GetStaysData["data"][number]
|
export type Stay = GetStaysData["data"][number]
|
||||||
|
|
||||||
|
export const getFriendTransactionsSchema = z.object({
|
||||||
|
data: z.array(
|
||||||
|
z.object({
|
||||||
|
attributes: z.object({
|
||||||
|
hotelOperaId: z.string(),
|
||||||
|
confirmationNumber: z.string(),
|
||||||
|
checkInDate: z.string(),
|
||||||
|
checkOutDate: z.string(),
|
||||||
|
nights: z.number(),
|
||||||
|
awardPoints: z.number(),
|
||||||
|
pointsCalculated: z.boolean(),
|
||||||
|
hotelInformation: z.object({
|
||||||
|
hotelName: z.string(),
|
||||||
|
city: z.string(),
|
||||||
|
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(),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
relationships: z.object({
|
||||||
|
hotel: z.object({
|
||||||
|
links: z.object({
|
||||||
|
related: z.string(),
|
||||||
|
}),
|
||||||
|
data: z.object({
|
||||||
|
id: z.string(),
|
||||||
|
type: z.string(),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
booking: z.object({
|
||||||
|
links: z.object({
|
||||||
|
related: z.string(),
|
||||||
|
}),
|
||||||
|
data: z.object({
|
||||||
|
id: z.string(),
|
||||||
|
type: z.string(),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
type: z.string(),
|
||||||
|
})
|
||||||
|
),
|
||||||
|
links: z
|
||||||
|
.object({
|
||||||
|
self: z.string(),
|
||||||
|
offset: z.number(),
|
||||||
|
limit: z.number(),
|
||||||
|
totalCount: z.number(),
|
||||||
|
})
|
||||||
|
.nullable(),
|
||||||
|
})
|
||||||
|
|
||||||
|
type GetFriendTransactionsData = z.infer<typeof getFriendTransactionsSchema>
|
||||||
|
|
||||||
|
export type Transaction = GetFriendTransactionsData["data"][number]
|
||||||
|
|||||||
@@ -8,9 +8,14 @@ import {
|
|||||||
} from "@/server/errors/trpc"
|
} from "@/server/errors/trpc"
|
||||||
import { protectedProcedure, router } from "@/server/trpc"
|
import { protectedProcedure, router } from "@/server/trpc"
|
||||||
|
|
||||||
import { staysInput } from "./input"
|
import { friendTransactionsInput, staysInput } from "./input"
|
||||||
import { getStaysSchema, getUserSchema } from "./output"
|
import {
|
||||||
|
getFriendTransactionsSchema,
|
||||||
|
getStaysSchema,
|
||||||
|
getUserSchema,
|
||||||
|
} from "./output"
|
||||||
import { benefits, extendedUser, nextLevelPerks } from "./temp"
|
import { benefits, extendedUser, nextLevelPerks } from "./temp"
|
||||||
|
import friendTransactionsMockJson from "./tempFriendTransactions.json"
|
||||||
|
|
||||||
function fakingRequest<T>(payload: T): Promise<T> {
|
function fakingRequest<T>(payload: T): Promise<T> {
|
||||||
return new Promise((resolve) => {
|
return new Promise((resolve) => {
|
||||||
@@ -177,4 +182,83 @@ export const userQueryRouter = router({
|
|||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
|
transaction: router({
|
||||||
|
friendTransactions: protectedProcedure
|
||||||
|
.input(friendTransactionsInput)
|
||||||
|
.query(async (opts) => {
|
||||||
|
try {
|
||||||
|
const { limit, cursor } = opts.input
|
||||||
|
|
||||||
|
const params = new URLSearchParams()
|
||||||
|
params.set("limit", limit.toString())
|
||||||
|
|
||||||
|
if (cursor) {
|
||||||
|
params.set("offset", cursor.toString())
|
||||||
|
}
|
||||||
|
// TODO: att these once API data is confirmed to be available
|
||||||
|
// const apiResponse = await api.get(
|
||||||
|
// api.endpoints.v1.friendTransactions,
|
||||||
|
// {
|
||||||
|
// 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()
|
||||||
|
const apiJson = friendTransactionsMockJson
|
||||||
|
debugger
|
||||||
|
|
||||||
|
if (!apiJson.data?.length) {
|
||||||
|
// throw internalServerError()
|
||||||
|
}
|
||||||
|
|
||||||
|
const verifiedData = getFriendTransactionsSchema.safeParse(apiJson)
|
||||||
|
|
||||||
|
if (!verifiedData.success) {
|
||||||
|
console.info(`Get Friend Transactions - Verified Data Error`)
|
||||||
|
console.error(verifiedData.error)
|
||||||
|
throw badRequestError()
|
||||||
|
}
|
||||||
|
|
||||||
|
const nextCursor =
|
||||||
|
verifiedData.data.links &&
|
||||||
|
verifiedData.data.links.offset < verifiedData.data.links.totalCount
|
||||||
|
? verifiedData.data.links.offset
|
||||||
|
: undefined
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: verifiedData.data.data.map(({ attributes }) => ({
|
||||||
|
checkInDate: attributes.checkInDate,
|
||||||
|
checkOutDate: attributes.checkOutDate,
|
||||||
|
awardPoints: attributes.awardPoints,
|
||||||
|
hotelName: attributes.hotelInformation.hotelName,
|
||||||
|
city: attributes.hotelInformation.city,
|
||||||
|
nights: attributes.nights,
|
||||||
|
confirmationNumber: attributes.confirmationNumber,
|
||||||
|
})),
|
||||||
|
|
||||||
|
nextCursor,
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.info(`Get Friend Transactions Error`)
|
||||||
|
console.error(error)
|
||||||
|
throw internalServerError()
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
}),
|
||||||
})
|
})
|
||||||
|
|||||||
116
server/routers/user/tempFriendTransactions.json
Normal file
116
server/routers/user/tempFriendTransactions.json
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
{
|
||||||
|
"data": [
|
||||||
|
{
|
||||||
|
"attributes": {
|
||||||
|
"hotelOperaId": "216",
|
||||||
|
"confirmationNumber": "991646189",
|
||||||
|
"checkInDate": "2023-09-16",
|
||||||
|
"checkOutDate": "2023-09-17",
|
||||||
|
"transactionDate": "2023-04-18",
|
||||||
|
"nights": 1,
|
||||||
|
"awardPoints": 1863,
|
||||||
|
"pointsCalculated": true,
|
||||||
|
"hotelInformation": {
|
||||||
|
"hotelName": "Scandic Landvetter",
|
||||||
|
"city": "Stockholm",
|
||||||
|
"hotelContent": {
|
||||||
|
"images": {
|
||||||
|
"metaData": {
|
||||||
|
"title": "Lobby",
|
||||||
|
"altText": "lobby at scandic landvetter in gothenburg",
|
||||||
|
"altText_En": "lobby at scandic landvetter in gothenburg",
|
||||||
|
"copyRight": "Werner Nystrand"
|
||||||
|
},
|
||||||
|
"imageSizes": {
|
||||||
|
"tiny": "https://test3.scandichotels.com/imagevault/publishedmedia/1cz71gn106ej1mz7u4nr/Scandic-Landvetter-lobby-0013-2-vald.jpg",
|
||||||
|
"small": "https://test3.scandichotels.com/imagevault/publishedmedia/29ejr75mwp7riv63nz0x/Scandic-Landvetter-lobby-0013-2-vald.jpg",
|
||||||
|
"medium": "https://test3.scandichotels.com/imagevault/publishedmedia/bldh2liyfddkv74szp9v/Scandic-Landvetter-lobby-0013-2-vald.jpg",
|
||||||
|
"large": "https://test3.scandichotels.com/imagevault/publishedmedia/kbmpmkb714o028ufcgu4/Scandic-Landvetter-lobby-0013-2-vald.jpg"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"relationships": {
|
||||||
|
"hotel": {
|
||||||
|
"links": {
|
||||||
|
"related": "https://api-test.scandichotels.com/hotels/V0/fdea883a-8092-4604-8afb-032391a59009/hotels"
|
||||||
|
},
|
||||||
|
"data": {
|
||||||
|
"id": "d98c7ab1-ebaa-4102-b351-758daf1ddf55",
|
||||||
|
"type": "hotels"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"booking": {
|
||||||
|
"links": {
|
||||||
|
"related": "https://api-test.scandichotels.com/booking/v1/bookings/991646189"
|
||||||
|
},
|
||||||
|
"data": {
|
||||||
|
"id": "991646189",
|
||||||
|
"type": "booking"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"type": "stay"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"attributes": {
|
||||||
|
"hotelOperaId": "216",
|
||||||
|
"confirmationNumber": "991646190",
|
||||||
|
"checkInDate": "2023-09-16",
|
||||||
|
"checkOutDate": "2023-09-17",
|
||||||
|
"transactionDate": "2023-04-18",
|
||||||
|
"nights": 1,
|
||||||
|
"awardPoints": 1863,
|
||||||
|
"pointsCalculated": true,
|
||||||
|
"hotelInformation": {
|
||||||
|
"hotelName": "Scandic Landvetter",
|
||||||
|
"city": "Stockholm",
|
||||||
|
"hotelContent": {
|
||||||
|
"images": {
|
||||||
|
"metaData": {
|
||||||
|
"title": "Lobby",
|
||||||
|
"altText": "lobby at scandic landvetter in gothenburg",
|
||||||
|
"altText_En": "lobby at scandic landvetter in gothenburg",
|
||||||
|
"copyRight": "Werner Nystrand"
|
||||||
|
},
|
||||||
|
"imageSizes": {
|
||||||
|
"tiny": "https://test3.scandichotels.com/imagevault/publishedmedia/1cz71gn106ej1mz7u4nr/Scandic-Landvetter-lobby-0013-2-vald.jpg",
|
||||||
|
"small": "https://test3.scandichotels.com/imagevault/publishedmedia/29ejr75mwp7riv63nz0x/Scandic-Landvetter-lobby-0013-2-vald.jpg",
|
||||||
|
"medium": "https://test3.scandichotels.com/imagevault/publishedmedia/bldh2liyfddkv74szp9v/Scandic-Landvetter-lobby-0013-2-vald.jpg",
|
||||||
|
"large": "https://test3.scandichotels.com/imagevault/publishedmedia/kbmpmkb714o028ufcgu4/Scandic-Landvetter-lobby-0013-2-vald.jpg"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"relationships": {
|
||||||
|
"hotel": {
|
||||||
|
"links": {
|
||||||
|
"related": "https://api-test.scandichotels.com/hotels/V0/fdea883a-8092-4604-8afb-032391a59009/hotels"
|
||||||
|
},
|
||||||
|
"data": {
|
||||||
|
"id": "d98c7ab1-ebaa-4102-b351-758daf1ddf55",
|
||||||
|
"type": "hotels"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"booking": {
|
||||||
|
"links": {
|
||||||
|
"related": "https://api-test.scandichotels.com/booking/v1/bookings/991646189"
|
||||||
|
},
|
||||||
|
"data": {
|
||||||
|
"id": "991646189",
|
||||||
|
"type": "booking"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"type": "stay"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"links": {
|
||||||
|
"self": "https://api-test.scandichotels.com/profile/v1/transaction/friendTransactions?language=en&offset=1&limit=20",
|
||||||
|
"offset": 2,
|
||||||
|
"limit": 20,
|
||||||
|
"totalCount": 40
|
||||||
|
}
|
||||||
|
}
|
||||||
33
types/components/myPages/myPage/earnAndBurn.ts
Normal file
33
types/components/myPages/myPage/earnAndBurn.ts
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
import { Lang } from "@/constants/languages"
|
||||||
|
|
||||||
|
export type EarnAndBurnProps = {
|
||||||
|
lang: Lang
|
||||||
|
}
|
||||||
|
|
||||||
|
type Transaction = {
|
||||||
|
checkInDate: string
|
||||||
|
checkOutDate: string
|
||||||
|
awardPoints: number
|
||||||
|
hotelName: string
|
||||||
|
city: string
|
||||||
|
nights: number
|
||||||
|
confirmationNumber: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export type Page = {
|
||||||
|
data: Transaction[]
|
||||||
|
nextCursor?: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export type RowProps = {
|
||||||
|
lang: Lang
|
||||||
|
transaction: {
|
||||||
|
checkInDate: string
|
||||||
|
checkOutDate: string
|
||||||
|
awardPoints: number
|
||||||
|
hotelName: string
|
||||||
|
city: string
|
||||||
|
nights: number
|
||||||
|
confirmationNumber: string
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user