@windstream/react-shared-components 0.1.79 → 0.1.80
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +635 -635
- package/dist/contentful/index.esm.js +3 -3
- package/dist/contentful/index.esm.js.map +1 -1
- package/dist/contentful/index.js +3 -3
- package/dist/contentful/index.js.map +1 -1
- package/dist/core.d.ts +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.esm.js +13 -5
- package/dist/index.esm.js.map +1 -1
- package/dist/index.js +13 -5
- package/dist/index.js.map +1 -1
- package/dist/next/index.esm.js +2 -2
- package/dist/next/index.esm.js.map +1 -1
- package/dist/next/index.js +2 -2
- package/dist/next/index.js.map +1 -1
- package/dist/utils/index.esm.js +1 -1
- package/dist/utils/index.esm.js.map +1 -1
- package/dist/utils/index.js +1 -1
- package/dist/utils/index.js.map +1 -1
- package/package.json +185 -185
- package/src/components/accordion/Accordion.stories.tsx +230 -230
- package/src/components/accordion/index.tsx +70 -70
- package/src/components/accordion/types.ts +12 -12
- package/src/components/alert-card/AlertCard.stories.tsx +171 -171
- package/src/components/alert-card/index.tsx +41 -41
- package/src/components/alert-card/types.ts +13 -13
- package/src/components/brand-button/BrandButton.stories.tsx +223 -223
- package/src/components/brand-button/helpers.ts +35 -35
- package/src/components/brand-button/index.tsx +120 -120
- package/src/components/brand-button/types.ts +38 -38
- package/src/components/button/Button.stories.tsx +108 -108
- package/src/components/button/index.tsx +27 -27
- package/src/components/button/types.ts +14 -14
- package/src/components/call-button/CallButton.stories.tsx +324 -324
- package/src/components/call-button/index.tsx +106 -106
- package/src/components/call-button/types.ts +16 -16
- package/src/components/checkbox/Checkbox.stories.tsx +247 -247
- package/src/components/checkbox/index.tsx +197 -197
- package/src/components/checkbox/types.ts +27 -27
- package/src/components/checklist/Checklist.stories.tsx +150 -150
- package/src/components/checklist/index.tsx +61 -61
- package/src/components/checklist/types.ts +17 -17
- package/src/components/collapse/Collapse.stories.tsx +255 -255
- package/src/components/collapse/index.tsx +46 -46
- package/src/components/collapse/types.ts +6 -6
- package/src/components/divider/Divider.stories.tsx +205 -205
- package/src/components/divider/index.tsx +22 -22
- package/src/components/divider/type.ts +3 -3
- package/src/components/image/Image.stories.tsx +113 -113
- package/src/components/image/index.tsx +25 -25
- package/src/components/image/types.ts +40 -40
- package/src/components/input/Input.stories.tsx +325 -325
- package/src/components/input/index.tsx +177 -177
- package/src/components/input/types.ts +37 -37
- package/src/components/link/Link.stories.tsx +163 -163
- package/src/components/link/index.tsx +116 -116
- package/src/components/link/types.ts +25 -25
- package/src/components/list/List.stories.tsx +272 -272
- package/src/components/list/index.tsx +88 -88
- package/src/components/list/list-item/index.tsx +38 -38
- package/src/components/list/list-item/types.ts +13 -13
- package/src/components/list/types.ts +29 -29
- package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
- package/src/components/material-icon/constants.ts +99 -99
- package/src/components/material-icon/index.tsx +47 -47
- package/src/components/material-icon/types.ts +31 -31
- package/src/components/modal/Modal.stories.tsx +171 -171
- package/src/components/modal/index.tsx +164 -164
- package/src/components/modal/types.ts +24 -24
- package/src/components/next-image/index.tsx +74 -74
- package/src/components/next-image/types.ts +1 -1
- package/src/components/pagination/index.tsx +91 -91
- package/src/components/pagination/types.ts +6 -6
- package/src/components/radio-button/RadioButton.stories.tsx +307 -307
- package/src/components/radio-button/index.tsx +75 -75
- package/src/components/radio-button/types.ts +21 -21
- package/src/components/see-more/SeeMore.stories.tsx +181 -181
- package/src/components/see-more/index.tsx +44 -44
- package/src/components/see-more/types.ts +4 -4
- package/src/components/select/Select.stories.tsx +411 -411
- package/src/components/select/index.tsx +155 -155
- package/src/components/select/types.ts +36 -36
- package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
- package/src/components/select-plan-button/index.tsx +63 -63
- package/src/components/select-plan-button/types.ts +17 -17
- package/src/components/skeleton/Skeleton.stories.tsx +179 -179
- package/src/components/skeleton/index.tsx +61 -61
- package/src/components/skeleton/types.ts +4 -4
- package/src/components/spinner/Spinner.stories.tsx +335 -335
- package/src/components/spinner/index.tsx +44 -44
- package/src/components/spinner/types.ts +5 -5
- package/src/components/text/Text.stories.tsx +321 -321
- package/src/components/text/index.tsx +25 -25
- package/src/components/text/types.ts +45 -45
- package/src/components/tooltip/Tooltip.stories.tsx +219 -219
- package/src/components/tooltip/index.tsx +74 -74
- package/src/components/tooltip/types.ts +7 -7
- package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
- package/src/components/view-cart-button/index.tsx +42 -42
- package/src/components/view-cart-button/types.ts +5 -5
- package/src/contentful/blocks/accordion/Accordion.stories.mocks.tsx +128 -128
- package/src/contentful/blocks/accordion/Accordion.stories.tsx +98 -98
- package/src/contentful/blocks/accordion/index.tsx +112 -112
- package/src/contentful/blocks/accordion/types.ts +34 -34
- package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
- package/src/contentful/blocks/address-input-banner/types.ts +14 -14
- package/src/contentful/blocks/anchored-bottom-banner/index.tsx +181 -181
- package/src/contentful/blocks/anchored-bottom-banner/types.ts +13 -13
- package/src/contentful/blocks/blogs-grid/BlogGrid.stories.mocks.tsx +144 -144
- package/src/contentful/blocks/blogs-grid/BlogGrid.stories.tsx +156 -156
- package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
- package/src/contentful/blocks/blogs-grid/types.ts +26 -26
- package/src/contentful/blocks/blogs-grid-base/index.tsx +119 -119
- package/src/contentful/blocks/blogs-grid-base/types.ts +36 -36
- package/src/contentful/blocks/breadcrumbs/BreadcrumbNavigation.stories.tsx +147 -147
- package/src/contentful/blocks/breadcrumbs/index.tsx +95 -95
- package/src/contentful/blocks/breadcrumbs/types.ts +8 -8
- package/src/contentful/blocks/button/Button.stories.tsx +40 -40
- package/src/contentful/blocks/button/index.tsx +131 -131
- package/src/contentful/blocks/button/types.ts +39 -39
- package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
- package/src/contentful/blocks/callout/index.tsx +265 -265
- package/src/contentful/blocks/callout/types.ts +72 -72
- package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
- package/src/contentful/blocks/cards/blog-card/index.tsx +129 -129
- package/src/contentful/blocks/cards/blog-card/types.ts +34 -34
- package/src/contentful/blocks/cards/floating-image-card/index.tsx +119 -119
- package/src/contentful/blocks/cards/floating-image-card/types.ts +30 -30
- package/src/contentful/blocks/cards/full-image-card/index.tsx +130 -130
- package/src/contentful/blocks/cards/full-image-card/types.ts +29 -29
- package/src/contentful/blocks/cards/index.tsx +13 -13
- package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
- package/src/contentful/blocks/cards/product-card/types.ts +28 -28
- package/src/contentful/blocks/cards/simple-card/index.tsx +325 -325
- package/src/contentful/blocks/cards/simple-card/types.ts +71 -71
- package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
- package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
- package/src/contentful/blocks/cards/types.ts +1 -1
- package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
- package/src/contentful/blocks/carousel/helper.tsx +440 -440
- package/src/contentful/blocks/carousel/index.tsx +85 -85
- package/src/contentful/blocks/carousel/types.ts +144 -144
- package/src/contentful/blocks/cart-retention-banner/index.tsx +105 -105
- package/src/contentful/blocks/cart-retention-banner/types.ts +11 -11
- package/src/contentful/blocks/comparison-table/index.tsx +29 -29
- package/src/contentful/blocks/comparison-table/types.ts +6 -6
- package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
- package/src/contentful/blocks/cookiebanner/type.ts +7 -7
- package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
- package/src/contentful/blocks/cta-callout/index.tsx +71 -71
- package/src/contentful/blocks/cta-callout/types.ts +26 -26
- package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
- package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
- package/src/contentful/blocks/email-input-block/index.tsx +116 -116
- package/src/contentful/blocks/email-input-block/types.ts +16 -16
- package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
- package/src/contentful/blocks/find-kinetic/index.tsx +130 -130
- package/src/contentful/blocks/find-kinetic/types.ts +19 -19
- package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
- package/src/contentful/blocks/floating-banner/index.tsx +97 -97
- package/src/contentful/blocks/floating-banner/types.ts +22 -22
- package/src/contentful/blocks/footer/Footer.stories.tsx +317 -317
- package/src/contentful/blocks/footer/index.tsx +91 -91
- package/src/contentful/blocks/footer/types.ts +13 -13
- package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
- package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
- package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
- package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
- package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
- package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
- package/src/contentful/blocks/modal/constants.ts +53 -53
- package/src/contentful/blocks/modal/index.tsx +107 -107
- package/src/contentful/blocks/modal/types.ts +12 -12
- package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +139 -139
- package/src/contentful/blocks/navigation/index.tsx +568 -568
- package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
- package/src/contentful/blocks/navigation/types.ts +71 -71
- package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
- package/src/contentful/blocks/primary-hero/index.tsx +236 -236
- package/src/contentful/blocks/primary-hero/types.ts +37 -37
- package/src/contentful/blocks/search-block/index.tsx +90 -90
- package/src/contentful/blocks/search-block/types.ts +15 -15
- package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
- package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
- package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
- package/src/contentful/blocks/text/Text.stories.tsx +23 -23
- package/src/contentful/blocks/text/index.tsx +12 -12
- package/src/contentful/blocks/text/types.ts +1 -1
- package/src/contentful/index.ts +105 -105
- package/src/hooks/contentful/use-contentful-rich-text.tsx +309 -309
- package/src/hooks/contentful/use-processed-check-list.ts +63 -63
- package/src/hooks/use-body-scroll-lock.ts +34 -34
- package/src/hooks/use-carousel-swipe.ts +264 -264
- package/src/hooks/use-outside-click.ts +17 -17
- package/src/index.ts +101 -101
- package/src/next/index.ts +5 -5
- package/src/setupTests.ts +46 -46
- package/src/stories/DocsTemplate.tsx +24 -24
- package/src/styles/globals.css +343 -343
- package/src/types/global.d.ts +9 -9
- package/src/types/micro-components.ts +99 -99
- package/src/types/utm.ts +49 -49
- package/src/utils/contentful/to-document.ts +24 -24
- package/src/utils/cookie.ts +84 -84
- package/src/utils/cx.ts +49 -49
- package/src/utils/index.ts +41 -41
- package/src/utils/speed-card-bg.ts +24 -24
- package/src/utils/utm.ts +221 -221
|
@@ -1,23 +1,23 @@
|
|
|
1
|
-
import { Cards } from "./index";
|
|
2
|
-
|
|
3
|
-
import { DocsPage } from "@shared/stories/DocsTemplate";
|
|
4
|
-
import type { Meta, StoryObj } from "@storybook/react";
|
|
5
|
-
|
|
6
|
-
const meta: Meta<typeof Cards> = {
|
|
7
|
-
title: "Contentful Blocks/Cards",
|
|
8
|
-
component: Cards,
|
|
9
|
-
tags: ["autodocs"],
|
|
10
|
-
parameters: {
|
|
11
|
-
layout: "centered",
|
|
12
|
-
docs: {
|
|
13
|
-
page: DocsPage,
|
|
14
|
-
description: {
|
|
15
|
-
component: "Contentful cards block.",
|
|
16
|
-
},
|
|
17
|
-
},
|
|
18
|
-
},
|
|
19
|
-
args: { fields: {} },
|
|
20
|
-
};
|
|
21
|
-
export default meta;
|
|
22
|
-
type Story = StoryObj<typeof meta>;
|
|
23
|
-
export const Default: Story = {};
|
|
1
|
+
import { Cards } from "./index";
|
|
2
|
+
|
|
3
|
+
import { DocsPage } from "@shared/stories/DocsTemplate";
|
|
4
|
+
import type { Meta, StoryObj } from "@storybook/react";
|
|
5
|
+
|
|
6
|
+
const meta: Meta<typeof Cards> = {
|
|
7
|
+
title: "Contentful Blocks/Cards",
|
|
8
|
+
component: Cards,
|
|
9
|
+
tags: ["autodocs"],
|
|
10
|
+
parameters: {
|
|
11
|
+
layout: "centered",
|
|
12
|
+
docs: {
|
|
13
|
+
page: DocsPage,
|
|
14
|
+
description: {
|
|
15
|
+
component: "Contentful cards block.",
|
|
16
|
+
},
|
|
17
|
+
},
|
|
18
|
+
},
|
|
19
|
+
args: { fields: {} },
|
|
20
|
+
};
|
|
21
|
+
export default meta;
|
|
22
|
+
type Story = StoryObj<typeof meta>;
|
|
23
|
+
export const Default: Story = {};
|
|
@@ -1,129 +1,129 @@
|
|
|
1
|
-
import React from "react";
|
|
2
|
-
import { BlogCardProps } from "./types";
|
|
3
|
-
import * as _NextLink from "next/link";
|
|
4
|
-
|
|
5
|
-
// import { Button } from "@shared/components/button";
|
|
6
|
-
import { MaterialIcon } from "@shared/components/material-icon";
|
|
7
|
-
import { Text } from "@shared/components/text";
|
|
8
|
-
|
|
9
|
-
// CJS/ESM interop: next/link uses standard __esModule exports so this is safe.
|
|
10
|
-
// next/image is intentionally avoided — its image-external.js uses the
|
|
11
|
-
// `0 && (module.exports = {...})` tree-shaking hint which confuses webpack's
|
|
12
|
-
// CJS→ESM interop and causes "got: object" errors in consuming apps.
|
|
13
|
-
const Link = ((_NextLink as any).default ??
|
|
14
|
-
_NextLink) as typeof import("next/link").default;
|
|
15
|
-
|
|
16
|
-
export const BlogCard: React.FC<BlogCardProps> = ({
|
|
17
|
-
href,
|
|
18
|
-
title,
|
|
19
|
-
description,
|
|
20
|
-
date,
|
|
21
|
-
category,
|
|
22
|
-
image,
|
|
23
|
-
imageComponent: ImageComponent,
|
|
24
|
-
readMoreText = "Read more",
|
|
25
|
-
asGrid = true,
|
|
26
|
-
lgWidth,
|
|
27
|
-
mdWidth,
|
|
28
|
-
index,
|
|
29
|
-
}: BlogCardProps) => {
|
|
30
|
-
const basehwClass = "lg:w-[calc(33.3333%-1rem)] md:w-[calc(50%-1rem)]";
|
|
31
|
-
const parentClassName = asGrid
|
|
32
|
-
? "flex h-full flex-col overflow-hidden rounded-card-lg bg-white shadow-drop transition-all duration-200 hover:-translate-y-0.5 hover:shadow-cardDrop"
|
|
33
|
-
: `callout-card ${basehwClass} ${lgWidth} ${mdWidth} w-full overflow-hidden rounded-card-lg bg-white shadow-drop transition-all duration-200 hover:-translate-y-0.5 hover:shadow-cardDrop`;
|
|
34
|
-
|
|
35
|
-
return (
|
|
36
|
-
<article
|
|
37
|
-
className={parentClassName}
|
|
38
|
-
data-section-type={"blog-card"}
|
|
39
|
-
data-section-index={index}
|
|
40
|
-
>
|
|
41
|
-
{/* Image */}
|
|
42
|
-
<Link href={href} tabIndex={-1} aria-hidden="true" className="block">
|
|
43
|
-
<div className="h-[232px] w-full flex-shrink-0 overflow-hidden bg-gray-100">
|
|
44
|
-
{image ? (
|
|
45
|
-
ImageComponent ? (
|
|
46
|
-
<ImageComponent
|
|
47
|
-
src={image.src}
|
|
48
|
-
alt={image.alt}
|
|
49
|
-
width={image.width}
|
|
50
|
-
height={image.height}
|
|
51
|
-
className="h-full w-full object-cover transition-transform duration-300 hover:scale-[1.03]"
|
|
52
|
-
sizes="(max-width: 640px) 100vw, (max-width: 1024px) 50vw, 33vw"
|
|
53
|
-
/>
|
|
54
|
-
) : (
|
|
55
|
-
<img
|
|
56
|
-
src={image.src}
|
|
57
|
-
alt={image.alt}
|
|
58
|
-
width={image.width}
|
|
59
|
-
height={image.height}
|
|
60
|
-
loading="lazy"
|
|
61
|
-
decoding="async"
|
|
62
|
-
className="h-full w-full object-cover transition-transform duration-300 hover:scale-[1.03]"
|
|
63
|
-
/>
|
|
64
|
-
)
|
|
65
|
-
) : (
|
|
66
|
-
<div
|
|
67
|
-
className="h-full w-full bg-gradient-to-br from-gray-200 to-gray-100"
|
|
68
|
-
aria-hidden="true"
|
|
69
|
-
/>
|
|
70
|
-
)}
|
|
71
|
-
</div>
|
|
72
|
-
</Link>
|
|
73
|
-
|
|
74
|
-
{/* Body */}
|
|
75
|
-
<div className="flex flex-1 flex-col gap-5 p-6 md:p-8">
|
|
76
|
-
{/* Meta: category + date */}
|
|
77
|
-
<div className="flex items-center gap-2 text-[13px]">
|
|
78
|
-
<span className="body2 text-text-brand">{category}</span>
|
|
79
|
-
{date && (
|
|
80
|
-
<>
|
|
81
|
-
<span className="footnote text-text" aria-hidden="true">
|
|
82
|
-
•
|
|
83
|
-
</span>
|
|
84
|
-
<time className="body2 text-text">{date}</time>
|
|
85
|
-
</>
|
|
86
|
-
)}
|
|
87
|
-
</div>
|
|
88
|
-
|
|
89
|
-
{/* Title */}
|
|
90
|
-
{title && (
|
|
91
|
-
<Link
|
|
92
|
-
href={href}
|
|
93
|
-
className="heading6 m-0 line-clamp-3 font-black transition-colors duration-150 hover:text-text-brand"
|
|
94
|
-
>
|
|
95
|
-
{title}
|
|
96
|
-
</Link>
|
|
97
|
-
)}
|
|
98
|
-
|
|
99
|
-
{/* Excerpt */}
|
|
100
|
-
{description && (
|
|
101
|
-
<Text className="body1 m-0 line-clamp-3 flex-1 text-text">
|
|
102
|
-
{description}
|
|
103
|
-
</Text>
|
|
104
|
-
)}
|
|
105
|
-
|
|
106
|
-
{/* Read more */}
|
|
107
|
-
<Link
|
|
108
|
-
href={href}
|
|
109
|
-
className="group mt-auto inline-flex items-center justify-start gap-2 pt-3 text-sm text-text-brand no-underline"
|
|
110
|
-
aria-label={`${readMoreText} about ${title || "this article"}`}
|
|
111
|
-
>
|
|
112
|
-
<Text
|
|
113
|
-
aria-label={`${readMoreText} about ${title || "this article"}`}
|
|
114
|
-
className="label1 text-nowrap"
|
|
115
|
-
>
|
|
116
|
-
{readMoreText}
|
|
117
|
-
</Text>
|
|
118
|
-
<MaterialIcon
|
|
119
|
-
name="expand_circle_right"
|
|
120
|
-
fill={1}
|
|
121
|
-
size={24}
|
|
122
|
-
weight="200"
|
|
123
|
-
/>
|
|
124
|
-
</Link>
|
|
125
|
-
</div>
|
|
126
|
-
</article>
|
|
127
|
-
);
|
|
128
|
-
};
|
|
129
|
-
export default BlogCard;
|
|
1
|
+
import React from "react";
|
|
2
|
+
import { BlogCardProps } from "./types";
|
|
3
|
+
import * as _NextLink from "next/link";
|
|
4
|
+
|
|
5
|
+
// import { Button } from "@shared/components/button";
|
|
6
|
+
import { MaterialIcon } from "@shared/components/material-icon";
|
|
7
|
+
import { Text } from "@shared/components/text";
|
|
8
|
+
|
|
9
|
+
// CJS/ESM interop: next/link uses standard __esModule exports so this is safe.
|
|
10
|
+
// next/image is intentionally avoided — its image-external.js uses the
|
|
11
|
+
// `0 && (module.exports = {...})` tree-shaking hint which confuses webpack's
|
|
12
|
+
// CJS→ESM interop and causes "got: object" errors in consuming apps.
|
|
13
|
+
const Link = ((_NextLink as any).default ??
|
|
14
|
+
_NextLink) as typeof import("next/link").default;
|
|
15
|
+
|
|
16
|
+
export const BlogCard: React.FC<BlogCardProps> = ({
|
|
17
|
+
href,
|
|
18
|
+
title,
|
|
19
|
+
description,
|
|
20
|
+
date,
|
|
21
|
+
category,
|
|
22
|
+
image,
|
|
23
|
+
imageComponent: ImageComponent,
|
|
24
|
+
readMoreText = "Read more",
|
|
25
|
+
asGrid = true,
|
|
26
|
+
lgWidth,
|
|
27
|
+
mdWidth,
|
|
28
|
+
index,
|
|
29
|
+
}: BlogCardProps) => {
|
|
30
|
+
const basehwClass = "lg:w-[calc(33.3333%-1rem)] md:w-[calc(50%-1rem)]";
|
|
31
|
+
const parentClassName = asGrid
|
|
32
|
+
? "flex h-full flex-col overflow-hidden rounded-card-lg bg-white shadow-drop transition-all duration-200 hover:-translate-y-0.5 hover:shadow-cardDrop"
|
|
33
|
+
: `callout-card ${basehwClass} ${lgWidth} ${mdWidth} w-full overflow-hidden rounded-card-lg bg-white shadow-drop transition-all duration-200 hover:-translate-y-0.5 hover:shadow-cardDrop`;
|
|
34
|
+
|
|
35
|
+
return (
|
|
36
|
+
<article
|
|
37
|
+
className={parentClassName}
|
|
38
|
+
data-section-type={"blog-card"}
|
|
39
|
+
data-section-index={index}
|
|
40
|
+
>
|
|
41
|
+
{/* Image */}
|
|
42
|
+
<Link href={href} tabIndex={-1} aria-hidden="true" className="block">
|
|
43
|
+
<div className="h-[232px] w-full flex-shrink-0 overflow-hidden bg-gray-100">
|
|
44
|
+
{image ? (
|
|
45
|
+
ImageComponent ? (
|
|
46
|
+
<ImageComponent
|
|
47
|
+
src={image.src}
|
|
48
|
+
alt={image.alt}
|
|
49
|
+
width={image.width}
|
|
50
|
+
height={image.height}
|
|
51
|
+
className="h-full w-full object-cover transition-transform duration-300 hover:scale-[1.03]"
|
|
52
|
+
sizes="(max-width: 640px) 100vw, (max-width: 1024px) 50vw, 33vw"
|
|
53
|
+
/>
|
|
54
|
+
) : (
|
|
55
|
+
<img
|
|
56
|
+
src={image.src}
|
|
57
|
+
alt={image.alt}
|
|
58
|
+
width={image.width}
|
|
59
|
+
height={image.height}
|
|
60
|
+
loading="lazy"
|
|
61
|
+
decoding="async"
|
|
62
|
+
className="h-full w-full object-cover transition-transform duration-300 hover:scale-[1.03]"
|
|
63
|
+
/>
|
|
64
|
+
)
|
|
65
|
+
) : (
|
|
66
|
+
<div
|
|
67
|
+
className="h-full w-full bg-gradient-to-br from-gray-200 to-gray-100"
|
|
68
|
+
aria-hidden="true"
|
|
69
|
+
/>
|
|
70
|
+
)}
|
|
71
|
+
</div>
|
|
72
|
+
</Link>
|
|
73
|
+
|
|
74
|
+
{/* Body */}
|
|
75
|
+
<div className="flex flex-1 flex-col gap-5 p-6 md:p-8">
|
|
76
|
+
{/* Meta: category + date */}
|
|
77
|
+
<div className="flex items-center gap-2 text-[13px]">
|
|
78
|
+
<span className="body2 text-text-brand">{category}</span>
|
|
79
|
+
{date && (
|
|
80
|
+
<>
|
|
81
|
+
<span className="footnote text-text" aria-hidden="true">
|
|
82
|
+
•
|
|
83
|
+
</span>
|
|
84
|
+
<time className="body2 text-text">{date}</time>
|
|
85
|
+
</>
|
|
86
|
+
)}
|
|
87
|
+
</div>
|
|
88
|
+
|
|
89
|
+
{/* Title */}
|
|
90
|
+
{title && (
|
|
91
|
+
<Link
|
|
92
|
+
href={href}
|
|
93
|
+
className="heading6 m-0 line-clamp-3 font-black transition-colors duration-150 hover:text-text-brand"
|
|
94
|
+
>
|
|
95
|
+
{title}
|
|
96
|
+
</Link>
|
|
97
|
+
)}
|
|
98
|
+
|
|
99
|
+
{/* Excerpt */}
|
|
100
|
+
{description && (
|
|
101
|
+
<Text className="body1 m-0 line-clamp-3 flex-1 text-text">
|
|
102
|
+
{description}
|
|
103
|
+
</Text>
|
|
104
|
+
)}
|
|
105
|
+
|
|
106
|
+
{/* Read more */}
|
|
107
|
+
<Link
|
|
108
|
+
href={href}
|
|
109
|
+
className="group mt-auto inline-flex items-center justify-start gap-2 pt-3 text-sm text-text-brand no-underline"
|
|
110
|
+
aria-label={`${readMoreText} about ${title || "this article"}`}
|
|
111
|
+
>
|
|
112
|
+
<Text
|
|
113
|
+
aria-label={`${readMoreText} about ${title || "this article"}`}
|
|
114
|
+
className="label1 text-nowrap"
|
|
115
|
+
>
|
|
116
|
+
{readMoreText}
|
|
117
|
+
</Text>
|
|
118
|
+
<MaterialIcon
|
|
119
|
+
name="expand_circle_right"
|
|
120
|
+
fill={1}
|
|
121
|
+
size={24}
|
|
122
|
+
weight="200"
|
|
123
|
+
/>
|
|
124
|
+
</Link>
|
|
125
|
+
</div>
|
|
126
|
+
</article>
|
|
127
|
+
);
|
|
128
|
+
};
|
|
129
|
+
export default BlogCard;
|
|
@@ -1,34 +1,34 @@
|
|
|
1
|
-
import React from "react";
|
|
2
|
-
|
|
3
|
-
export interface BlogCardImageProps {
|
|
4
|
-
src: string;
|
|
5
|
-
alt: string;
|
|
6
|
-
width: number;
|
|
7
|
-
height: number;
|
|
8
|
-
className?: string;
|
|
9
|
-
sizes?: string;
|
|
10
|
-
loading?: "lazy" | "eager";
|
|
11
|
-
decoding?: "async" | "auto" | "sync";
|
|
12
|
-
[key: string]: unknown;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export interface BlogCardProps {
|
|
16
|
-
href: string;
|
|
17
|
-
title?: string;
|
|
18
|
-
description?: string;
|
|
19
|
-
date?: string;
|
|
20
|
-
category?: string;
|
|
21
|
-
image?: {
|
|
22
|
-
src: string;
|
|
23
|
-
alt: string;
|
|
24
|
-
width: number;
|
|
25
|
-
height: number;
|
|
26
|
-
};
|
|
27
|
-
/** Pass next/image's Image component here to enable image optimization */
|
|
28
|
-
imageComponent?: React.ComponentType<BlogCardImageProps>;
|
|
29
|
-
readMoreText?: string;
|
|
30
|
-
asGrid?: boolean;
|
|
31
|
-
lgWidth?: string;
|
|
32
|
-
mdWidth?: string;
|
|
33
|
-
index?: number;
|
|
34
|
-
}
|
|
1
|
+
import React from "react";
|
|
2
|
+
|
|
3
|
+
export interface BlogCardImageProps {
|
|
4
|
+
src: string;
|
|
5
|
+
alt: string;
|
|
6
|
+
width: number;
|
|
7
|
+
height: number;
|
|
8
|
+
className?: string;
|
|
9
|
+
sizes?: string;
|
|
10
|
+
loading?: "lazy" | "eager";
|
|
11
|
+
decoding?: "async" | "auto" | "sync";
|
|
12
|
+
[key: string]: unknown;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface BlogCardProps {
|
|
16
|
+
href: string;
|
|
17
|
+
title?: string;
|
|
18
|
+
description?: string;
|
|
19
|
+
date?: string;
|
|
20
|
+
category?: string;
|
|
21
|
+
image?: {
|
|
22
|
+
src: string;
|
|
23
|
+
alt: string;
|
|
24
|
+
width: number;
|
|
25
|
+
height: number;
|
|
26
|
+
};
|
|
27
|
+
/** Pass next/image's Image component here to enable image optimization */
|
|
28
|
+
imageComponent?: React.ComponentType<BlogCardImageProps>;
|
|
29
|
+
readMoreText?: string;
|
|
30
|
+
asGrid?: boolean;
|
|
31
|
+
lgWidth?: string;
|
|
32
|
+
mdWidth?: string;
|
|
33
|
+
index?: number;
|
|
34
|
+
}
|
|
@@ -1,119 +1,119 @@
|
|
|
1
|
-
import React from "react";
|
|
2
|
-
import { Button } from "../../button";
|
|
3
|
-
import { FloatingImageCardProps } from "./types";
|
|
4
|
-
|
|
5
|
-
import { NextImage } from "@shared/components/next-image";
|
|
6
|
-
import { Text } from "@shared/components/text";
|
|
7
|
-
import { cx } from "@shared/utils";
|
|
8
|
-
|
|
9
|
-
const alignmentClasses: Record<string, string> = {
|
|
10
|
-
left: "flex w-full justify-start",
|
|
11
|
-
right: "flex w-full justify-end",
|
|
12
|
-
center: "flex w-full justify-center",
|
|
13
|
-
};
|
|
14
|
-
|
|
15
|
-
export const FloatingImageCard: React.FC<FloatingImageCardProps> = ({
|
|
16
|
-
card,
|
|
17
|
-
lgWidth,
|
|
18
|
-
mdWidth,
|
|
19
|
-
}) => {
|
|
20
|
-
const {
|
|
21
|
-
image,
|
|
22
|
-
title,
|
|
23
|
-
header,
|
|
24
|
-
body,
|
|
25
|
-
cta,
|
|
26
|
-
ctaAlignment,
|
|
27
|
-
imageLink,
|
|
28
|
-
imageClassName,
|
|
29
|
-
className,
|
|
30
|
-
isPromoBarImg = false,
|
|
31
|
-
shadow = false,
|
|
32
|
-
} = card;
|
|
33
|
-
|
|
34
|
-
const ctaAlignmentClass = ctaAlignment
|
|
35
|
-
? alignmentClasses[ctaAlignment]
|
|
36
|
-
: alignmentClasses.left;
|
|
37
|
-
const imageLinkAvailableClass = imageLink ? "cursor-pointer" : "";
|
|
38
|
-
|
|
39
|
-
const handleImageClick = imageLink
|
|
40
|
-
? () => {
|
|
41
|
-
if (typeof window !== "undefined") {
|
|
42
|
-
window.open(imageLink, "_blank");
|
|
43
|
-
}
|
|
44
|
-
}
|
|
45
|
-
: undefined;
|
|
46
|
-
|
|
47
|
-
if (isPromoBarImg) {
|
|
48
|
-
return image?.href ? (
|
|
49
|
-
<NextImage
|
|
50
|
-
src={image.href}
|
|
51
|
-
alt={image.title ?? title ?? "promo-bar-image"}
|
|
52
|
-
width={148}
|
|
53
|
-
height={40}
|
|
54
|
-
onClick={handleImageClick}
|
|
55
|
-
className={imageLinkAvailableClass}
|
|
56
|
-
/>
|
|
57
|
-
) : null;
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
return (
|
|
61
|
-
<section
|
|
62
|
-
aria-labelledby="Image Card"
|
|
63
|
-
className={cx(
|
|
64
|
-
"callout-card relative flex h-full w-full flex-col gap-4 p-0 md:items-start lg:gap-6",
|
|
65
|
-
shadow && "rounded-3xl shadow-card",
|
|
66
|
-
mdWidth,
|
|
67
|
-
lgWidth,
|
|
68
|
-
className
|
|
69
|
-
)}
|
|
70
|
-
>
|
|
71
|
-
{image?.href ? (
|
|
72
|
-
<div
|
|
73
|
-
className={cx(
|
|
74
|
-
"relative aspect-video w-full overflow-hidden rounded-[40px] sm:mb-1 md:flex-none lg:mb-0",
|
|
75
|
-
imageClassName
|
|
76
|
-
)}
|
|
77
|
-
>
|
|
78
|
-
<NextImage
|
|
79
|
-
src={image.href}
|
|
80
|
-
alt={image.title ?? title ?? "card-image"}
|
|
81
|
-
width={image.width || 400}
|
|
82
|
-
height={image.height || 285}
|
|
83
|
-
onClick={handleImageClick}
|
|
84
|
-
className={cx(
|
|
85
|
-
"h-full max-h-[285px] w-full object-cover",
|
|
86
|
-
imageLinkAvailableClass
|
|
87
|
-
)}
|
|
88
|
-
/>
|
|
89
|
-
</div>
|
|
90
|
-
) : null}
|
|
91
|
-
|
|
92
|
-
{header ? (
|
|
93
|
-
<Text as="h6" className="heading6 font-black text-text lg:pb-2">
|
|
94
|
-
{header}
|
|
95
|
-
</Text>
|
|
96
|
-
) : null}
|
|
97
|
-
|
|
98
|
-
{body ? (
|
|
99
|
-
<Text as="div" className="body2 mb-0 flex-1 text-text">
|
|
100
|
-
{body}
|
|
101
|
-
</Text>
|
|
102
|
-
) : null}
|
|
103
|
-
|
|
104
|
-
<div className={cx(ctaAlignmentClass, "mt-auto")}>
|
|
105
|
-
{cta ? (
|
|
106
|
-
<Button
|
|
107
|
-
linkVariant="unstyled"
|
|
108
|
-
linkClassName="label1 flex items-center gap-2 text-text"
|
|
109
|
-
{...cta}
|
|
110
|
-
iconName="expand_circle_right"
|
|
111
|
-
iconFill={1}
|
|
112
|
-
/>
|
|
113
|
-
) : null}
|
|
114
|
-
</div>
|
|
115
|
-
</section>
|
|
116
|
-
);
|
|
117
|
-
};
|
|
118
|
-
|
|
119
|
-
export default FloatingImageCard;
|
|
1
|
+
import React from "react";
|
|
2
|
+
import { Button } from "../../button";
|
|
3
|
+
import { FloatingImageCardProps } from "./types";
|
|
4
|
+
|
|
5
|
+
import { NextImage } from "@shared/components/next-image";
|
|
6
|
+
import { Text } from "@shared/components/text";
|
|
7
|
+
import { cx } from "@shared/utils";
|
|
8
|
+
|
|
9
|
+
const alignmentClasses: Record<string, string> = {
|
|
10
|
+
left: "flex w-full justify-start",
|
|
11
|
+
right: "flex w-full justify-end",
|
|
12
|
+
center: "flex w-full justify-center",
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
export const FloatingImageCard: React.FC<FloatingImageCardProps> = ({
|
|
16
|
+
card,
|
|
17
|
+
lgWidth,
|
|
18
|
+
mdWidth,
|
|
19
|
+
}) => {
|
|
20
|
+
const {
|
|
21
|
+
image,
|
|
22
|
+
title,
|
|
23
|
+
header,
|
|
24
|
+
body,
|
|
25
|
+
cta,
|
|
26
|
+
ctaAlignment,
|
|
27
|
+
imageLink,
|
|
28
|
+
imageClassName,
|
|
29
|
+
className,
|
|
30
|
+
isPromoBarImg = false,
|
|
31
|
+
shadow = false,
|
|
32
|
+
} = card;
|
|
33
|
+
|
|
34
|
+
const ctaAlignmentClass = ctaAlignment
|
|
35
|
+
? alignmentClasses[ctaAlignment]
|
|
36
|
+
: alignmentClasses.left;
|
|
37
|
+
const imageLinkAvailableClass = imageLink ? "cursor-pointer" : "";
|
|
38
|
+
|
|
39
|
+
const handleImageClick = imageLink
|
|
40
|
+
? () => {
|
|
41
|
+
if (typeof window !== "undefined") {
|
|
42
|
+
window.open(imageLink, "_blank");
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
: undefined;
|
|
46
|
+
|
|
47
|
+
if (isPromoBarImg) {
|
|
48
|
+
return image?.href ? (
|
|
49
|
+
<NextImage
|
|
50
|
+
src={image.href}
|
|
51
|
+
alt={image.title ?? title ?? "promo-bar-image"}
|
|
52
|
+
width={148}
|
|
53
|
+
height={40}
|
|
54
|
+
onClick={handleImageClick}
|
|
55
|
+
className={imageLinkAvailableClass}
|
|
56
|
+
/>
|
|
57
|
+
) : null;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
return (
|
|
61
|
+
<section
|
|
62
|
+
aria-labelledby="Image Card"
|
|
63
|
+
className={cx(
|
|
64
|
+
"callout-card relative flex h-full w-full flex-col gap-4 p-0 md:items-start lg:gap-6",
|
|
65
|
+
shadow && "rounded-3xl shadow-card",
|
|
66
|
+
mdWidth,
|
|
67
|
+
lgWidth,
|
|
68
|
+
className
|
|
69
|
+
)}
|
|
70
|
+
>
|
|
71
|
+
{image?.href ? (
|
|
72
|
+
<div
|
|
73
|
+
className={cx(
|
|
74
|
+
"relative aspect-video w-full overflow-hidden rounded-[40px] sm:mb-1 md:flex-none lg:mb-0",
|
|
75
|
+
imageClassName
|
|
76
|
+
)}
|
|
77
|
+
>
|
|
78
|
+
<NextImage
|
|
79
|
+
src={image.href}
|
|
80
|
+
alt={image.title ?? title ?? "card-image"}
|
|
81
|
+
width={image.width || 400}
|
|
82
|
+
height={image.height || 285}
|
|
83
|
+
onClick={handleImageClick}
|
|
84
|
+
className={cx(
|
|
85
|
+
"h-full max-h-[285px] w-full object-cover",
|
|
86
|
+
imageLinkAvailableClass
|
|
87
|
+
)}
|
|
88
|
+
/>
|
|
89
|
+
</div>
|
|
90
|
+
) : null}
|
|
91
|
+
|
|
92
|
+
{header ? (
|
|
93
|
+
<Text as="h6" className="heading6 font-black text-text lg:pb-2">
|
|
94
|
+
{header}
|
|
95
|
+
</Text>
|
|
96
|
+
) : null}
|
|
97
|
+
|
|
98
|
+
{body ? (
|
|
99
|
+
<Text as="div" className="body2 mb-0 flex-1 text-text">
|
|
100
|
+
{body}
|
|
101
|
+
</Text>
|
|
102
|
+
) : null}
|
|
103
|
+
|
|
104
|
+
<div className={cx(ctaAlignmentClass, "mt-auto")}>
|
|
105
|
+
{cta ? (
|
|
106
|
+
<Button
|
|
107
|
+
linkVariant="unstyled"
|
|
108
|
+
linkClassName="label1 flex items-center gap-2 text-text"
|
|
109
|
+
{...cta}
|
|
110
|
+
iconName="expand_circle_right"
|
|
111
|
+
iconFill={1}
|
|
112
|
+
/>
|
|
113
|
+
) : null}
|
|
114
|
+
</div>
|
|
115
|
+
</section>
|
|
116
|
+
);
|
|
117
|
+
};
|
|
118
|
+
|
|
119
|
+
export default FloatingImageCard;
|
|
@@ -1,30 +1,30 @@
|
|
|
1
|
-
import type { ReactNode } from "react";
|
|
2
|
-
import type { ButtonProps } from "../../button/types";
|
|
3
|
-
|
|
4
|
-
export type FloatingImageAsset = {
|
|
5
|
-
href: string;
|
|
6
|
-
title?: string;
|
|
7
|
-
width?: number;
|
|
8
|
-
height?: number;
|
|
9
|
-
};
|
|
10
|
-
|
|
11
|
-
export type FloatingImageCardItem = {
|
|
12
|
-
title?: string;
|
|
13
|
-
header?: string;
|
|
14
|
-
body?: ReactNode;
|
|
15
|
-
cta?: ButtonProps;
|
|
16
|
-
ctaAlignment?: "left" | "right" | "center";
|
|
17
|
-
image?: FloatingImageAsset;
|
|
18
|
-
imageLink?: string;
|
|
19
|
-
imageClassName?: string;
|
|
20
|
-
className?: string;
|
|
21
|
-
isPromoBarImg?: boolean;
|
|
22
|
-
/** Applies `shadow-card` to the outer card element. */
|
|
23
|
-
shadow?: boolean;
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
export type FloatingImageCardProps = {
|
|
27
|
-
card: FloatingImageCardItem;
|
|
28
|
-
lgWidth?: string;
|
|
29
|
-
mdWidth?: string;
|
|
30
|
-
};
|
|
1
|
+
import type { ReactNode } from "react";
|
|
2
|
+
import type { ButtonProps } from "../../button/types";
|
|
3
|
+
|
|
4
|
+
export type FloatingImageAsset = {
|
|
5
|
+
href: string;
|
|
6
|
+
title?: string;
|
|
7
|
+
width?: number;
|
|
8
|
+
height?: number;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
export type FloatingImageCardItem = {
|
|
12
|
+
title?: string;
|
|
13
|
+
header?: string;
|
|
14
|
+
body?: ReactNode;
|
|
15
|
+
cta?: ButtonProps;
|
|
16
|
+
ctaAlignment?: "left" | "right" | "center";
|
|
17
|
+
image?: FloatingImageAsset;
|
|
18
|
+
imageLink?: string;
|
|
19
|
+
imageClassName?: string;
|
|
20
|
+
className?: string;
|
|
21
|
+
isPromoBarImg?: boolean;
|
|
22
|
+
/** Applies `shadow-card` to the outer card element. */
|
|
23
|
+
shadow?: boolean;
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
export type FloatingImageCardProps = {
|
|
27
|
+
card: FloatingImageCardItem;
|
|
28
|
+
lgWidth?: string;
|
|
29
|
+
mdWidth?: string;
|
|
30
|
+
};
|