feat(SW-2285): Added campaign essentials block on campaign page
Approved-by: Christian Andolf Approved-by: Matilda Landström
This commit is contained in:
@@ -72,3 +72,5 @@ SAS_OCP_APIM=""
|
|||||||
SAS_AUTH_CLIENTID=""
|
SAS_AUTH_CLIENTID=""
|
||||||
|
|
||||||
LOKALISE_API_KEY=""
|
LOKALISE_API_KEY=""
|
||||||
|
|
||||||
|
CAMPAIGN_PAGES_ENABLED="0" # 0 - disabled, 1 - enabled
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import CampaignPageSkeleton from "@/components/ContentType/CampaignPage/Campaign
|
|||||||
export { generateMetadata } from "@/utils/generateMetadata"
|
export { generateMetadata } from "@/utils/generateMetadata"
|
||||||
|
|
||||||
export default async function CampaignPagePage() {
|
export default async function CampaignPagePage() {
|
||||||
if (env.NEW_SITE_LIVE_STATUS === "NOT_LIVE") {
|
if (!env.CAMPAIGN_PAGES_ENABLED) {
|
||||||
return notFound()
|
return notFound()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,108 @@
|
|||||||
|
.essentialsSection {
|
||||||
|
display: grid;
|
||||||
|
gap: var(--Space-x5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
display: grid;
|
||||||
|
gap: var(--Space-x2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.heading {
|
||||||
|
color: var(--Text-Heading);
|
||||||
|
}
|
||||||
|
|
||||||
|
.list {
|
||||||
|
list-style: none;
|
||||||
|
display: grid;
|
||||||
|
grid-auto-rows: auto;
|
||||||
|
row-gap: var(--Space-x15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.listItem {
|
||||||
|
position: relative;
|
||||||
|
display: grid;
|
||||||
|
gap: 19px; /* Special from Figma */
|
||||||
|
color: var(--Text-Secondary);
|
||||||
|
justify-items: center;
|
||||||
|
padding: 0 var(--Space-x1);
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
content: "";
|
||||||
|
width: 1px;
|
||||||
|
background-color: var(--Border-Default);
|
||||||
|
height: 82px; /* Special from Figma */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.text {
|
||||||
|
display: grid;
|
||||||
|
gap: var(--Space-x1);
|
||||||
|
justify-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 949px) {
|
||||||
|
.list {
|
||||||
|
grid-template-columns: repeat(2, 1fr);
|
||||||
|
|
||||||
|
.listItem:nth-child(2n)::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (min-width: 950px) and (max-width: 1366px) {
|
||||||
|
.list {
|
||||||
|
grid-template-columns: repeat(4, 1fr);
|
||||||
|
|
||||||
|
&:not(.count3, .count5, .count6) .listItem:nth-child(4n)::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.count3,
|
||||||
|
&.count5,
|
||||||
|
&.count6 {
|
||||||
|
grid-template-columns: repeat(3, 1fr);
|
||||||
|
|
||||||
|
.listItem:nth-child(3n)::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (min-width: 1367px) {
|
||||||
|
.list {
|
||||||
|
grid-template-columns: repeat(4, 1fr);
|
||||||
|
|
||||||
|
&:not(.count3, .count5, .count6) .listItem:nth-child(4n)::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.count3 {
|
||||||
|
grid-template-columns: repeat(3, 1fr);
|
||||||
|
|
||||||
|
.listItem:nth-child(3n)::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&.count5 {
|
||||||
|
grid-template-columns: repeat(5, 1fr);
|
||||||
|
|
||||||
|
.listItem:nth-child(5n)::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&.count6 {
|
||||||
|
grid-template-columns: repeat(6, 1fr);
|
||||||
|
|
||||||
|
.listItem:nth-child(6n)::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
53
apps/scandic-web/components/Blocks/Essentials/index.tsx
Normal file
53
apps/scandic-web/components/Blocks/Essentials/index.tsx
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { cx } from "class-variance-authority"
|
||||||
|
|
||||||
|
import { Typography } from "@scandic-hotels/design-system/Typography"
|
||||||
|
|
||||||
|
import IconByCSSelect from "@/components/Icons/IconByCSSelect"
|
||||||
|
|
||||||
|
import styles from "./essentials.module.css"
|
||||||
|
|
||||||
|
import type { EssentialsBlock } from "@/types/trpc/routers/contentstack/campaignPage"
|
||||||
|
|
||||||
|
interface EssentialsProps {
|
||||||
|
content: EssentialsBlock
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function Essentials({ content }: EssentialsProps) {
|
||||||
|
const { title, preamble, items } = content
|
||||||
|
|
||||||
|
return (
|
||||||
|
<section className={styles.essentialsSection}>
|
||||||
|
<header className={styles.header}>
|
||||||
|
<Typography variant="Title/sm">
|
||||||
|
<h3 className={styles.heading}>{title}</h3>
|
||||||
|
</Typography>
|
||||||
|
{preamble ? (
|
||||||
|
<Typography variant="Body/Paragraph/mdRegular">
|
||||||
|
<p>{preamble}</p>
|
||||||
|
</Typography>
|
||||||
|
) : null}
|
||||||
|
</header>
|
||||||
|
<ul className={cx(styles.list, styles[`count${items.length}`])}>
|
||||||
|
{items.map((item) => (
|
||||||
|
<li
|
||||||
|
key={`${item.label}-${item.icon_identifier}`}
|
||||||
|
className={styles.listItem}
|
||||||
|
>
|
||||||
|
<IconByCSSelect identifier={item.icon_identifier} size={42} />
|
||||||
|
|
||||||
|
<div className={styles.text}>
|
||||||
|
<Typography variant="Title/Overline/sm">
|
||||||
|
<span>{item.label}</span>
|
||||||
|
</Typography>
|
||||||
|
{item.description ? (
|
||||||
|
<Typography variant="Body/Supporting text (caption)/smRegular">
|
||||||
|
<span>{item.description}</span>
|
||||||
|
</Typography>
|
||||||
|
) : null}
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</section>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -10,6 +10,7 @@ import JsonToHtml from "@/components/JsonToHtml"
|
|||||||
|
|
||||||
import AccordionSection from "./Accordion"
|
import AccordionSection from "./Accordion"
|
||||||
import CardGallery from "./CardGallery"
|
import CardGallery from "./CardGallery"
|
||||||
|
import Essentials from "./Essentials"
|
||||||
import FullWidthCampaign from "./FullWidthCampaign"
|
import FullWidthCampaign from "./FullWidthCampaign"
|
||||||
import HotelListing from "./HotelListing"
|
import HotelListing from "./HotelListing"
|
||||||
import JoinScandicFriends from "./JoinScandicFriends"
|
import JoinScandicFriends from "./JoinScandicFriends"
|
||||||
@@ -112,6 +113,8 @@ export default function Blocks({ blocks }: BlocksProps) {
|
|||||||
<JoinScandicFriends content={block.join_scandic_friends} />
|
<JoinScandicFriends content={block.join_scandic_friends} />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
)
|
)
|
||||||
|
case BlocksEnums.block.Essentials:
|
||||||
|
return <Essentials content={block.essentials} />
|
||||||
default:
|
default:
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,17 @@
|
|||||||
|
import Essentials from "@/components/Blocks/Essentials"
|
||||||
|
|
||||||
|
import type { BlocksProps } from "@/types/components/blocks"
|
||||||
|
import { BlocksEnums } from "@/types/enums/blocks"
|
||||||
|
|
||||||
|
export default function Blocks({ blocks }: BlocksProps) {
|
||||||
|
return blocks.map(async (block) => {
|
||||||
|
switch (block.typename) {
|
||||||
|
case BlocksEnums.block.Essentials:
|
||||||
|
return (
|
||||||
|
<Essentials key={block.essentials.title} content={block.essentials} />
|
||||||
|
)
|
||||||
|
default:
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -6,6 +6,7 @@ import { Typography } from "@scandic-hotels/design-system/Typography"
|
|||||||
|
|
||||||
import { getCampaignPage } from "@/lib/trpc/memoizedRequests"
|
import { getCampaignPage } from "@/lib/trpc/memoizedRequests"
|
||||||
|
|
||||||
|
import Blocks from "./Blocks"
|
||||||
import CampaignPageSkeleton from "./CampaignPageSkeleton"
|
import CampaignPageSkeleton from "./CampaignPageSkeleton"
|
||||||
|
|
||||||
import styles from "./campaignPage.module.css"
|
import styles from "./campaignPage.module.css"
|
||||||
@@ -18,7 +19,7 @@ export default async function CampaignPage() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const { campaignPage } = pageData
|
const { campaignPage } = pageData
|
||||||
const { heading, subheading, preamble } = campaignPage
|
const { heading, subheading, preamble, blocks } = campaignPage
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -52,6 +53,7 @@ export default async function CampaignPage() {
|
|||||||
</div>
|
</div>
|
||||||
</Typography>
|
</Typography>
|
||||||
</div>
|
</div>
|
||||||
|
{blocks ? <Blocks blocks={blocks} /> : null}
|
||||||
</div>
|
</div>
|
||||||
</Suspense>
|
</Suspense>
|
||||||
{/* <TrackingSDK pageData={tracking} /> */}
|
{/* <TrackingSDK pageData={tracking} /> */}
|
||||||
|
|||||||
@@ -6,19 +6,26 @@ import {
|
|||||||
} from "@scandic-hotels/design-system/Icons/MaterialIcon"
|
} from "@scandic-hotels/design-system/Icons/MaterialIcon"
|
||||||
import PalmTreeIcon from "@scandic-hotels/design-system/Icons/PalmTreeIcon"
|
import PalmTreeIcon from "@scandic-hotels/design-system/Icons/PalmTreeIcon"
|
||||||
|
|
||||||
interface FilterIconProps {
|
import type { IconProps } from "@scandic-hotels/design-system/Icons"
|
||||||
|
|
||||||
|
interface IconByCSSelectProps extends IconProps {
|
||||||
identifier: string
|
identifier: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function FilterIcon({ identifier }: FilterIconProps) {
|
export default function IconByCSSelect({
|
||||||
|
identifier,
|
||||||
|
color = "CurrentColor",
|
||||||
|
size = 24,
|
||||||
|
...props
|
||||||
|
}: IconByCSSelectProps) {
|
||||||
switch (identifier) {
|
switch (identifier) {
|
||||||
// These are custom icons
|
// These are custom icons
|
||||||
case "discount-2-2":
|
case "discount-2-2":
|
||||||
return <DiscountIcon size={24} color="CurrentColor" />
|
return <DiscountIcon size={size} color={color} {...props} />
|
||||||
case "bouquet":
|
case "bouquet":
|
||||||
return <BouquetIcon size={24} color="CurrentColor" />
|
return <BouquetIcon size={size} color={color} {...props} />
|
||||||
case "palm_tree":
|
case "palm_tree":
|
||||||
return <PalmTreeIcon size={24} color="CurrentColor" />
|
return <PalmTreeIcon size={size} color={color} {...props} />
|
||||||
|
|
||||||
// These are all Material Icons
|
// These are all Material Icons
|
||||||
case "electric_car":
|
case "electric_car":
|
||||||
@@ -72,10 +79,14 @@ export default function FilterIcon({ identifier }: FilterIconProps) {
|
|||||||
return (
|
return (
|
||||||
<MaterialIcon
|
<MaterialIcon
|
||||||
icon={identifier as MaterialIconProps["icon"]}
|
icon={identifier as MaterialIconProps["icon"]}
|
||||||
color="CurrentColor"
|
size={size}
|
||||||
|
color={color}
|
||||||
|
{...props}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
default:
|
default:
|
||||||
return <MaterialIcon icon="favorite" color="CurrentColor" />
|
return (
|
||||||
|
<MaterialIcon size={size} icon="favorite" color={color} {...props} />
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -6,7 +6,7 @@ import { Typography } from "@scandic-hotels/design-system/Typography"
|
|||||||
|
|
||||||
import useScrollShadows from "@/hooks/useScrollShadows"
|
import useScrollShadows from "@/hooks/useScrollShadows"
|
||||||
|
|
||||||
import FilterIcon from "./FilterIcon"
|
import IconByCSSelect from "../Icons/IconByCSSelect"
|
||||||
|
|
||||||
import styles from "./tabFilters.module.css"
|
import styles from "./tabFilters.module.css"
|
||||||
|
|
||||||
@@ -51,7 +51,7 @@ export default function TabFilters({
|
|||||||
})}
|
})}
|
||||||
type="button"
|
type="button"
|
||||||
>
|
>
|
||||||
<FilterIcon identifier={category.iconIdentifier} />
|
<IconByCSSelect identifier={category.iconIdentifier} />
|
||||||
{category.label}
|
{category.label}
|
||||||
</button>
|
</button>
|
||||||
</Typography>
|
</Typography>
|
||||||
|
|||||||
6
apps/scandic-web/env/server.ts
vendored
6
apps/scandic-web/env/server.ts
vendored
@@ -203,6 +203,11 @@ const _env = createEnv({
|
|||||||
return val.split(",")
|
return val.split(",")
|
||||||
})
|
})
|
||||||
.default(""),
|
.default(""),
|
||||||
|
CAMPAIGN_PAGES_ENABLED: z
|
||||||
|
.string()
|
||||||
|
.refine((s) => s === "1" || s === "0")
|
||||||
|
.transform((s) => s === "1")
|
||||||
|
.default("0"),
|
||||||
},
|
},
|
||||||
emptyStringAsUndefined: true,
|
emptyStringAsUndefined: true,
|
||||||
runtimeEnv: {
|
runtimeEnv: {
|
||||||
@@ -296,6 +301,7 @@ const _env = createEnv({
|
|||||||
ENABLE_WARMUP_HOTEL: process.env.ENABLE_WARMUP_HOTEL,
|
ENABLE_WARMUP_HOTEL: process.env.ENABLE_WARMUP_HOTEL,
|
||||||
WARMUP_TOKEN: process.env.WARMUP_TOKEN,
|
WARMUP_TOKEN: process.env.WARMUP_TOKEN,
|
||||||
NEW_SITE_LIVE_FOR_LANGS: process.env.NEXT_PUBLIC_NEW_SITE_LIVE_FOR_LANGS,
|
NEW_SITE_LIVE_FOR_LANGS: process.env.NEXT_PUBLIC_NEW_SITE_LIVE_FOR_LANGS,
|
||||||
|
CAMPAIGN_PAGES_ENABLED: process.env.CAMPAIGN_PAGES_ENABLED,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,11 @@
|
|||||||
|
fragment Essentials_CampaignPage on CampaignPageBlocksEssentials {
|
||||||
|
essentials {
|
||||||
|
title
|
||||||
|
preamble
|
||||||
|
items {
|
||||||
|
label
|
||||||
|
icon_identifier
|
||||||
|
description
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
#import "../../Fragments/System.graphql"
|
#import "../../Fragments/System.graphql"
|
||||||
|
#import "../../Fragments/Blocks/Essentials.graphql"
|
||||||
|
|
||||||
query GetCampaignPage($locale: String!, $uid: String!) {
|
query GetCampaignPage($locale: String!, $uid: String!) {
|
||||||
campaign_page(uid: $uid, locale: $locale) {
|
campaign_page(uid: $uid, locale: $locale) {
|
||||||
@@ -11,6 +12,10 @@ query GetCampaignPage($locale: String!, $uid: String!) {
|
|||||||
first_column
|
first_column
|
||||||
second_column
|
second_column
|
||||||
}
|
}
|
||||||
|
blocks {
|
||||||
|
__typename
|
||||||
|
...Essentials_CampaignPage
|
||||||
|
}
|
||||||
system {
|
system {
|
||||||
...System
|
...System
|
||||||
created_at
|
created_at
|
||||||
|
|||||||
@@ -1,7 +1,22 @@
|
|||||||
import { z } from "zod"
|
import { z } from "zod"
|
||||||
|
|
||||||
|
import { discriminatedUnionArray } from "@/lib/discriminatedUnion"
|
||||||
|
|
||||||
|
import { essentialsBlockSchema } from "../schemas/blocks/essentials"
|
||||||
import { systemSchema } from "../schemas/system"
|
import { systemSchema } from "../schemas/system"
|
||||||
|
|
||||||
|
import { CampaignPageEnum } from "@/types/enums/campaignPage"
|
||||||
|
|
||||||
|
const campaignPageEssentials = z
|
||||||
|
.object({
|
||||||
|
__typename: z.literal(CampaignPageEnum.ContentStack.blocks.Essentials),
|
||||||
|
})
|
||||||
|
.merge(essentialsBlockSchema)
|
||||||
|
|
||||||
|
export const blocksSchema = z.discriminatedUnion("__typename", [
|
||||||
|
campaignPageEssentials,
|
||||||
|
])
|
||||||
|
|
||||||
export const campaignPageSchema = z.object({
|
export const campaignPageSchema = z.object({
|
||||||
campaign_page: z.object({
|
campaign_page: z.object({
|
||||||
title: z.string(),
|
title: z.string(),
|
||||||
@@ -13,6 +28,7 @@ export const campaignPageSchema = z.object({
|
|||||||
first_column: z.string(),
|
first_column: z.string(),
|
||||||
second_column: z.string(),
|
second_column: z.string(),
|
||||||
}),
|
}),
|
||||||
|
blocks: discriminatedUnionArray(blocksSchema.options),
|
||||||
system: systemSchema.merge(
|
system: systemSchema.merge(
|
||||||
z.object({
|
z.object({
|
||||||
created_at: z.string(),
|
created_at: z.string(),
|
||||||
|
|||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import { z } from "zod"
|
||||||
|
|
||||||
|
import { BlocksEnums } from "@/types/enums/blocks"
|
||||||
|
|
||||||
|
export const essentialsSchema = z.object({
|
||||||
|
essentials: z.object({
|
||||||
|
title: z.string(),
|
||||||
|
preamble: z.string().nullish(),
|
||||||
|
items: z.array(
|
||||||
|
z.object({
|
||||||
|
label: z.string(),
|
||||||
|
icon_identifier: z.string(),
|
||||||
|
description: z.string().nullish(),
|
||||||
|
})
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
|
||||||
|
export const essentialsBlockSchema = z
|
||||||
|
.object({
|
||||||
|
typename: z
|
||||||
|
.literal(BlocksEnums.block.Essentials)
|
||||||
|
.optional()
|
||||||
|
.default(BlocksEnums.block.Essentials),
|
||||||
|
})
|
||||||
|
.merge(essentialsSchema)
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
import type { Block as AccountPageBlock } from "@/types/trpc/routers/contentstack/accountPage"
|
import type { Block as AccountPageBlock } from "@/types/trpc/routers/contentstack/accountPage"
|
||||||
|
import type { Block as CampaignPageBlock } from "@/types/trpc/routers/contentstack/campaignPage"
|
||||||
import type { Block as CollectionPageBlock } from "@/types/trpc/routers/contentstack/collectionPage"
|
import type { Block as CollectionPageBlock } from "@/types/trpc/routers/contentstack/collectionPage"
|
||||||
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"
|
||||||
@@ -9,6 +10,7 @@ import type { Block as StartPageBlock } from "@/types/trpc/routers/contentstack/
|
|||||||
|
|
||||||
export type Blocks =
|
export type Blocks =
|
||||||
| AccountPageBlock
|
| AccountPageBlock
|
||||||
|
| CampaignPageBlock
|
||||||
| CollectionPageBlock
|
| CollectionPageBlock
|
||||||
| ContentPageBlock
|
| ContentPageBlock
|
||||||
| DestinationCityPageBlock
|
| DestinationCityPageBlock
|
||||||
|
|||||||
@@ -14,5 +14,6 @@ export namespace BlocksEnums {
|
|||||||
TextCols = "TextCols",
|
TextCols = "TextCols",
|
||||||
TextContent = "TextContent",
|
TextContent = "TextContent",
|
||||||
UspGrid = "UspGrid",
|
UspGrid = "UspGrid",
|
||||||
|
Essentials = "Essentials",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
7
apps/scandic-web/types/enums/campaignPage.ts
Normal file
7
apps/scandic-web/types/enums/campaignPage.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
export namespace CampaignPageEnum {
|
||||||
|
export namespace ContentStack {
|
||||||
|
export const enum blocks {
|
||||||
|
Essentials = "CampaignPageBlocksEssentials",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,9 +1,11 @@
|
|||||||
import type { z } from "zod"
|
import type { z } from "zod"
|
||||||
|
|
||||||
import type {
|
import type {
|
||||||
|
blocksSchema,
|
||||||
campaignPageRefsSchema,
|
campaignPageRefsSchema,
|
||||||
campaignPageSchema,
|
campaignPageSchema,
|
||||||
} from "@/server/routers/contentstack/campaignPage/output"
|
} from "@/server/routers/contentstack/campaignPage/output"
|
||||||
|
import type { essentialsSchema } from "@/server/routers/contentstack/schemas/blocks/essentials"
|
||||||
|
|
||||||
export interface GetCampaignPageData
|
export interface GetCampaignPageData
|
||||||
extends z.input<typeof campaignPageSchema> {}
|
extends z.input<typeof campaignPageSchema> {}
|
||||||
@@ -15,3 +17,7 @@ export interface GetCampaignPageRefsData
|
|||||||
|
|
||||||
export interface CampaignPageRefs
|
export interface CampaignPageRefs
|
||||||
extends z.output<typeof campaignPageRefsSchema> {}
|
extends z.output<typeof campaignPageRefsSchema> {}
|
||||||
|
|
||||||
|
export type Block = z.output<typeof blocksSchema>
|
||||||
|
|
||||||
|
export type EssentialsBlock = z.output<typeof essentialsSchema>["essentials"]
|
||||||
|
|||||||
Reference in New Issue
Block a user