Merged in feat/SW-1443-carousel-destination-overview-page (pull request #1289)
feat(SW-1443): added carousel block on hotel overview page * feat(SW-1443): added carousel block on hotel overview page Approved-by: Fredrik Thorsson Approved-by: Matilda Landström Approved-by: Chuma Mcphoy (We Ahead)
This commit is contained in:
@@ -17,9 +17,9 @@ export default function CarouselCards({ carousel_cards }: CarouselCardsProps) {
|
|||||||
const {
|
const {
|
||||||
heading,
|
heading,
|
||||||
enableFilters,
|
enableFilters,
|
||||||
|
defaultFilter,
|
||||||
filterCategories,
|
filterCategories,
|
||||||
cards,
|
cards,
|
||||||
defaultFilter,
|
|
||||||
link,
|
link,
|
||||||
} = carousel_cards
|
} = carousel_cards
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,11 @@
|
|||||||
.container {
|
.container {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: var(--Spacing-x4);
|
gap: var(--Spacing-x4);
|
||||||
padding: var(--Spacing-x5) var(--Spacing-x2) var(--Spacing-x7);
|
padding: var(--Spacing-x5) 0 var(--Spacing-x7);
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (min-width: 768px) {
|
@media screen and (min-width: 768px) {
|
||||||
.container {
|
.container {
|
||||||
gap: var(--Spacing-x7);
|
gap: var(--Spacing-x7);
|
||||||
padding: var(--Spacing-x5) 9.625rem var(--Spacing-x9) 9.625rem;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,28 @@
|
|||||||
.pageContainer {
|
.mapContainer {
|
||||||
position: relative;
|
position: relative;
|
||||||
display: grid;
|
display: grid;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: var(--max-width);
|
max-width: var(--max-width);
|
||||||
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (min-width: 768px) {
|
.main {
|
||||||
.pageContainer {
|
display: grid;
|
||||||
margin: 0 auto;
|
gap: var(--Spacing-x9);
|
||||||
}
|
padding: var(--Spacing-x7) 0 var(--Spacing-x9);
|
||||||
|
background-color: var(--Base-Surface-Primary-light-Normal);
|
||||||
|
}
|
||||||
|
|
||||||
|
.blocks {
|
||||||
|
display: grid;
|
||||||
|
gap: var(--Spacing-x9);
|
||||||
|
width: 100%;
|
||||||
|
max-width: var(--max-width-content);
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hotelsAccordions {
|
||||||
|
width: 100%;
|
||||||
|
max-width: var(--max-width-content);
|
||||||
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import {
|
|||||||
getDestinationsList,
|
getDestinationsList,
|
||||||
} from "@/lib/trpc/memoizedRequests"
|
} from "@/lib/trpc/memoizedRequests"
|
||||||
|
|
||||||
|
import Blocks from "@/components/Blocks"
|
||||||
import TrackingSDK from "@/components/TrackingSDK"
|
import TrackingSDK from "@/components/TrackingSDK"
|
||||||
|
|
||||||
import HotelsSection from "./HotelsSection"
|
import HotelsSection from "./HotelsSection"
|
||||||
@@ -30,12 +31,21 @@ export default async function DestinationOverviewPage() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={styles.pageContainer}>
|
{googleMapsApiKey ? (
|
||||||
{googleMapsApiKey ? (
|
<div className={styles.mapContainer}>
|
||||||
<OverviewMapContainer apiKey={googleMapsApiKey} mapId={googleMapId} />
|
<OverviewMapContainer apiKey={googleMapsApiKey} mapId={googleMapId} />
|
||||||
) : null}
|
</div>
|
||||||
{destinationsData && <HotelsSection destinations={destinationsData} />}
|
) : null}
|
||||||
</div>
|
<main className={styles.main}>
|
||||||
|
<div className={styles.blocks}>
|
||||||
|
<Blocks blocks={destinationOverviewPage.blocks} />
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
{destinationsData && (
|
||||||
|
<aside className={styles.hotelsAccordions}>
|
||||||
|
<HotelsSection destinations={destinationsData} />
|
||||||
|
</aside>
|
||||||
|
)}
|
||||||
<Suspense fallback={null}>
|
<Suspense fallback={null}>
|
||||||
<TrackingSDK pageData={tracking} />
|
<TrackingSDK pageData={tracking} />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
|
|||||||
@@ -24,10 +24,8 @@ fragment CarouselCards_StartPage on StartPageBlocksCarouselCards {
|
|||||||
heading
|
heading
|
||||||
enable_filters
|
enable_filters
|
||||||
card_groups {
|
card_groups {
|
||||||
filter_category {
|
filter_label
|
||||||
filter_identifier
|
filter_identifier
|
||||||
filter_label
|
|
||||||
}
|
|
||||||
cardConnection {
|
cardConnection {
|
||||||
edges {
|
edges {
|
||||||
node {
|
node {
|
||||||
@@ -93,3 +91,76 @@ fragment CarouselCards_StartPageRefs on StartPageBlocksCarouselCards {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fragment CarouselCards_DestinationOverviewPage on DestinationOverviewPageBlocksCarouselCards {
|
||||||
|
carousel_cards {
|
||||||
|
heading
|
||||||
|
enable_filters
|
||||||
|
card_groups {
|
||||||
|
filter_identifier
|
||||||
|
filter_label
|
||||||
|
cardConnection {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
...ContentCardBlock
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
link {
|
||||||
|
cta_text
|
||||||
|
is_contentstack_link
|
||||||
|
open_in_new_tab
|
||||||
|
external_link {
|
||||||
|
href
|
||||||
|
title
|
||||||
|
}
|
||||||
|
linkConnection {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
__typename
|
||||||
|
...AccountPageLink
|
||||||
|
...CollectionPageLink
|
||||||
|
...ContentPageLink
|
||||||
|
...DestinationCityPageLink
|
||||||
|
...DestinationCountryPageLink
|
||||||
|
...DestinationOverviewPageLink
|
||||||
|
...HotelPageLink
|
||||||
|
...LoyaltyPageLink
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fragment CarouselCards_DestinationOverviewPageRefs on DestinationOverviewPageBlocksCarouselCards {
|
||||||
|
carousel_cards {
|
||||||
|
card_groups {
|
||||||
|
cardConnection {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
...ContentCardBlockRef
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
link {
|
||||||
|
linkConnection {
|
||||||
|
edges {
|
||||||
|
node {
|
||||||
|
__typename
|
||||||
|
...AccountPageRef
|
||||||
|
...CollectionPageRef
|
||||||
|
...ContentPageRef
|
||||||
|
...DestinationCityPageRef
|
||||||
|
...DestinationCountryPageRef
|
||||||
|
...DestinationOverviewPageRef
|
||||||
|
...HotelPageRef
|
||||||
|
...LoyaltyPageRef
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,9 +1,15 @@
|
|||||||
#import "../../Fragments/System.graphql"
|
#import "../../Fragments/System.graphql"
|
||||||
|
|
||||||
|
#import "../../Fragments/Blocks/CarouselCards.graphql"
|
||||||
|
|
||||||
query GetDestinationOverviewPage($locale: String!, $uid: String!) {
|
query GetDestinationOverviewPage($locale: String!, $uid: String!) {
|
||||||
destination_overview_page(uid: $uid, locale: $locale) {
|
destination_overview_page(uid: $uid, locale: $locale) {
|
||||||
title
|
title
|
||||||
url
|
url
|
||||||
|
blocks {
|
||||||
|
__typename
|
||||||
|
...CarouselCards_DestinationOverviewPage
|
||||||
|
}
|
||||||
system {
|
system {
|
||||||
...System
|
...System
|
||||||
created_at
|
created_at
|
||||||
@@ -17,6 +23,10 @@ query GetDestinationOverviewPage($locale: String!, $uid: String!) {
|
|||||||
|
|
||||||
query GetDestinationOverviewPageRefs($locale: String!, $uid: String!) {
|
query GetDestinationOverviewPageRefs($locale: String!, $uid: String!) {
|
||||||
destination_overview_page(locale: $locale, uid: $uid) {
|
destination_overview_page(locale: $locale, uid: $uid) {
|
||||||
|
blocks {
|
||||||
|
__typename
|
||||||
|
...CarouselCards_DestinationOverviewPageRefs
|
||||||
|
}
|
||||||
system {
|
system {
|
||||||
...System
|
...System
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,33 @@
|
|||||||
import { z } from "zod"
|
import { z } from "zod"
|
||||||
|
|
||||||
|
import { discriminatedUnionArray } from "@/lib/discriminatedUnion"
|
||||||
|
|
||||||
import { removeMultipleSlashes } from "@/utils/url"
|
import { removeMultipleSlashes } from "@/utils/url"
|
||||||
|
|
||||||
|
import {
|
||||||
|
carouselCardsRefsSchema,
|
||||||
|
carouselCardsSchema,
|
||||||
|
} from "../schemas/blocks/carouselCards"
|
||||||
import { systemSchema } from "../schemas/system"
|
import { systemSchema } from "../schemas/system"
|
||||||
|
|
||||||
|
import { DestinationOverviewPageEnum } from "@/types/enums/destinationOverviewPage"
|
||||||
|
|
||||||
|
const destinationOverviewPageCarouselCards = z
|
||||||
|
.object({
|
||||||
|
__typename: z.literal(
|
||||||
|
DestinationOverviewPageEnum.ContentStack.blocks.CarouselCards
|
||||||
|
),
|
||||||
|
})
|
||||||
|
.merge(carouselCardsSchema)
|
||||||
|
|
||||||
|
export const blocksSchema = z.discriminatedUnion("__typename", [
|
||||||
|
destinationOverviewPageCarouselCards,
|
||||||
|
])
|
||||||
|
|
||||||
export const destinationOverviewPageSchema = z.object({
|
export const destinationOverviewPageSchema = z.object({
|
||||||
destination_overview_page: z.object({
|
destination_overview_page: z.object({
|
||||||
title: z.string(),
|
title: z.string(),
|
||||||
|
blocks: discriminatedUnionArray(blocksSchema.options),
|
||||||
system: systemSchema.merge(
|
system: systemSchema.merge(
|
||||||
z.object({
|
z.object({
|
||||||
created_at: z.string(),
|
created_at: z.string(),
|
||||||
@@ -41,8 +62,21 @@ export const countryPageUrlSchema = z
|
|||||||
)
|
)
|
||||||
|
|
||||||
/** REFS */
|
/** REFS */
|
||||||
|
const destinationOverviewPageCarouselCardsRef = z
|
||||||
|
.object({
|
||||||
|
__typename: z.literal(
|
||||||
|
DestinationOverviewPageEnum.ContentStack.blocks.CarouselCards
|
||||||
|
),
|
||||||
|
})
|
||||||
|
.merge(carouselCardsRefsSchema)
|
||||||
|
|
||||||
|
const blocksRefsSchema = z.discriminatedUnion("__typename", [
|
||||||
|
destinationOverviewPageCarouselCardsRef,
|
||||||
|
])
|
||||||
|
|
||||||
export const destinationOverviewPageRefsSchema = z.object({
|
export const destinationOverviewPageRefsSchema = z.object({
|
||||||
destination_overview_page: z.object({
|
destination_overview_page: z.object({
|
||||||
|
blocks: discriminatedUnionArray(blocksRefsSchema.options).nullable(),
|
||||||
system: systemSchema,
|
system: systemSchema,
|
||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -24,10 +24,8 @@ const carouselCardsWithFilters = z.object({
|
|||||||
enable_filters: z.literal(true),
|
enable_filters: z.literal(true),
|
||||||
card_groups: z.array(
|
card_groups: z.array(
|
||||||
z.object({
|
z.object({
|
||||||
filter_category: z.object({
|
filter_identifier: z.nativeEnum(CarouselCardFilterEnum),
|
||||||
filter_identifier: z.nativeEnum(CarouselCardFilterEnum),
|
filter_label: z.string(),
|
||||||
filter_label: z.string(),
|
|
||||||
}),
|
|
||||||
cardConnection: z.object({
|
cardConnection: z.object({
|
||||||
edges: z.array(z.object({ node: contentCardSchema })),
|
edges: z.array(z.object({ node: contentCardSchema })),
|
||||||
}),
|
}),
|
||||||
@@ -40,10 +38,8 @@ const carouselCardsWithoutFilters = z.object({
|
|||||||
enable_filters: z.literal(false),
|
enable_filters: z.literal(false),
|
||||||
card_groups: z.array(
|
card_groups: z.array(
|
||||||
z.object({
|
z.object({
|
||||||
filter_category: z.object({
|
filter_identifier: z.null(),
|
||||||
filter_identifier: z.null(),
|
filter_label: z.string(),
|
||||||
filter_label: z.string(),
|
|
||||||
}),
|
|
||||||
cardConnection: z.object({
|
cardConnection: z.object({
|
||||||
edges: z.array(z.object({ node: contentCardSchema })),
|
edges: z.array(z.object({ node: contentCardSchema })),
|
||||||
}),
|
}),
|
||||||
@@ -74,7 +70,6 @@ export const carouselCardsSchema = z.object({
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
.filter((card): card is NonNullable<typeof card> => card !== null),
|
.filter((card): card is NonNullable<typeof card> => card !== null),
|
||||||
defaultFilter: null,
|
|
||||||
link: data.link
|
link: data.link
|
||||||
? { href: data.link.href, text: data.link.title }
|
? { href: data.link.href, text: data.link.title }
|
||||||
: undefined,
|
: undefined,
|
||||||
@@ -87,11 +82,11 @@ export const carouselCardsSchema = z.object({
|
|||||||
label: string
|
label: string
|
||||||
}>
|
}>
|
||||||
>((acc, group) => {
|
>((acc, group) => {
|
||||||
const identifier = group.filter_category.filter_identifier
|
const identifier = group.filter_identifier
|
||||||
if (!acc.some((category) => category.identifier === identifier)) {
|
if (!acc.some((category) => category.identifier === identifier)) {
|
||||||
acc.push({
|
acc.push({
|
||||||
identifier,
|
identifier,
|
||||||
label: group.filter_category.filter_label,
|
label: group.filter_label,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return acc
|
return acc
|
||||||
@@ -107,11 +102,11 @@ export const carouselCardsSchema = z.object({
|
|||||||
.filter((card): card is NonNullable<typeof card> => card !== null)
|
.filter((card): card is NonNullable<typeof card> => card !== null)
|
||||||
.map((card) => ({
|
.map((card) => ({
|
||||||
...card,
|
...card,
|
||||||
filterId: group.filter_category.filter_identifier,
|
filterId: group.filter_identifier,
|
||||||
}))
|
}))
|
||||||
),
|
),
|
||||||
defaultFilter:
|
defaultFilter:
|
||||||
data.card_groups[0]?.filter_category.filter_identifier ??
|
data.card_groups[0]?.filter_identifier ??
|
||||||
filterCategories[0]?.identifier,
|
filterCategories[0]?.identifier,
|
||||||
link: data.link
|
link: data.link
|
||||||
? { href: data.link.href, text: data.link.title }
|
? { href: data.link.href, text: data.link.title }
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import type { Block as CollectionPageBlock } from "@/types/trpc/routers/contents
|
|||||||
import type { Block as ContentPageBlock } from "@/types/trpc/routers/contentstack/contentPage"
|
import type { Block as ContentPageBlock } from "@/types/trpc/routers/contentstack/contentPage"
|
||||||
import type { Block as DestinationCityPageBlock } from "@/types/trpc/routers/contentstack/destinationCityPage"
|
import type { Block as DestinationCityPageBlock } from "@/types/trpc/routers/contentstack/destinationCityPage"
|
||||||
import type { Block as DestinationCountryPageBlock } from "@/types/trpc/routers/contentstack/destinationCountryPage"
|
import type { Block as DestinationCountryPageBlock } from "@/types/trpc/routers/contentstack/destinationCountryPage"
|
||||||
|
import type { Block as DestinationOverviewPageBlock } from "@/types/trpc/routers/contentstack/destinationOverviewPage"
|
||||||
import type { Block as LoyaltyPageBlock } from "@/types/trpc/routers/contentstack/loyaltyPage"
|
import type { Block as LoyaltyPageBlock } from "@/types/trpc/routers/contentstack/loyaltyPage"
|
||||||
import type { Block as StartPageBlock } from "@/types/trpc/routers/contentstack/startPage"
|
import type { Block as StartPageBlock } from "@/types/trpc/routers/contentstack/startPage"
|
||||||
|
|
||||||
@@ -12,6 +13,7 @@ export type Blocks =
|
|||||||
| ContentPageBlock
|
| ContentPageBlock
|
||||||
| DestinationCityPageBlock
|
| DestinationCityPageBlock
|
||||||
| DestinationCountryPageBlock
|
| DestinationCountryPageBlock
|
||||||
|
| DestinationOverviewPageBlock
|
||||||
| LoyaltyPageBlock
|
| LoyaltyPageBlock
|
||||||
| StartPageBlock
|
| StartPageBlock
|
||||||
|
|
||||||
|
|||||||
7
types/enums/destinationOverviewPage.ts
Normal file
7
types/enums/destinationOverviewPage.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
export namespace DestinationOverviewPageEnum {
|
||||||
|
export namespace ContentStack {
|
||||||
|
export const enum blocks {
|
||||||
|
CarouselCards = "DestinationOverviewPageBlocksCarouselCards",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import type { z } from "zod"
|
import type { z } from "zod"
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
|
blocksSchema,
|
||||||
countryPageUrlSchema,
|
countryPageUrlSchema,
|
||||||
destinationOverviewPageRefsSchema,
|
destinationOverviewPageRefsSchema,
|
||||||
destinationOverviewPageSchema,
|
destinationOverviewPageSchema,
|
||||||
@@ -19,3 +20,5 @@ export interface DestinationOverviewPageRefs
|
|||||||
|
|
||||||
export interface GetCountryPageUrlData
|
export interface GetCountryPageUrlData
|
||||||
extends z.input<typeof countryPageUrlSchema> {}
|
extends z.input<typeof countryPageUrlSchema> {}
|
||||||
|
|
||||||
|
export type Block = z.output<typeof blocksSchema>
|
||||||
|
|||||||
Reference in New Issue
Block a user