@windstream/react-shared-components 0.1.44 → 0.1.45

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.
Files changed (186) hide show
  1. package/README.md +635 -635
  2. package/dist/contentful/index.d.ts +18 -5
  3. package/dist/contentful/index.esm.js +1 -1
  4. package/dist/contentful/index.esm.js.map +1 -1
  5. package/dist/contentful/index.js +1 -1
  6. package/dist/contentful/index.js.map +1 -1
  7. package/dist/core.d.ts +1 -1
  8. package/dist/index.d.ts +3 -3
  9. package/dist/index.js.map +1 -1
  10. package/dist/styles.css +1 -1
  11. package/package.json +185 -185
  12. package/src/components/accordion/Accordion.stories.tsx +230 -230
  13. package/src/components/accordion/index.tsx +70 -70
  14. package/src/components/accordion/types.ts +12 -12
  15. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  16. package/src/components/alert-card/index.tsx +41 -41
  17. package/src/components/alert-card/types.ts +13 -13
  18. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  19. package/src/components/brand-button/helpers.ts +35 -35
  20. package/src/components/brand-button/index.tsx +120 -120
  21. package/src/components/brand-button/types.ts +38 -38
  22. package/src/components/button/Button.stories.tsx +108 -108
  23. package/src/components/button/index.tsx +27 -27
  24. package/src/components/button/types.ts +14 -14
  25. package/src/components/call-button/CallButton.stories.tsx +324 -324
  26. package/src/components/call-button/index.tsx +86 -86
  27. package/src/components/call-button/types.ts +11 -11
  28. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  29. package/src/components/checkbox/index.tsx +197 -197
  30. package/src/components/checkbox/types.ts +27 -27
  31. package/src/components/checklist/Checklist.stories.tsx +150 -150
  32. package/src/components/checklist/index.tsx +61 -61
  33. package/src/components/checklist/types.ts +17 -17
  34. package/src/components/collapse/Collapse.stories.tsx +255 -255
  35. package/src/components/collapse/index.tsx +46 -46
  36. package/src/components/collapse/types.ts +6 -6
  37. package/src/components/divider/Divider.stories.tsx +205 -205
  38. package/src/components/divider/index.tsx +22 -22
  39. package/src/components/divider/type.ts +3 -3
  40. package/src/components/image/Image.stories.tsx +113 -113
  41. package/src/components/image/index.tsx +25 -25
  42. package/src/components/image/types.ts +40 -40
  43. package/src/components/input/Input.stories.tsx +325 -325
  44. package/src/components/input/index.tsx +177 -177
  45. package/src/components/input/types.ts +37 -37
  46. package/src/components/link/Link.stories.tsx +163 -163
  47. package/src/components/link/index.tsx +109 -109
  48. package/src/components/link/types.ts +25 -25
  49. package/src/components/list/List.stories.tsx +272 -272
  50. package/src/components/list/index.tsx +88 -88
  51. package/src/components/list/list-item/index.tsx +38 -38
  52. package/src/components/list/list-item/types.ts +13 -13
  53. package/src/components/list/types.ts +29 -29
  54. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  55. package/src/components/material-icon/constants.ts +98 -98
  56. package/src/components/material-icon/index.tsx +47 -47
  57. package/src/components/material-icon/types.ts +31 -31
  58. package/src/components/modal/Modal.stories.tsx +171 -171
  59. package/src/components/modal/index.tsx +164 -164
  60. package/src/components/modal/types.ts +24 -24
  61. package/src/components/next-image/index.tsx +54 -54
  62. package/src/components/next-image/types.ts +1 -1
  63. package/src/components/pagination/index.tsx +100 -100
  64. package/src/components/pagination/types.ts +6 -6
  65. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  66. package/src/components/radio-button/index.tsx +75 -75
  67. package/src/components/radio-button/types.ts +21 -21
  68. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  69. package/src/components/see-more/index.tsx +44 -44
  70. package/src/components/see-more/types.ts +4 -4
  71. package/src/components/select/Select.stories.tsx +411 -411
  72. package/src/components/select/index.tsx +155 -155
  73. package/src/components/select/types.ts +36 -36
  74. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  75. package/src/components/select-plan-button/index.tsx +63 -63
  76. package/src/components/select-plan-button/types.ts +17 -17
  77. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  78. package/src/components/skeleton/index.tsx +61 -61
  79. package/src/components/skeleton/types.ts +4 -4
  80. package/src/components/spinner/Spinner.stories.tsx +335 -335
  81. package/src/components/spinner/index.tsx +44 -44
  82. package/src/components/spinner/types.ts +5 -5
  83. package/src/components/text/Text.stories.tsx +321 -321
  84. package/src/components/text/index.tsx +25 -25
  85. package/src/components/text/types.ts +45 -45
  86. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  87. package/src/components/tooltip/index.tsx +74 -74
  88. package/src/components/tooltip/types.ts +7 -7
  89. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  90. package/src/components/view-cart-button/index.tsx +42 -42
  91. package/src/components/view-cart-button/types.ts +5 -5
  92. package/src/contentful/blocks/accordion/Accordion.stories.tsx +34 -29
  93. package/src/contentful/blocks/accordion/index.tsx +112 -62
  94. package/src/contentful/blocks/accordion/types.ts +34 -17
  95. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  96. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  97. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +70 -70
  98. package/src/contentful/blocks/anchored-bottom-banner/types.ts +10 -10
  99. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  100. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  101. package/src/contentful/blocks/breadcrumbs/index.tsx +51 -51
  102. package/src/contentful/blocks/breadcrumbs/types.ts +5 -5
  103. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  104. package/src/contentful/blocks/button/index.tsx +130 -130
  105. package/src/contentful/blocks/button/types.ts +39 -39
  106. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  107. package/src/contentful/blocks/callout/index.tsx +88 -88
  108. package/src/contentful/blocks/callout/types.ts +15 -15
  109. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  110. package/src/contentful/blocks/cards/blog-card/index.tsx +110 -110
  111. package/src/contentful/blocks/cards/blog-card/types.ts +18 -18
  112. package/src/contentful/blocks/cards/index.tsx +13 -13
  113. package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
  114. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  115. package/src/contentful/blocks/cards/simple-card/index.tsx +89 -89
  116. package/src/contentful/blocks/cards/simple-card/types.ts +28 -28
  117. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  118. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  119. package/src/contentful/blocks/cards/types.ts +1 -1
  120. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  121. package/src/contentful/blocks/carousel/helper.tsx +440 -440
  122. package/src/contentful/blocks/carousel/index.tsx +85 -85
  123. package/src/contentful/blocks/carousel/types.ts +144 -144
  124. package/src/contentful/blocks/comparison-table/index.tsx +27 -27
  125. package/src/contentful/blocks/comparison-table/types.ts +6 -6
  126. package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
  127. package/src/contentful/blocks/cookiebanner/type.ts +7 -7
  128. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  129. package/src/contentful/blocks/cta-callout/index.tsx +71 -69
  130. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  131. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  132. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  133. package/src/contentful/blocks/email-input-block/index.tsx +117 -117
  134. package/src/contentful/blocks/email-input-block/types.ts +16 -16
  135. package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
  136. package/src/contentful/blocks/find-kinetic/index.tsx +130 -130
  137. package/src/contentful/blocks/find-kinetic/types.ts +19 -19
  138. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  139. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  140. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  141. package/src/contentful/blocks/footer/Footer.stories.tsx +30 -30
  142. package/src/contentful/blocks/footer/index.tsx +91 -91
  143. package/src/contentful/blocks/footer/types.ts +13 -13
  144. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  145. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  146. package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
  147. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  148. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  149. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  150. package/src/contentful/blocks/modal/constants.ts +53 -53
  151. package/src/contentful/blocks/modal/index.tsx +107 -107
  152. package/src/contentful/blocks/modal/types.ts +12 -12
  153. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +113 -113
  154. package/src/contentful/blocks/navigation/index.tsx +394 -394
  155. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  156. package/src/contentful/blocks/navigation/types.ts +41 -41
  157. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  158. package/src/contentful/blocks/primary-hero/index.tsx +236 -236
  159. package/src/contentful/blocks/primary-hero/types.ts +37 -37
  160. package/src/contentful/blocks/search-block/index.tsx +90 -90
  161. package/src/contentful/blocks/search-block/types.ts +15 -15
  162. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  163. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  164. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  165. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  166. package/src/contentful/blocks/text/index.tsx +12 -12
  167. package/src/contentful/blocks/text/types.ts +1 -1
  168. package/src/contentful/index.ts +99 -99
  169. package/src/hooks/contentful/use-contentful-rich-text.tsx +310 -310
  170. package/src/hooks/contentful/use-processed-check-list.ts +63 -63
  171. package/src/hooks/use-body-scroll-lock.ts +34 -34
  172. package/src/hooks/use-carousel-swipe.ts +264 -264
  173. package/src/hooks/use-outside-click.ts +17 -17
  174. package/src/index.ts +101 -101
  175. package/src/next/index.ts +5 -5
  176. package/src/setupTests.ts +46 -46
  177. package/src/stories/DocsTemplate.tsx +24 -24
  178. package/src/styles/globals.css +343 -343
  179. package/src/types/global.d.ts +9 -9
  180. package/src/types/micro-components.ts +99 -99
  181. package/src/types/utm.ts +49 -49
  182. package/src/utils/contentful/to-document.ts +24 -24
  183. package/src/utils/cookie.ts +84 -84
  184. package/src/utils/cx.ts +49 -49
  185. package/src/utils/index.ts +38 -38
  186. package/src/utils/utm.ts +221 -221
@@ -1,42 +1,42 @@
1
- import { Button } from "../button";
2
- import { MaterialIcon } from "../material-icon";
3
- import { Text } from "../text";
4
- import { ViewCartButtonProps } from "./types";
5
-
6
- import { cx } from "@shared/utils";
7
-
8
- const baseClasses =
9
- "btn-medium rounded-button pl-15 pr-15 inline-flex gap-2 items-center justify-center outline-none focus:ring-2 focus:ring-offset-2 cursor-pointer transition-colors duration-200 align-top border-2 border-border-secondary-on-bg-fill bg-bg text-text focus:ring-bg-surface-inverse";
10
- export const ViewCartButton: React.FC<ViewCartButtonProps> = props => {
11
- const { cartTotalText, onClick, isOpen, ...rest } = props;
12
- return (
13
- <Button
14
- className={cx(
15
- baseClasses,
16
- "btn-small w-full border-border pl-6 pr-4 md:btn-medium"
17
- )}
18
- onClick={onClick}
19
- {...rest}
20
- >
21
- <div className="flex w-full items-center justify-center gap-2">
22
- <div className="flex flex-col items-center justify-center">
23
- <Text className="label4 -mb-1 text-text">View cart</Text>
24
- <Text className="label3 text-text md:label1" data-testid="cart-total">
25
- {cartTotalText}
26
- </Text>
27
- </div>
28
- <MaterialIcon
29
- name="keyboard_arrow_up"
30
- size={24}
31
- fill={0}
32
- className={cx(
33
- "h-6 w-6 transition-transform",
34
- isOpen ? "rotate-180" : "rotate-0"
35
- )}
36
- />
37
- </div>
38
- </Button>
39
- );
40
- };
41
-
42
- export type { ViewCartButtonProps } from "./types";
1
+ import { Button } from "../button";
2
+ import { MaterialIcon } from "../material-icon";
3
+ import { Text } from "../text";
4
+ import { ViewCartButtonProps } from "./types";
5
+
6
+ import { cx } from "@shared/utils";
7
+
8
+ const baseClasses =
9
+ "btn-medium rounded-button pl-15 pr-15 inline-flex gap-2 items-center justify-center outline-none focus:ring-2 focus:ring-offset-2 cursor-pointer transition-colors duration-200 align-top border-2 border-border-secondary-on-bg-fill bg-bg text-text focus:ring-bg-surface-inverse";
10
+ export const ViewCartButton: React.FC<ViewCartButtonProps> = props => {
11
+ const { cartTotalText, onClick, isOpen, ...rest } = props;
12
+ return (
13
+ <Button
14
+ className={cx(
15
+ baseClasses,
16
+ "btn-small w-full border-border pl-6 pr-4 md:btn-medium"
17
+ )}
18
+ onClick={onClick}
19
+ {...rest}
20
+ >
21
+ <div className="flex w-full items-center justify-center gap-2">
22
+ <div className="flex flex-col items-center justify-center">
23
+ <Text className="label4 -mb-1 text-text">View cart</Text>
24
+ <Text className="label3 text-text md:label1" data-testid="cart-total">
25
+ {cartTotalText}
26
+ </Text>
27
+ </div>
28
+ <MaterialIcon
29
+ name="keyboard_arrow_up"
30
+ size={24}
31
+ fill={0}
32
+ className={cx(
33
+ "h-6 w-6 transition-transform",
34
+ isOpen ? "rotate-180" : "rotate-0"
35
+ )}
36
+ />
37
+ </div>
38
+ </Button>
39
+ );
40
+ };
41
+
42
+ export type { ViewCartButtonProps } from "./types";
@@ -1,5 +1,5 @@
1
- export type ViewCartButtonProps = {
2
- cartTotalText: string;
3
- onClick: () => void;
4
- isOpen: boolean;
5
- } & React.ButtonHTMLAttributes<HTMLButtonElement>;
1
+ export type ViewCartButtonProps = {
2
+ cartTotalText: string;
3
+ onClick: () => void;
4
+ isOpen: boolean;
5
+ } & React.ButtonHTMLAttributes<HTMLButtonElement>;
@@ -1,29 +1,34 @@
1
- import { Accordion } from "./index";
2
-
3
- import { DocsPage } from "@shared/stories/DocsTemplate";
4
- import type { Meta, StoryObj } from "@storybook/react";
5
-
6
- const meta: Meta<typeof Accordion> = {
7
- title: "Contentful Blocks/Accordion",
8
- component: Accordion,
9
- tags: ["autodocs"],
10
- parameters: {
11
- layout: "centered",
12
- docs: {
13
- page: DocsPage,
14
- description: {
15
- component: "Contentful accordion block.",
16
- },
17
- },
18
- },
19
- args: {
20
- background: "white",
21
- enableHeading: false,
22
- items: [],
23
- maxWidth: true,
24
- title: "test",
25
- },
26
- };
27
- export default meta;
28
- type Story = StoryObj<typeof meta>;
29
- export const Default: Story = {};
1
+ import { Accordion } from "./index";
2
+
3
+ import { DocsPage } from "@shared/stories/DocsTemplate";
4
+ import type { Meta, StoryObj } from "@storybook/react";
5
+
6
+ const meta: Meta<typeof Accordion> = {
7
+ title: "Contentful Blocks/Accordion",
8
+ component: Accordion,
9
+ tags: ["autodocs"],
10
+ parameters: {
11
+ layout: "centered",
12
+ docs: {
13
+ page: DocsPage,
14
+ description: {
15
+ component: "Contentful accordion block.",
16
+ },
17
+ },
18
+ },
19
+ args: {
20
+ background: "white",
21
+ enableHeading: false,
22
+ items: [],
23
+ maxWidth: true,
24
+ title: "test",
25
+ anchorId: undefined,
26
+ initialExpandedItems: [],
27
+ hashAutoExpand: false,
28
+ descriptionMaxWidth: "max-w-[980px]",
29
+ itemContainerClassName: undefined,
30
+ },
31
+ };
32
+ export default meta;
33
+ type Story = StoryObj<typeof meta>;
34
+ export const Default: Story = {};
@@ -1,62 +1,112 @@
1
- import React from "react";
2
- import { AccordionProps, ThemeKey } from "./types";
3
-
4
- import { Accordion as AccordionComponent } from "@shared/components/accordion";
5
- import { Divider } from "@shared/components/divider";
6
- import { Text } from "@shared/components/text";
7
-
8
- export const Accordion: React.FC<AccordionProps> = ({
9
- items,
10
- title,
11
- background = "white",
12
- enableHeading = false,
13
- maxWidth = true,
14
- }) => {
15
- const bgColorClasses: Record<ThemeKey, string> = {
16
- blue: "bg-[#07B2E2]",
17
- green: "bg-[#26B170]",
18
- yellow: "bg-[#F5FF1E]",
19
- purple: "bg-[#931D69]",
20
- white: "bg-white",
21
- navy: "bg-[#00002D]",
22
- };
23
- return (
24
- <div className={`${bgColorClasses[background]} component-container`}>
25
- <div
26
- className={`${maxWidth ? "max-w-120 xl:mx-auto" : ""} mx-5 my-8 lg:mx-20 lg:my-18`}
27
- >
28
- <div>
29
- <Text
30
- as={enableHeading ? "h1" : "h2"}
31
- className="heading2 text-left text-text md:heading1 lg:text-center"
32
- >
33
- {title}
34
- </Text>
35
- </div>
36
- <div className="question-answer-section mt-10 lg:mt-16">
37
- {items?.map((item: any, index: number) => (
38
- <div key={`item-${index}`}>
39
- <AccordionComponent
40
- title={item.title}
41
- containerClassName="border-none"
42
- buttonClassName="p-0 label1"
43
- titleClassName="label3 md:label1 text-text"
44
- className="p-0 pt-6 text-text"
45
- iconClassName="text-icon"
46
- borderRadiusNone={true}
47
- openOnlyOnDesktop={false}
48
- >
49
- <Text as="div">{item.description}</Text>
50
- </AccordionComponent>
51
- {index < items.length - 1 && (
52
- <Divider key={`divider-${index}`} className="my-6 md:my-10" />
53
- )}
54
- </div>
55
- ))}
56
- </div>
57
- </div>
58
- </div>
59
- );
60
- };
61
-
62
- export default Accordion;
1
+ "use client";
2
+
3
+ import React, { useEffect, useState } from "react";
4
+ import { AccordionItem, AccordionProps, ThemeKey } from "./types";
5
+
6
+ import { Accordion as AccordionComponent } from "@shared/components/accordion";
7
+ import { Divider } from "@shared/components/divider";
8
+ import { Text } from "@shared/components/text";
9
+
10
+ export const Accordion: React.FC<AccordionProps> = ({
11
+ items,
12
+ title,
13
+ anchorId,
14
+ background = "white",
15
+ enableHeading = false,
16
+ maxWidth = true,
17
+ initialExpandedItems: initialExpandedItemsProp = [],
18
+ hashAutoExpand = false,
19
+ descriptionMaxWidth = "max-w-[980px]",
20
+ itemContainerClassName,
21
+ }) => {
22
+ const [expandedItems, setExpandedItems] = useState<string[]>(
23
+ initialExpandedItemsProp
24
+ );
25
+
26
+ useEffect(() => {
27
+ if (hashAutoExpand && typeof window !== "undefined") {
28
+ const hash = window.location.hash;
29
+ if (hash) {
30
+ const expandItemId = hash.substring(hash.indexOf("#") + 1);
31
+ setExpandedItems(prev => {
32
+ if (prev.includes(expandItemId)) return prev;
33
+ return [...prev, expandItemId];
34
+ });
35
+ }
36
+ }
37
+ }, [hashAutoExpand]);
38
+
39
+ const handleToggle = (itemAnchorId: string) => {
40
+ setExpandedItems(prev =>
41
+ prev.includes(itemAnchorId)
42
+ ? prev.filter(id => id !== itemAnchorId)
43
+ : [...prev, itemAnchorId]
44
+ );
45
+ };
46
+
47
+ const bgColorClasses: Record<ThemeKey, string> = {
48
+ blue: "bg-[#07B2E2]",
49
+ green: "bg-[#26B170]",
50
+ yellow: "bg-[#F5FF1E]",
51
+ purple: "bg-[#931D69]",
52
+ white: "bg-white",
53
+ navy: "bg-[#00002D]",
54
+ cream500: "bg-[#FFFEEF]",
55
+ };
56
+
57
+ return (
58
+ <section
59
+ id={anchorId}
60
+ className={`${bgColorClasses[background]} flex flex-col px-2 pb-20 pt-4 md:px-10`}
61
+ >
62
+ <div className={`${maxWidth ? "max-w-120 xl:mx-auto" : ""}`}>
63
+ <Text
64
+ as={enableHeading ? "h1" : "h2"}
65
+ className="heading2 mb-10 pt-12 text-left text-text md:heading1 lg:text-center"
66
+ >
67
+ {title}
68
+ </Text>
69
+ <div className={`pt-6 md:pt-10 ${itemContainerClassName ?? ""}`}>
70
+ {items?.map((item: AccordionItem, index: number) => {
71
+ const itemKey = item.anchorId || `item-${index}`;
72
+ const isOpen = item.anchorId
73
+ ? expandedItems.includes(item.anchorId)
74
+ : undefined;
75
+
76
+ return (
77
+ <div
78
+ key={itemKey}
79
+ id={item.anchorId}
80
+ className="snap-start scroll-mt-48 lg:scroll-mt-60"
81
+ >
82
+ <AccordionComponent
83
+ title={item.title}
84
+ defaultOpen={isOpen}
85
+ containerClassName="border-none"
86
+ buttonClassName="p-0 label1"
87
+ titleClassName="label3 md:label1 text-text"
88
+ className="p-0 pt-6 text-text"
89
+ iconClassName="text-icon"
90
+ borderRadiusNone={true}
91
+ openOnlyOnDesktop={false}
92
+ {...(item.anchorId && {
93
+ onClick: () => handleToggle(item.anchorId!),
94
+ })}
95
+ >
96
+ <div className={descriptionMaxWidth}>
97
+ <Text as="div">{item.description}</Text>
98
+ </div>
99
+ </AccordionComponent>
100
+ {index < items.length - 1 && (
101
+ <Divider key={`divider-${index}`} className="my-6 md:my-10" />
102
+ )}
103
+ </div>
104
+ );
105
+ })}
106
+ </div>
107
+ </div>
108
+ </section>
109
+ );
110
+ };
111
+
112
+ export default Accordion;
@@ -1,17 +1,34 @@
1
- import React from "react";
2
-
3
- export type AccordionProps = {
4
- title: string;
5
- enableHeading?: boolean;
6
- background?: "blue" | "green" | "navy" | "purple" | "white" | "yellow";
7
- items: { title: string; description: React.ReactNode }[];
8
- maxWidth?: boolean;
9
- };
10
-
11
- export type ThemeKey =
12
- | "blue"
13
- | "green"
14
- | "yellow"
15
- | "purple"
16
- | "white"
17
- | "navy";
1
+ import React from "react";
2
+
3
+ export type AccordionItem = {
4
+ anchorId?: string;
5
+ title: string;
6
+ description: React.ReactNode;
7
+ };
8
+
9
+ export type AccordionProps = {
10
+ /** Section-level anchor ID applied to the outer <section> element */
11
+ anchorId?: string;
12
+ title: string;
13
+ enableHeading?: boolean;
14
+ background?: ThemeKey;
15
+ items: AccordionItem[];
16
+ maxWidth?: boolean;
17
+ /** Array of anchorIds that should be expanded on mount */
18
+ initialExpandedItems?: string[];
19
+ /** Enable URL hash auto-expand on mount */
20
+ hashAutoExpand?: boolean;
21
+ /** Max-width class for item descriptions (default: "max-w-[980px]") */
22
+ descriptionMaxWidth?: string;
23
+ /** Custom class for the accordion list wrapper */
24
+ itemContainerClassName?: string;
25
+ };
26
+
27
+ export type ThemeKey =
28
+ | "blue"
29
+ | "green"
30
+ | "yellow"
31
+ | "purple"
32
+ | "white"
33
+ | "navy"
34
+ | "cream500";
@@ -1,52 +1,52 @@
1
- import { FC } from "react";
2
- import { AddressInputBannerProps } from "./types";
3
-
4
- import { Text } from "@shared/components/text";
5
- import { cx } from "@shared/utils";
6
-
7
- const variantStyles: Record<string, { bg: string; text: string }> = {
8
- yellow: { bg: "bg-fill-brand-accent", text: "text" },
9
- white: { bg: "bg-white", text: "text" },
10
- navy: { bg: "bg-bg-fill-inverse", text: "text-inverse" },
11
- green: { bg: "bg-border-success", text: "text-inverse" },
12
- };
13
-
14
- export const AddressInputBanner: FC<AddressInputBannerProps> = props => {
15
- const {
16
- title,
17
- variant = "yellow",
18
- cta,
19
- navHeight,
20
- isVisible,
21
- renderCheckPlans,
22
- } = props;
23
-
24
- const style = variantStyles[variant?.toLowerCase()] || variantStyles.yellow;
25
-
26
- if (!isVisible) return null;
27
-
28
- const ctaLabel = cta?.buttonLabel || "Check plans";
29
- const renderedCheckPlans = renderCheckPlans?.({
30
- ctaText: ctaLabel,
31
- buttonVariant: cta?.buttonVariant,
32
- showButtonAs: "solid",
33
- cta: cta,
34
- });
35
-
36
- return (
37
- <div
38
- style={{
39
- top: `${navHeight}px`,
40
- }}
41
- className={cx(
42
- `sticky left-0 right-0 z-[89] w-full shadow-drop transition-all duration-200 lg:fixed bg-${style.bg} text-${style.text}`,
43
- "flex flex-col items-center justify-center gap-3 p-[10px] lg:flex-row lg:gap-8 lg:px-6 lg:py-[10px]"
44
- )}
45
- >
46
- <Text className="label3 w-full text-center text-text lg:w-auto lg:text-left">
47
- {title}
48
- </Text>
49
- {renderedCheckPlans}
50
- </div>
51
- );
52
- };
1
+ import { FC } from "react";
2
+ import { AddressInputBannerProps } from "./types";
3
+
4
+ import { Text } from "@shared/components/text";
5
+ import { cx } from "@shared/utils";
6
+
7
+ const variantStyles: Record<string, { bg: string; text: string }> = {
8
+ yellow: { bg: "bg-fill-brand-accent", text: "text" },
9
+ white: { bg: "bg-white", text: "text" },
10
+ navy: { bg: "bg-bg-fill-inverse", text: "text-inverse" },
11
+ green: { bg: "bg-border-success", text: "text-inverse" },
12
+ };
13
+
14
+ export const AddressInputBanner: FC<AddressInputBannerProps> = props => {
15
+ const {
16
+ title,
17
+ variant = "yellow",
18
+ cta,
19
+ navHeight,
20
+ isVisible,
21
+ renderCheckPlans,
22
+ } = props;
23
+
24
+ const style = variantStyles[variant?.toLowerCase()] || variantStyles.yellow;
25
+
26
+ if (!isVisible) return null;
27
+
28
+ const ctaLabel = cta?.buttonLabel || "Check plans";
29
+ const renderedCheckPlans = renderCheckPlans?.({
30
+ ctaText: ctaLabel,
31
+ buttonVariant: cta?.buttonVariant,
32
+ showButtonAs: "solid",
33
+ cta: cta,
34
+ });
35
+
36
+ return (
37
+ <div
38
+ style={{
39
+ top: `${navHeight}px`,
40
+ }}
41
+ className={cx(
42
+ `sticky left-0 right-0 z-[89] w-full shadow-drop transition-all duration-200 lg:fixed bg-${style.bg} text-${style.text}`,
43
+ "flex flex-col items-center justify-center gap-3 p-[10px] lg:flex-row lg:gap-8 lg:px-6 lg:py-[10px]"
44
+ )}
45
+ >
46
+ <Text className="label3 w-full text-center text-text lg:w-auto lg:text-left">
47
+ {title}
48
+ </Text>
49
+ {renderedCheckPlans}
50
+ </div>
51
+ );
52
+ };
@@ -1,14 +1,14 @@
1
- import React from "react";
2
-
3
- import { ButtonProps } from "@shared/contentful/blocks/button/types";
4
- import { CheckPlansProps } from "@shared/types/micro-components";
5
-
6
- export interface AddressInputBannerProps {
7
- entryName?: string;
8
- title: string;
9
- variant?: string;
10
- cta?: ButtonProps;
11
- navHeight?: number;
12
- isVisible?: boolean;
13
- renderCheckPlans?: (overrides?: CheckPlansProps) => React.ReactNode;
14
- }
1
+ import React from "react";
2
+
3
+ import { ButtonProps } from "@shared/contentful/blocks/button/types";
4
+ import { CheckPlansProps } from "@shared/types/micro-components";
5
+
6
+ export interface AddressInputBannerProps {
7
+ entryName?: string;
8
+ title: string;
9
+ variant?: string;
10
+ cta?: ButtonProps;
11
+ navHeight?: number;
12
+ isVisible?: boolean;
13
+ renderCheckPlans?: (overrides?: CheckPlansProps) => React.ReactNode;
14
+ }
@@ -1,70 +1,70 @@
1
- import React from "react";
2
- import { AnchoredBottomBannerProps } from "./types";
3
- import Link from "next/link";
4
-
5
- import { MaterialIcon } from "@shared/components/material-icon";
6
-
7
- export const AnchoredBottomBanner: React.FC<AnchoredBottomBannerProps> = ({
8
- ctaSuffixText,
9
- backgroundColor,
10
- iconName,
11
- boxShadow,
12
- ctaButtonLabel,
13
- ctaButtonLink,
14
- ctaButtonTarget,
15
- anchorId = "anchored-banner",
16
- }) => {
17
- const backGroundColorClasses = {
18
- navy: "bg-bg-fill-inverse",
19
- green: "bg-bg-fill-success",
20
- blue: "bg-bg-fill-brand-supporting",
21
- purple: "bg-bg-fill-brand-tertiary",
22
- yellow: "bg-bg-fill-brand-accent",
23
- };
24
-
25
- const bgClass = backgroundColor
26
- ? backGroundColorClasses[backgroundColor]
27
- : "bg-bg-fill-brand-accent";
28
-
29
- const isYellow = backgroundColor === "yellow" || !backgroundColor;
30
- const textColorClass = isYellow ? "text-text-primary" : "text-white";
31
-
32
- return (
33
- <section id={anchorId}>
34
- <div
35
- className={`fixed bottom-0 left-0 right-0 z-[30] flex w-full items-center justify-center px-4 py-3 transition-all duration-300 ${bgClass} ${
36
- boxShadow ? boxShadow : "shadow-[0_-4px_10px_rgba(0,0,0,0.1)]"
37
- }`}
38
- >
39
- <Link
40
- href={ctaButtonLink || "#"}
41
- target={ctaButtonTarget || "_self"}
42
- className="max-w-screen-xl w-full transition-all"
43
- >
44
- <div className="flex flex-col items-center justify-center gap-1 text-center font-black">
45
- <div
46
- className={`${textColorClass} break-words text-body1 font-black leading-snug md:text-[18px]`}
47
- >
48
- {iconName && (
49
- <MaterialIcon
50
- name={
51
- iconName as React.ComponentProps<
52
- typeof MaterialIcon
53
- >["name"]
54
- }
55
- size={24}
56
- fill={1}
57
- className={`${textColorClass} align-text-bottom`}
58
- />
59
- )}
60
- {ctaButtonLabel && ctaButtonLabel}{" "}
61
- {ctaSuffixText && <span className="ml-0.5">{ctaSuffixText}</span>}
62
- </div>
63
- </div>
64
- </Link>
65
- </div>
66
- </section>
67
- );
68
- };
69
-
70
- export default AnchoredBottomBanner;
1
+ import React from "react";
2
+ import { AnchoredBottomBannerProps } from "./types";
3
+ import Link from "next/link";
4
+
5
+ import { MaterialIcon } from "@shared/components/material-icon";
6
+
7
+ export const AnchoredBottomBanner: React.FC<AnchoredBottomBannerProps> = ({
8
+ ctaSuffixText,
9
+ backgroundColor,
10
+ iconName,
11
+ boxShadow,
12
+ ctaButtonLabel,
13
+ ctaButtonLink,
14
+ ctaButtonTarget,
15
+ anchorId = "anchored-banner",
16
+ }) => {
17
+ const backGroundColorClasses = {
18
+ navy: "bg-bg-fill-inverse",
19
+ green: "bg-bg-fill-success",
20
+ blue: "bg-bg-fill-brand-supporting",
21
+ purple: "bg-bg-fill-brand-tertiary",
22
+ yellow: "bg-bg-fill-brand-accent",
23
+ };
24
+
25
+ const bgClass = backgroundColor
26
+ ? backGroundColorClasses[backgroundColor]
27
+ : "bg-bg-fill-brand-accent";
28
+
29
+ const isYellow = backgroundColor === "yellow" || !backgroundColor;
30
+ const textColorClass = isYellow ? "text-text-primary" : "text-white";
31
+
32
+ return (
33
+ <section id={anchorId}>
34
+ <div
35
+ className={`fixed bottom-0 left-0 right-0 z-[30] flex w-full items-center justify-center px-4 py-3 transition-all duration-300 ${bgClass} ${
36
+ boxShadow ? boxShadow : "shadow-[0_-4px_10px_rgba(0,0,0,0.1)]"
37
+ }`}
38
+ >
39
+ <Link
40
+ href={ctaButtonLink || "#"}
41
+ target={ctaButtonTarget || "_self"}
42
+ className="max-w-screen-xl w-full transition-all"
43
+ >
44
+ <div className="flex flex-col items-center justify-center gap-1 text-center font-black">
45
+ <div
46
+ className={`${textColorClass} break-words text-body1 font-black leading-snug md:text-[18px]`}
47
+ >
48
+ {iconName && (
49
+ <MaterialIcon
50
+ name={
51
+ iconName as React.ComponentProps<
52
+ typeof MaterialIcon
53
+ >["name"]
54
+ }
55
+ size={24}
56
+ fill={1}
57
+ className={`${textColorClass} align-text-bottom`}
58
+ />
59
+ )}
60
+ {ctaButtonLabel && ctaButtonLabel}{" "}
61
+ {ctaSuffixText && <span className="ml-0.5">{ctaSuffixText}</span>}
62
+ </div>
63
+ </div>
64
+ </Link>
65
+ </div>
66
+ </section>
67
+ );
68
+ };
69
+
70
+ export default AnchoredBottomBanner;