@windstream/react-shared-components 0.1.93 → 0.1.94

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 (298) hide show
  1. package/README.md +635 -635
  2. package/dist/contentful/index.esm.js +3 -3
  3. package/dist/contentful/index.esm.js.map +1 -1
  4. package/dist/contentful/index.js +3 -3
  5. package/dist/contentful/index.js.map +1 -1
  6. package/dist/core.d.ts +6 -6
  7. package/dist/index.d.ts +2 -2
  8. package/dist/index.esm.js +13 -5
  9. package/dist/index.esm.js.map +1 -1
  10. package/dist/index.js +13 -5
  11. package/dist/index.js.map +1 -1
  12. package/dist/next/index.esm.js +2 -2
  13. package/dist/next/index.esm.js.map +1 -1
  14. package/dist/next/index.js +2 -2
  15. package/dist/next/index.js.map +1 -1
  16. package/dist/styles.css +1 -1
  17. package/dist/utils/index.esm.js +1 -1
  18. package/dist/utils/index.esm.js.map +1 -1
  19. package/dist/utils/index.js +1 -1
  20. package/dist/utils/index.js.map +1 -1
  21. package/package.json +191 -185
  22. package/src/components/accordion/Accordion.stories.tsx +230 -230
  23. package/src/components/accordion/index.test.tsx +270 -0
  24. package/src/components/accordion/index.tsx +70 -70
  25. package/src/components/accordion/types.ts +12 -12
  26. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  27. package/src/components/alert-card/index.test.tsx +152 -0
  28. package/src/components/alert-card/index.tsx +41 -41
  29. package/src/components/alert-card/types.ts +13 -13
  30. package/src/components/animation-wrapper/index.test.tsx +424 -0
  31. package/src/components/animation-wrapper/index.tsx +129 -129
  32. package/src/components/animation-wrapper/types.ts +11 -11
  33. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  34. package/src/components/brand-button/helpers.ts +35 -35
  35. package/src/components/brand-button/index.test.tsx +292 -0
  36. package/src/components/brand-button/index.tsx +120 -120
  37. package/src/components/brand-button/types.ts +38 -38
  38. package/src/components/button/Button.stories.tsx +108 -108
  39. package/src/components/button/index.test.tsx +91 -0
  40. package/src/components/button/index.tsx +27 -27
  41. package/src/components/button/types.ts +14 -14
  42. package/src/components/call-button/CallButton.stories.tsx +324 -324
  43. package/src/components/call-button/index.test.tsx +260 -0
  44. package/src/components/call-button/index.tsx +106 -106
  45. package/src/components/call-button/types.ts +16 -16
  46. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  47. package/src/components/checkbox/index.test.tsx +252 -0
  48. package/src/components/checkbox/index.tsx +197 -197
  49. package/src/components/checkbox/types.ts +27 -27
  50. package/src/components/checklist/Checklist.stories.tsx +150 -150
  51. package/src/components/checklist/index.test.tsx +231 -0
  52. package/src/components/checklist/index.tsx +96 -61
  53. package/src/components/checklist/types.ts +23 -17
  54. package/src/components/collapse/Collapse.stories.tsx +255 -255
  55. package/src/components/collapse/index.test.tsx +277 -0
  56. package/src/components/collapse/index.tsx +47 -46
  57. package/src/components/collapse/types.ts +6 -6
  58. package/src/components/divider/Divider.stories.tsx +205 -205
  59. package/src/components/divider/index.test.tsx +53 -0
  60. package/src/components/divider/index.tsx +22 -22
  61. package/src/components/divider/type.ts +3 -3
  62. package/src/components/image/Image.stories.tsx +113 -113
  63. package/src/components/image/index.test.tsx +174 -0
  64. package/src/components/image/index.tsx +25 -25
  65. package/src/components/image/types.ts +40 -40
  66. package/src/components/input/Input.stories.tsx +325 -325
  67. package/src/components/input/index.test.tsx +348 -0
  68. package/src/components/input/index.tsx +177 -177
  69. package/src/components/input/types.ts +37 -37
  70. package/src/components/link/Link.stories.tsx +163 -163
  71. package/src/components/link/index.test.tsx +199 -0
  72. package/src/components/link/index.tsx +116 -116
  73. package/src/components/link/types.ts +25 -25
  74. package/src/components/list/List.stories.tsx +272 -272
  75. package/src/components/list/index.test.tsx +166 -0
  76. package/src/components/list/index.tsx +88 -88
  77. package/src/components/list/list-item/index.tsx +38 -38
  78. package/src/components/list/list-item/types.ts +13 -13
  79. package/src/components/list/types.ts +29 -29
  80. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  81. package/src/components/material-icon/constants.ts +99 -99
  82. package/src/components/material-icon/index.test.tsx +130 -0
  83. package/src/components/material-icon/index.tsx +47 -47
  84. package/src/components/material-icon/types.ts +31 -31
  85. package/src/components/modal/Modal.stories.tsx +171 -171
  86. package/src/components/modal/index.test.tsx +310 -0
  87. package/src/components/modal/index.tsx +164 -164
  88. package/src/components/modal/types.ts +24 -24
  89. package/src/components/next-image/index.test.tsx +406 -0
  90. package/src/components/next-image/index.tsx +74 -74
  91. package/src/components/next-image/types.ts +1 -1
  92. package/src/components/pagination/index.test.tsx +521 -0
  93. package/src/components/pagination/index.tsx +91 -91
  94. package/src/components/pagination/types.ts +6 -6
  95. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  96. package/src/components/radio-button/index.test.tsx +151 -0
  97. package/src/components/radio-button/index.tsx +75 -75
  98. package/src/components/radio-button/types.ts +21 -21
  99. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  100. package/src/components/see-more/index.test.tsx +96 -0
  101. package/src/components/see-more/index.tsx +44 -44
  102. package/src/components/see-more/types.ts +4 -4
  103. package/src/components/select/Select.stories.tsx +411 -411
  104. package/src/components/select/index.test.tsx +256 -0
  105. package/src/components/select/index.tsx +155 -155
  106. package/src/components/select/types.ts +36 -36
  107. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  108. package/src/components/select-plan-button/index.test.tsx +173 -0
  109. package/src/components/select-plan-button/index.tsx +63 -63
  110. package/src/components/select-plan-button/types.ts +17 -17
  111. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  112. package/src/components/skeleton/index.test.tsx +74 -0
  113. package/src/components/skeleton/index.tsx +61 -61
  114. package/src/components/skeleton/types.ts +4 -4
  115. package/src/components/spinner/Spinner.stories.tsx +335 -335
  116. package/src/components/spinner/index.test.tsx +76 -0
  117. package/src/components/spinner/index.tsx +44 -44
  118. package/src/components/spinner/types.ts +5 -5
  119. package/src/components/text/Text.stories.tsx +321 -321
  120. package/src/components/text/index.test.tsx +65 -0
  121. package/src/components/text/index.tsx +25 -25
  122. package/src/components/text/types.ts +45 -45
  123. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  124. package/src/components/tooltip/index.test.tsx +50 -0
  125. package/src/components/tooltip/index.tsx +74 -74
  126. package/src/components/tooltip/types.ts +7 -7
  127. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  128. package/src/components/view-cart-button/index.test.tsx +57 -0
  129. package/src/components/view-cart-button/index.tsx +42 -42
  130. package/src/components/view-cart-button/types.ts +5 -5
  131. package/src/contentful/blocks/accordion/Accordion.stories.mocks.tsx +128 -128
  132. package/src/contentful/blocks/accordion/Accordion.stories.tsx +98 -98
  133. package/src/contentful/blocks/accordion/index.test.tsx +218 -0
  134. package/src/contentful/blocks/accordion/index.tsx +114 -112
  135. package/src/contentful/blocks/accordion/types.ts +34 -34
  136. package/src/contentful/blocks/address-input-banner/index.test.tsx +132 -0
  137. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  138. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  139. package/src/contentful/blocks/anchored-bottom-banner/index.test.tsx +287 -0
  140. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +181 -181
  141. package/src/contentful/blocks/anchored-bottom-banner/types.ts +13 -13
  142. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.mocks.tsx +144 -144
  143. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.tsx +157 -156
  144. package/src/contentful/blocks/blogs-grid/index.test.tsx +355 -0
  145. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  146. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  147. package/src/contentful/blocks/blogs-grid-base/index.test.tsx +274 -0
  148. package/src/contentful/blocks/blogs-grid-base/index.tsx +119 -119
  149. package/src/contentful/blocks/blogs-grid-base/types.ts +36 -36
  150. package/src/contentful/blocks/breadcrumbs/BreadcrumbNavigation.stories.tsx +147 -147
  151. package/src/contentful/blocks/breadcrumbs/index.test.tsx +281 -0
  152. package/src/contentful/blocks/breadcrumbs/index.tsx +95 -95
  153. package/src/contentful/blocks/breadcrumbs/types.ts +8 -8
  154. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  155. package/src/contentful/blocks/button/index.test.tsx +339 -0
  156. package/src/contentful/blocks/button/index.tsx +131 -131
  157. package/src/contentful/blocks/button/types.ts +39 -39
  158. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  159. package/src/contentful/blocks/callout/index.test.tsx +539 -0
  160. package/src/contentful/blocks/callout/index.tsx +277 -277
  161. package/src/contentful/blocks/callout/types.ts +78 -78
  162. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  163. package/src/contentful/blocks/cards/blog-card/index.test.tsx +218 -0
  164. package/src/contentful/blocks/cards/blog-card/index.tsx +129 -129
  165. package/src/contentful/blocks/cards/blog-card/types.ts +34 -34
  166. package/src/contentful/blocks/cards/floating-image-card/index.test.tsx +201 -0
  167. package/src/contentful/blocks/cards/floating-image-card/index.tsx +119 -119
  168. package/src/contentful/blocks/cards/floating-image-card/types.ts +30 -30
  169. package/src/contentful/blocks/cards/full-image-card/index.test.tsx +216 -0
  170. package/src/contentful/blocks/cards/full-image-card/index.tsx +130 -130
  171. package/src/contentful/blocks/cards/full-image-card/types.ts +29 -29
  172. package/src/contentful/blocks/cards/index.test.tsx +39 -0
  173. package/src/contentful/blocks/cards/index.tsx +13 -13
  174. package/src/contentful/blocks/cards/product-card/index.test.tsx +263 -0
  175. package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
  176. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  177. package/src/contentful/blocks/cards/simple-card/index.test.tsx +364 -0
  178. package/src/contentful/blocks/cards/simple-card/index.tsx +325 -325
  179. package/src/contentful/blocks/cards/simple-card/types.ts +71 -71
  180. package/src/contentful/blocks/cards/testimonial-card/index.test.tsx +180 -0
  181. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  182. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  183. package/src/contentful/blocks/cards/types.ts +1 -1
  184. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  185. package/src/contentful/blocks/carousel/helper.test.tsx +539 -0
  186. package/src/contentful/blocks/carousel/helper.tsx +494 -494
  187. package/src/contentful/blocks/carousel/index.test.tsx +308 -0
  188. package/src/contentful/blocks/carousel/index.tsx +87 -87
  189. package/src/contentful/blocks/carousel/types.test.ts +16 -0
  190. package/src/contentful/blocks/carousel/types.ts +145 -145
  191. package/src/contentful/blocks/cart-retention-banner/index.test.tsx +409 -0
  192. package/src/contentful/blocks/cart-retention-banner/index.tsx +109 -109
  193. package/src/contentful/blocks/cart-retention-banner/types.ts +11 -11
  194. package/src/contentful/blocks/comparison-table/index.test.tsx +114 -0
  195. package/src/contentful/blocks/comparison-table/index.tsx +29 -29
  196. package/src/contentful/blocks/comparison-table/types.ts +6 -6
  197. package/src/contentful/blocks/cookiebanner/index.test.tsx +277 -0
  198. package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
  199. package/src/contentful/blocks/cookiebanner/type.ts +7 -7
  200. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  201. package/src/contentful/blocks/cta-callout/index.test.tsx +244 -0
  202. package/src/contentful/blocks/cta-callout/index.tsx +73 -73
  203. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  204. package/src/contentful/blocks/dynamic-tabs/index.test.tsx +240 -0
  205. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  206. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  207. package/src/contentful/blocks/email-input-block/index.test.tsx +213 -0
  208. package/src/contentful/blocks/email-input-block/index.tsx +121 -116
  209. package/src/contentful/blocks/email-input-block/types.ts +16 -16
  210. package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
  211. package/src/contentful/blocks/find-kinetic/index.test.tsx +269 -0
  212. package/src/contentful/blocks/find-kinetic/index.tsx +138 -138
  213. package/src/contentful/blocks/find-kinetic/types.ts +20 -20
  214. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  215. package/src/contentful/blocks/floating-banner/index.test.tsx +246 -0
  216. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  217. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  218. package/src/contentful/blocks/footer/Footer.stories.tsx +317 -317
  219. package/src/contentful/blocks/footer/index.test.tsx +302 -0
  220. package/src/contentful/blocks/footer/index.tsx +91 -91
  221. package/src/contentful/blocks/footer/types.ts +13 -13
  222. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  223. package/src/contentful/blocks/image-promo-bar/helper.test.tsx +61 -0
  224. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  225. package/src/contentful/blocks/image-promo-bar/index.test.tsx +467 -0
  226. package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
  227. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  228. package/src/contentful/blocks/image-promo-bar/vimeo-embed.test.tsx +142 -0
  229. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  230. package/src/contentful/blocks/image-promo-bar/youtube-embed.test.tsx +104 -0
  231. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  232. package/src/contentful/blocks/modal/constants.ts +53 -53
  233. package/src/contentful/blocks/modal/index.test.tsx +209 -0
  234. package/src/contentful/blocks/modal/index.tsx +108 -108
  235. package/src/contentful/blocks/modal/types.ts +12 -12
  236. package/src/contentful/blocks/navigation/Navigation.stories.mocks.tsx +78 -78
  237. package/src/contentful/blocks/navigation/Navigation.stories.tsx +138 -138
  238. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.test.tsx +208 -0
  239. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +141 -141
  240. package/src/contentful/blocks/navigation/index.test.tsx +924 -0
  241. package/src/contentful/blocks/navigation/index.tsx +569 -569
  242. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.test.tsx +131 -0
  243. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  244. package/src/contentful/blocks/navigation/types.ts +71 -71
  245. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  246. package/src/contentful/blocks/primary-hero/index.test.tsx +286 -0
  247. package/src/contentful/blocks/primary-hero/index.tsx +239 -236
  248. package/src/contentful/blocks/primary-hero/types.ts +37 -37
  249. package/src/contentful/blocks/search-block/index.test.tsx +268 -0
  250. package/src/contentful/blocks/search-block/index.tsx +90 -90
  251. package/src/contentful/blocks/search-block/types.ts +15 -15
  252. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  253. package/src/contentful/blocks/shape-background-wrapper/index.test.tsx +284 -0
  254. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  255. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  256. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  257. package/src/contentful/blocks/text/index.test.tsx +36 -0
  258. package/src/contentful/blocks/text/index.tsx +12 -12
  259. package/src/contentful/blocks/text/types.ts +1 -1
  260. package/src/contentful/index.test.ts +45 -0
  261. package/src/contentful/index.ts +105 -105
  262. package/src/global-mocks/contentful/to-document.ts +25 -0
  263. package/src/global-mocks/cookie.ts +48 -0
  264. package/src/global-mocks/cx.ts +37 -0
  265. package/src/global-mocks/index.ts +89 -0
  266. package/src/global-mocks/speed-card-bg.ts +27 -0
  267. package/src/global-mocks/utm.ts +49 -0
  268. package/src/hooks/contentful/use-contentful-rich-text.test.tsx +1758 -0
  269. package/src/hooks/contentful/use-contentful-rich-text.tsx +309 -309
  270. package/src/hooks/contentful/use-processed-check-list.test.tsx +277 -0
  271. package/src/hooks/contentful/use-processed-check-list.ts +63 -63
  272. package/src/hooks/use-body-scroll-lock.test.ts +134 -0
  273. package/src/hooks/use-body-scroll-lock.ts +34 -34
  274. package/src/hooks/use-carousel-swipe.test.ts +393 -0
  275. package/src/hooks/use-carousel-swipe.ts +264 -264
  276. package/src/hooks/use-outside-click.test.ts +142 -0
  277. package/src/hooks/use-outside-click.ts +17 -17
  278. package/src/index.ts +107 -107
  279. package/src/next/index.test.ts +7 -0
  280. package/src/next/index.ts +5 -5
  281. package/src/setupTests.ts +52 -46
  282. package/src/stories/DocsTemplate.tsx +24 -24
  283. package/src/styles/globals.css +343 -343
  284. package/src/types/global.d.ts +9 -9
  285. package/src/types/micro-components.ts +99 -99
  286. package/src/types/utm.ts +49 -49
  287. package/src/utils/contentful/to-document.test.ts +85 -0
  288. package/src/utils/contentful/to-document.ts +24 -24
  289. package/src/utils/cookie.test.ts +180 -0
  290. package/src/utils/cookie.ts +84 -84
  291. package/src/utils/cx.test.ts +90 -0
  292. package/src/utils/cx.ts +49 -49
  293. package/src/utils/index.test.ts +115 -0
  294. package/src/utils/index.ts +41 -41
  295. package/src/utils/speed-card-bg.test.ts +46 -0
  296. package/src/utils/speed-card-bg.ts +24 -24
  297. package/src/utils/utm.test.ts +359 -0
  298. package/src/utils/utm.ts +221 -221
@@ -0,0 +1,287 @@
1
+ import { AnchoredBottomBanner } from "./index";
2
+
3
+ import { act, render, screen } from "@testing-library/react";
4
+
5
+ // Mock next/link
6
+ jest.mock("next/link", () => {
7
+ // eslint-disable-next-line react/display-name
8
+ return ({ children, href, target, ...rest }: any) => (
9
+ <a href={href} target={target} {...rest}>
10
+ {children}
11
+ </a>
12
+ );
13
+ });
14
+
15
+ describe("AnchoredBottomBanner", () => {
16
+ beforeEach(() => {
17
+ jest.useFakeTimers();
18
+ jest.setSystemTime(new Date("2026-05-19T12:00:00Z"));
19
+ });
20
+
21
+ afterEach(() => {
22
+ jest.useRealTimers();
23
+ });
24
+
25
+ it("renders with default anchorId", () => {
26
+ const { container } = render(<AnchoredBottomBanner />);
27
+ expect(container.querySelector("#anchored-banner")).toBeInTheDocument();
28
+ });
29
+
30
+ it("renders with custom anchorId", () => {
31
+ const { container } = render(
32
+ <AnchoredBottomBanner anchorId="custom-banner" />
33
+ );
34
+ expect(container.querySelector("#custom-banner")).toBeInTheDocument();
35
+ });
36
+
37
+ it("renders ctaButtonLabel", () => {
38
+ render(<AnchoredBottomBanner ctaButtonLabel="Shop now" />);
39
+ expect(screen.getByText("Shop now")).toBeInTheDocument();
40
+ });
41
+
42
+ it("renders ctaSuffixText", () => {
43
+ render(<AnchoredBottomBanner ctaSuffixText="Limited time" />);
44
+ expect(screen.getByText("Limited time")).toBeInTheDocument();
45
+ });
46
+
47
+ it("renders iconName", () => {
48
+ render(<AnchoredBottomBanner iconName="local_offer" />);
49
+ expect(screen.getByText("local_offer")).toBeInTheDocument();
50
+ });
51
+
52
+ it("does not render icon when iconName is not provided", () => {
53
+ const { container } = render(<AnchoredBottomBanner />);
54
+ expect(
55
+ container.querySelector(".material-symbols-rounded")
56
+ ).not.toBeInTheDocument();
57
+ });
58
+
59
+ it("uses ctaButtonLink in the link href", () => {
60
+ render(<AnchoredBottomBanner ctaButtonLink="/offers" />);
61
+ const link = screen.getByRole("link");
62
+ expect(link).toHaveAttribute("href", "/offers");
63
+ });
64
+
65
+ it("defaults link href to # when ctaButtonLink is not provided", () => {
66
+ render(<AnchoredBottomBanner />);
67
+ const link = screen.getByRole("link");
68
+ expect(link).toHaveAttribute("href", "#");
69
+ });
70
+
71
+ it("uses ctaButtonTarget for link target", () => {
72
+ render(<AnchoredBottomBanner ctaButtonTarget="_blank" />);
73
+ const link = screen.getByRole("link");
74
+ expect(link).toHaveAttribute("target", "_blank");
75
+ });
76
+
77
+ it("defaults link target to _self", () => {
78
+ render(<AnchoredBottomBanner />);
79
+ const link = screen.getByRole("link");
80
+ expect(link).toHaveAttribute("target", "_self");
81
+ });
82
+
83
+ // Background colors
84
+ it("applies yellow background by default (no backgroundColor)", () => {
85
+ const { container } = render(<AnchoredBottomBanner />);
86
+ const div = container.querySelector(".bg-bg-fill-brand-accent");
87
+ expect(div).toBeInTheDocument();
88
+ });
89
+
90
+ it("applies navy background", () => {
91
+ const { container } = render(
92
+ <AnchoredBottomBanner backgroundColor="navy" />
93
+ );
94
+ expect(container.querySelector(".bg-bg-fill-inverse")).toBeInTheDocument();
95
+ });
96
+
97
+ it("applies green background", () => {
98
+ const { container } = render(
99
+ <AnchoredBottomBanner backgroundColor="green" />
100
+ );
101
+ expect(container.querySelector(".bg-bg-fill-success")).toBeInTheDocument();
102
+ });
103
+
104
+ it("applies blue background", () => {
105
+ const { container } = render(
106
+ <AnchoredBottomBanner backgroundColor="blue" />
107
+ );
108
+ expect(
109
+ container.querySelector(".bg-bg-fill-brand-supporting")
110
+ ).toBeInTheDocument();
111
+ });
112
+
113
+ it("applies purple background", () => {
114
+ const { container } = render(
115
+ <AnchoredBottomBanner backgroundColor="purple" />
116
+ );
117
+ expect(
118
+ container.querySelector(".bg-bg-fill-brand-tertiary")
119
+ ).toBeInTheDocument();
120
+ });
121
+
122
+ it("applies white background", () => {
123
+ const { container } = render(
124
+ <AnchoredBottomBanner backgroundColor="white" />
125
+ );
126
+ expect(container.querySelector(".bg-white")).toBeInTheDocument();
127
+ });
128
+
129
+ // Text colors
130
+ it("uses dark text for yellow/white/no background", () => {
131
+ const { container } = render(
132
+ <AnchoredBottomBanner ctaButtonLabel="Test" />
133
+ );
134
+ expect(container.querySelector(".text-text-primary")).toBeInTheDocument();
135
+ });
136
+
137
+ it("uses white text for dark backgrounds", () => {
138
+ const { container } = render(
139
+ <AnchoredBottomBanner backgroundColor="navy" ctaButtonLabel="Test" />
140
+ );
141
+ expect(container.querySelector(".text-white")).toBeInTheDocument();
142
+ });
143
+
144
+ // Box shadow
145
+ it("uses default shadow when boxShadow is not provided", () => {
146
+ const { container } = render(<AnchoredBottomBanner />);
147
+ const div = container.querySelector(
148
+ '[class*="shadow-[0_-4px_10px_rgba(0,0,0,0.1)]"]'
149
+ );
150
+ expect(div).toBeInTheDocument();
151
+ });
152
+
153
+ it("uses boxShadow value when provided", () => {
154
+ const { container } = render(
155
+ <AnchoredBottomBanner boxShadow={true as any} />
156
+ );
157
+ const div = container.querySelector(
158
+ '[class*="shadow-[0_-4px_10px_rgba(0,0,0,0.1)]"]'
159
+ );
160
+ expect(div).not.toBeInTheDocument();
161
+ });
162
+
163
+ // Countdown timer
164
+ it("shows countdown when timer is enabled and within range", () => {
165
+ const end = new Date("2026-05-19T13:00:00Z").toISOString();
166
+ render(
167
+ <AnchoredBottomBanner
168
+ enableCountdownTimer={true}
169
+ countdownEndDateTime={end}
170
+ />
171
+ );
172
+ // 1 hour remaining: 01H : 00M : 00
173
+ expect(screen.getByText(/01H : 00M : 00/)).toBeInTheDocument();
174
+ });
175
+
176
+ it("does not show countdown when timer is disabled", () => {
177
+ const end = new Date("2026-05-19T13:00:00Z").toISOString();
178
+ render(<AnchoredBottomBanner countdownEndDateTime={end} />);
179
+ expect(screen.queryByText(/H :/)).not.toBeInTheDocument();
180
+ });
181
+
182
+ it("does not show countdown before start time", () => {
183
+ const start = new Date("2026-05-19T14:00:00Z").toISOString();
184
+ const end = new Date("2026-05-19T15:00:00Z").toISOString();
185
+ render(
186
+ <AnchoredBottomBanner
187
+ enableCountdownTimer={true}
188
+ countdownStartDateTime={start}
189
+ countdownEndDateTime={end}
190
+ />
191
+ );
192
+ expect(screen.queryByText(/H :/)).not.toBeInTheDocument();
193
+ });
194
+
195
+ it("does not show countdown after end time", () => {
196
+ const end = new Date("2026-05-19T11:00:00Z").toISOString();
197
+ render(
198
+ <AnchoredBottomBanner
199
+ enableCountdownTimer={true}
200
+ countdownEndDateTime={end}
201
+ />
202
+ );
203
+ expect(screen.queryByText(/H :/)).not.toBeInTheDocument();
204
+ });
205
+
206
+ it("updates countdown every second", () => {
207
+ const end = new Date("2026-05-19T12:00:10Z").toISOString();
208
+ render(
209
+ <AnchoredBottomBanner
210
+ enableCountdownTimer={true}
211
+ countdownEndDateTime={end}
212
+ />
213
+ );
214
+ expect(screen.getByText(/00H : 00M : 10/)).toBeInTheDocument();
215
+
216
+ act(() => {
217
+ jest.advanceTimersByTime(1000);
218
+ });
219
+ expect(screen.getByText(/00H : 00M : 09/)).toBeInTheDocument();
220
+ });
221
+
222
+ it("clears interval when countdown reaches zero", () => {
223
+ const end = new Date("2026-05-19T12:00:01Z").toISOString();
224
+ render(
225
+ <AnchoredBottomBanner
226
+ enableCountdownTimer={true}
227
+ countdownEndDateTime={end}
228
+ />
229
+ );
230
+
231
+ act(() => {
232
+ jest.advanceTimersByTime(2000);
233
+ });
234
+ // After expiry, countdown should not show
235
+ expect(screen.queryByText(/H :/)).not.toBeInTheDocument();
236
+ });
237
+
238
+ it("handles invalid countdownEndDateTime gracefully", () => {
239
+ const consoleSpy = jest.spyOn(console, "error").mockImplementation(() => {
240
+ /* noop */
241
+ });
242
+ render(
243
+ <AnchoredBottomBanner
244
+ enableCountdownTimer={true}
245
+ countdownEndDateTime="not-a-date"
246
+ />
247
+ );
248
+ expect(screen.queryByText(/H :/)).not.toBeInTheDocument();
249
+ consoleSpy.mockRestore();
250
+ });
251
+
252
+ it("handles invalid countdownStartDateTime gracefully", () => {
253
+ const consoleSpy = jest.spyOn(console, "error").mockImplementation(() => {
254
+ /* noop */
255
+ });
256
+ const end = new Date("2026-05-19T13:00:00Z").toISOString();
257
+ render(
258
+ <AnchoredBottomBanner
259
+ enableCountdownTimer={true}
260
+ countdownStartDateTime="not-a-date"
261
+ countdownEndDateTime={end}
262
+ />
263
+ );
264
+ expect(screen.queryByText(/H :/)).not.toBeInTheDocument();
265
+ consoleSpy.mockRestore();
266
+ });
267
+
268
+ it("logs error when start is after end", () => {
269
+ const consoleSpy = jest.spyOn(console, "error").mockImplementation(() => {
270
+ /* noop */
271
+ });
272
+ const start = new Date("2026-05-19T15:00:00Z").toISOString();
273
+ const end = new Date("2026-05-19T13:00:00Z").toISOString();
274
+ render(
275
+ <AnchoredBottomBanner
276
+ enableCountdownTimer={true}
277
+ countdownStartDateTime={start}
278
+ countdownEndDateTime={end}
279
+ />
280
+ );
281
+ expect(consoleSpy).toHaveBeenCalledWith(
282
+ "Invalid countdown range: start must be before end",
283
+ expect.any(Object)
284
+ );
285
+ consoleSpy.mockRestore();
286
+ });
287
+ });
@@ -1,181 +1,181 @@
1
- import React, {
2
- useCallback,
3
- useEffect,
4
- useMemo,
5
- useRef,
6
- useState,
7
- } from "react";
8
- import { AnchoredBottomBannerProps } from "./types";
9
- import Link from "next/link";
10
-
11
- import { MaterialIcon } from "@shared/components/material-icon";
12
-
13
- function parseCountdownDateTime(value?: string): number | undefined {
14
- if (!value) return undefined;
15
- const parsed = Date.parse(value);
16
- if (!Number.isFinite(parsed)) {
17
- console.error("Invalid countdown datetime", { value });
18
- return undefined;
19
- }
20
- return parsed;
21
- }
22
-
23
- function formatCountdown(totalSeconds: number) {
24
- const safeSeconds = Math.max(0, Math.floor(totalSeconds));
25
- const hours = Math.floor(safeSeconds / 3600);
26
- const minutes = Math.floor((safeSeconds % 3600) / 60);
27
- const seconds = safeSeconds % 60;
28
-
29
- return `${String(hours).padStart(2, "0")}H : ${String(minutes).padStart(2, "0")}M : ${String(seconds).padStart(2, "0")}`;
30
- }
31
-
32
- export const AnchoredBottomBanner: React.FC<AnchoredBottomBannerProps> = ({
33
- ctaSuffixText,
34
- backgroundColor,
35
- iconName,
36
- boxShadow,
37
- ctaButtonLabel,
38
- ctaButtonLink,
39
- ctaButtonTarget,
40
- anchorId = "anchored-banner",
41
- enableCountdownTimer,
42
- countdownStartDateTime,
43
- countdownEndDateTime,
44
- }) => {
45
- const backGroundColorClasses = {
46
- navy: "bg-bg-fill-inverse",
47
- green: "bg-bg-fill-success",
48
- blue: "bg-bg-fill-brand-supporting",
49
- purple: "bg-bg-fill-brand-tertiary",
50
- yellow: "bg-bg-fill-brand-accent",
51
- white: "bg-white",
52
- };
53
-
54
- const bgClass = backgroundColor
55
- ? backGroundColorClasses[backgroundColor]
56
- : "bg-bg-fill-brand-accent";
57
-
58
- const isLightBackground =
59
- backgroundColor === "yellow" ||
60
- backgroundColor === "white" ||
61
- !backgroundColor;
62
- const textColorClass = isLightBackground ? "text-text-primary" : "text-white";
63
-
64
- // Memoize parsed timestamps so they aren't re-parsed every second
65
- const endMs = useMemo(
66
- () => parseCountdownDateTime(countdownEndDateTime),
67
- [countdownEndDateTime]
68
- );
69
- const startMs = useMemo(
70
- () => parseCountdownDateTime(countdownStartDateTime),
71
- [countdownStartDateTime]
72
- );
73
-
74
- const isTimerValid = useMemo(() => {
75
- if (!enableCountdownTimer || endMs === undefined) return false;
76
- if (countdownStartDateTime && startMs === undefined) return false;
77
- if (startMs !== undefined && startMs >= endMs) {
78
- console.error("Invalid countdown range: start must be before end", {
79
- countdownStartDateTime,
80
- countdownEndDateTime,
81
- });
82
- return false;
83
- }
84
- return true;
85
- }, [
86
- enableCountdownTimer,
87
- endMs,
88
- startMs,
89
- countdownStartDateTime,
90
- countdownEndDateTime,
91
- ]);
92
-
93
- const [nowMs, setNowMs] = useState(() => Date.now());
94
- const intervalRef = useRef<number | null>(null);
95
-
96
- const clearTimer = useCallback(() => {
97
- if (intervalRef.current !== null) {
98
- window.clearInterval(intervalRef.current);
99
- intervalRef.current = null;
100
- }
101
- }, []);
102
-
103
- useEffect(() => {
104
- if (!isTimerValid) {
105
- clearTimer();
106
- return;
107
- }
108
-
109
- intervalRef.current = window.setInterval(() => {
110
- const now = Date.now();
111
- // Auto-clear interval once countdown expires
112
- if (endMs !== undefined && now >= endMs) {
113
- clearTimer();
114
- }
115
- setNowMs(now);
116
- }, 1000);
117
-
118
- return clearTimer;
119
- }, [isTimerValid, endMs, clearTimer]);
120
-
121
- const countdown = useMemo(() => {
122
- if (!isTimerValid || endMs === undefined) {
123
- return { shouldShow: false, text: "" };
124
- }
125
-
126
- const isBeforeStart = startMs !== undefined && nowMs < startMs;
127
- const isAfterEnd = nowMs >= endMs;
128
- if (isBeforeStart || isAfterEnd) return { shouldShow: false, text: "" };
129
-
130
- const remainingSeconds = (endMs - nowMs) / 1000;
131
- return {
132
- shouldShow: remainingSeconds > 0,
133
- text: formatCountdown(remainingSeconds),
134
- };
135
- }, [isTimerValid, endMs, startMs, nowMs]);
136
-
137
- return (
138
- <section id={anchorId}>
139
- <div
140
- 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} ${
141
- boxShadow ? boxShadow : "shadow-[0_-4px_10px_rgba(0,0,0,0.1)]"
142
- }`}
143
- >
144
- <Link
145
- href={ctaButtonLink || "#"}
146
- target={ctaButtonTarget || "_self"}
147
- className="max-w-screen-xl w-full transition-all"
148
- >
149
- <div className="flex flex-col items-center justify-center gap-1 text-center font-black">
150
- <div
151
- className={`${textColorClass} break-words text-body1 font-black leading-snug md:text-[18px]`}
152
- >
153
- {iconName && (
154
- <MaterialIcon
155
- name={
156
- iconName as React.ComponentProps<
157
- typeof MaterialIcon
158
- >["name"]
159
- }
160
- size={24}
161
- fill={1}
162
- className={`${textColorClass} align-text-bottom`}
163
- />
164
- )}
165
- {countdown.shouldShow && (
166
- <span className="inline-block whitespace-nowrap rounded-lg bg-white px-1 tabular-nums text-text">
167
- {countdown.text}
168
- </span>
169
- )}
170
- {countdown.shouldShow ? " " : null}
171
- {ctaButtonLabel && ctaButtonLabel}{" "}
172
- {ctaSuffixText && <span className="ml-0.5">{ctaSuffixText}</span>}
173
- </div>
174
- </div>
175
- </Link>
176
- </div>
177
- </section>
178
- );
179
- };
180
-
181
- export default AnchoredBottomBanner;
1
+ import React, {
2
+ useCallback,
3
+ useEffect,
4
+ useMemo,
5
+ useRef,
6
+ useState,
7
+ } from "react";
8
+ import { AnchoredBottomBannerProps } from "./types";
9
+ import Link from "next/link";
10
+
11
+ import { MaterialIcon } from "@shared/components/material-icon";
12
+
13
+ function parseCountdownDateTime(value?: string): number | undefined {
14
+ if (!value) return undefined;
15
+ const parsed = Date.parse(value);
16
+ if (!Number.isFinite(parsed)) {
17
+ console.error("Invalid countdown datetime", { value });
18
+ return undefined;
19
+ }
20
+ return parsed;
21
+ }
22
+
23
+ function formatCountdown(totalSeconds: number) {
24
+ const safeSeconds = Math.max(0, Math.floor(totalSeconds));
25
+ const hours = Math.floor(safeSeconds / 3600);
26
+ const minutes = Math.floor((safeSeconds % 3600) / 60);
27
+ const seconds = safeSeconds % 60;
28
+
29
+ return `${String(hours).padStart(2, "0")}H : ${String(minutes).padStart(2, "0")}M : ${String(seconds).padStart(2, "0")}`;
30
+ }
31
+
32
+ export const AnchoredBottomBanner: React.FC<AnchoredBottomBannerProps> = ({
33
+ ctaSuffixText,
34
+ backgroundColor,
35
+ iconName,
36
+ boxShadow,
37
+ ctaButtonLabel,
38
+ ctaButtonLink,
39
+ ctaButtonTarget,
40
+ anchorId = "anchored-banner",
41
+ enableCountdownTimer,
42
+ countdownStartDateTime,
43
+ countdownEndDateTime,
44
+ }) => {
45
+ const backGroundColorClasses = {
46
+ navy: "bg-bg-fill-inverse",
47
+ green: "bg-bg-fill-success",
48
+ blue: "bg-bg-fill-brand-supporting",
49
+ purple: "bg-bg-fill-brand-tertiary",
50
+ yellow: "bg-bg-fill-brand-accent",
51
+ white: "bg-white",
52
+ };
53
+
54
+ const bgClass = backgroundColor
55
+ ? backGroundColorClasses[backgroundColor]
56
+ : "bg-bg-fill-brand-accent";
57
+
58
+ const isLightBackground =
59
+ backgroundColor === "yellow" ||
60
+ backgroundColor === "white" ||
61
+ !backgroundColor;
62
+ const textColorClass = isLightBackground ? "text-text-primary" : "text-white";
63
+
64
+ // Memoize parsed timestamps so they aren't re-parsed every second
65
+ const endMs = useMemo(
66
+ () => parseCountdownDateTime(countdownEndDateTime),
67
+ [countdownEndDateTime]
68
+ );
69
+ const startMs = useMemo(
70
+ () => parseCountdownDateTime(countdownStartDateTime),
71
+ [countdownStartDateTime]
72
+ );
73
+
74
+ const isTimerValid = useMemo(() => {
75
+ if (!enableCountdownTimer || endMs === undefined) return false;
76
+ if (countdownStartDateTime && startMs === undefined) return false;
77
+ if (startMs !== undefined && startMs >= endMs) {
78
+ console.error("Invalid countdown range: start must be before end", {
79
+ countdownStartDateTime,
80
+ countdownEndDateTime,
81
+ });
82
+ return false;
83
+ }
84
+ return true;
85
+ }, [
86
+ enableCountdownTimer,
87
+ endMs,
88
+ startMs,
89
+ countdownStartDateTime,
90
+ countdownEndDateTime,
91
+ ]);
92
+
93
+ const [nowMs, setNowMs] = useState(() => Date.now());
94
+ const intervalRef = useRef<number | null>(null);
95
+
96
+ const clearTimer = useCallback(() => {
97
+ if (intervalRef.current !== null) {
98
+ window.clearInterval(intervalRef.current);
99
+ intervalRef.current = null;
100
+ }
101
+ }, []);
102
+
103
+ useEffect(() => {
104
+ if (!isTimerValid) {
105
+ clearTimer();
106
+ return;
107
+ }
108
+
109
+ intervalRef.current = window.setInterval(() => {
110
+ const now = Date.now();
111
+ // Auto-clear interval once countdown expires
112
+ if (endMs !== undefined && now >= endMs) {
113
+ clearTimer();
114
+ }
115
+ setNowMs(now);
116
+ }, 1000);
117
+
118
+ return clearTimer;
119
+ }, [isTimerValid, endMs, clearTimer]);
120
+
121
+ const countdown = useMemo(() => {
122
+ if (!isTimerValid || endMs === undefined) {
123
+ return { shouldShow: false, text: "" };
124
+ }
125
+
126
+ const isBeforeStart = startMs !== undefined && nowMs < startMs;
127
+ const isAfterEnd = nowMs >= endMs;
128
+ if (isBeforeStart || isAfterEnd) return { shouldShow: false, text: "" };
129
+
130
+ const remainingSeconds = (endMs - nowMs) / 1000;
131
+ return {
132
+ shouldShow: remainingSeconds > 0,
133
+ text: formatCountdown(remainingSeconds),
134
+ };
135
+ }, [isTimerValid, endMs, startMs, nowMs]);
136
+
137
+ return (
138
+ <section id={anchorId}>
139
+ <div
140
+ 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} ${
141
+ boxShadow ? boxShadow : "shadow-[0_-4px_10px_rgba(0,0,0,0.1)]"
142
+ }`}
143
+ >
144
+ <Link
145
+ href={ctaButtonLink || "#"}
146
+ target={ctaButtonTarget || "_self"}
147
+ className="max-w-screen-xl w-full transition-all"
148
+ >
149
+ <div className="flex flex-col items-center justify-center gap-1 text-center font-black">
150
+ <div
151
+ className={`${textColorClass} break-words text-body1 font-black leading-snug md:text-[18px]`}
152
+ >
153
+ {iconName && (
154
+ <MaterialIcon
155
+ name={
156
+ iconName as React.ComponentProps<
157
+ typeof MaterialIcon
158
+ >["name"]
159
+ }
160
+ size={24}
161
+ fill={1}
162
+ className={`${textColorClass} align-text-bottom`}
163
+ />
164
+ )}
165
+ {countdown.shouldShow && (
166
+ <span className="inline-block whitespace-nowrap rounded-lg bg-white px-1 tabular-nums text-text">
167
+ {countdown.text}
168
+ </span>
169
+ )}
170
+ {countdown.shouldShow ? " " : null}
171
+ {ctaButtonLabel && ctaButtonLabel}{" "}
172
+ {ctaSuffixText && <span className="ml-0.5">{ctaSuffixText}</span>}
173
+ </div>
174
+ </div>
175
+ </Link>
176
+ </div>
177
+ </section>
178
+ );
179
+ };
180
+
181
+ export default AnchoredBottomBanner;
@@ -1,13 +1,13 @@
1
- export interface AnchoredBottomBannerProps {
2
- ctaSuffixText?: string;
3
- backgroundColor?: "navy" | "yellow" | "green" | "purple" | "blue" | "white";
4
- iconName?: string;
5
- boxShadow?: boolean;
6
- ctaButtonLabel?: string;
7
- ctaButtonLink?: string;
8
- ctaButtonTarget?: string;
9
- anchorId?: string;
10
- enableCountdownTimer?: boolean;
11
- countdownStartDateTime?: string;
12
- countdownEndDateTime?: string;
13
- }
1
+ export interface AnchoredBottomBannerProps {
2
+ ctaSuffixText?: string;
3
+ backgroundColor?: "navy" | "yellow" | "green" | "purple" | "blue" | "white";
4
+ iconName?: string;
5
+ boxShadow?: boolean;
6
+ ctaButtonLabel?: string;
7
+ ctaButtonLink?: string;
8
+ ctaButtonTarget?: string;
9
+ anchorId?: string;
10
+ enableCountdownTimer?: boolean;
11
+ countdownStartDateTime?: string;
12
+ countdownEndDateTime?: string;
13
+ }