@windstream/react-shared-components 0.1.92 → 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 (293) hide show
  1. package/README.md +635 -635
  2. package/dist/contentful/index.d.ts +1 -0
  3. package/dist/contentful/index.esm.js +2 -2
  4. package/dist/contentful/index.esm.js.map +1 -1
  5. package/dist/contentful/index.js +3 -3
  6. package/dist/contentful/index.js.map +1 -1
  7. package/dist/core.d.ts +5 -5
  8. package/dist/index.d.ts +2 -2
  9. package/dist/index.esm.js +1 -1
  10. package/dist/index.esm.js.map +1 -1
  11. package/dist/index.js +1 -1
  12. package/dist/index.js.map +1 -1
  13. package/dist/styles.css +1 -1
  14. package/dist/utils/index.esm.js +1 -1
  15. package/dist/utils/index.js +1 -1
  16. package/package.json +191 -185
  17. package/src/components/accordion/Accordion.stories.tsx +230 -230
  18. package/src/components/accordion/index.test.tsx +270 -0
  19. package/src/components/accordion/index.tsx +70 -70
  20. package/src/components/accordion/types.ts +12 -12
  21. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  22. package/src/components/alert-card/index.test.tsx +152 -0
  23. package/src/components/alert-card/index.tsx +41 -41
  24. package/src/components/alert-card/types.ts +13 -13
  25. package/src/components/animation-wrapper/index.test.tsx +424 -0
  26. package/src/components/animation-wrapper/index.tsx +129 -129
  27. package/src/components/animation-wrapper/types.ts +11 -11
  28. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  29. package/src/components/brand-button/helpers.ts +35 -35
  30. package/src/components/brand-button/index.test.tsx +292 -0
  31. package/src/components/brand-button/index.tsx +120 -120
  32. package/src/components/brand-button/types.ts +38 -38
  33. package/src/components/button/Button.stories.tsx +108 -108
  34. package/src/components/button/index.test.tsx +91 -0
  35. package/src/components/button/index.tsx +27 -27
  36. package/src/components/button/types.ts +14 -14
  37. package/src/components/call-button/CallButton.stories.tsx +324 -324
  38. package/src/components/call-button/index.test.tsx +260 -0
  39. package/src/components/call-button/index.tsx +106 -106
  40. package/src/components/call-button/types.ts +16 -16
  41. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  42. package/src/components/checkbox/index.test.tsx +252 -0
  43. package/src/components/checkbox/index.tsx +197 -197
  44. package/src/components/checkbox/types.ts +27 -27
  45. package/src/components/checklist/Checklist.stories.tsx +150 -150
  46. package/src/components/checklist/index.test.tsx +231 -0
  47. package/src/components/checklist/index.tsx +96 -61
  48. package/src/components/checklist/types.ts +23 -17
  49. package/src/components/collapse/Collapse.stories.tsx +255 -255
  50. package/src/components/collapse/index.test.tsx +277 -0
  51. package/src/components/collapse/index.tsx +47 -46
  52. package/src/components/collapse/types.ts +6 -6
  53. package/src/components/divider/Divider.stories.tsx +205 -205
  54. package/src/components/divider/index.test.tsx +53 -0
  55. package/src/components/divider/index.tsx +22 -22
  56. package/src/components/divider/type.ts +3 -3
  57. package/src/components/image/Image.stories.tsx +113 -113
  58. package/src/components/image/index.test.tsx +174 -0
  59. package/src/components/image/index.tsx +25 -25
  60. package/src/components/image/types.ts +40 -40
  61. package/src/components/input/Input.stories.tsx +325 -325
  62. package/src/components/input/index.test.tsx +348 -0
  63. package/src/components/input/index.tsx +177 -177
  64. package/src/components/input/types.ts +37 -37
  65. package/src/components/link/Link.stories.tsx +163 -163
  66. package/src/components/link/index.test.tsx +199 -0
  67. package/src/components/link/index.tsx +116 -116
  68. package/src/components/link/types.ts +25 -25
  69. package/src/components/list/List.stories.tsx +272 -272
  70. package/src/components/list/index.test.tsx +166 -0
  71. package/src/components/list/index.tsx +88 -88
  72. package/src/components/list/list-item/index.tsx +38 -38
  73. package/src/components/list/list-item/types.ts +13 -13
  74. package/src/components/list/types.ts +29 -29
  75. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  76. package/src/components/material-icon/constants.ts +99 -99
  77. package/src/components/material-icon/index.test.tsx +130 -0
  78. package/src/components/material-icon/index.tsx +47 -47
  79. package/src/components/material-icon/types.ts +31 -31
  80. package/src/components/modal/Modal.stories.tsx +171 -171
  81. package/src/components/modal/index.test.tsx +310 -0
  82. package/src/components/modal/index.tsx +164 -164
  83. package/src/components/modal/types.ts +24 -24
  84. package/src/components/next-image/index.test.tsx +406 -0
  85. package/src/components/next-image/index.tsx +74 -74
  86. package/src/components/next-image/types.ts +1 -1
  87. package/src/components/pagination/index.test.tsx +521 -0
  88. package/src/components/pagination/index.tsx +91 -91
  89. package/src/components/pagination/types.ts +6 -6
  90. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  91. package/src/components/radio-button/index.test.tsx +151 -0
  92. package/src/components/radio-button/index.tsx +75 -75
  93. package/src/components/radio-button/types.ts +21 -21
  94. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  95. package/src/components/see-more/index.test.tsx +96 -0
  96. package/src/components/see-more/index.tsx +44 -44
  97. package/src/components/see-more/types.ts +4 -4
  98. package/src/components/select/Select.stories.tsx +411 -411
  99. package/src/components/select/index.test.tsx +256 -0
  100. package/src/components/select/index.tsx +155 -155
  101. package/src/components/select/types.ts +36 -36
  102. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  103. package/src/components/select-plan-button/index.test.tsx +173 -0
  104. package/src/components/select-plan-button/index.tsx +63 -63
  105. package/src/components/select-plan-button/types.ts +17 -17
  106. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  107. package/src/components/skeleton/index.test.tsx +74 -0
  108. package/src/components/skeleton/index.tsx +61 -61
  109. package/src/components/skeleton/types.ts +4 -4
  110. package/src/components/spinner/Spinner.stories.tsx +335 -335
  111. package/src/components/spinner/index.test.tsx +76 -0
  112. package/src/components/spinner/index.tsx +44 -44
  113. package/src/components/spinner/types.ts +5 -5
  114. package/src/components/text/Text.stories.tsx +321 -321
  115. package/src/components/text/index.test.tsx +65 -0
  116. package/src/components/text/index.tsx +25 -25
  117. package/src/components/text/types.ts +45 -45
  118. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  119. package/src/components/tooltip/index.test.tsx +50 -0
  120. package/src/components/tooltip/index.tsx +74 -74
  121. package/src/components/tooltip/types.ts +7 -7
  122. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  123. package/src/components/view-cart-button/index.test.tsx +57 -0
  124. package/src/components/view-cart-button/index.tsx +42 -42
  125. package/src/components/view-cart-button/types.ts +5 -5
  126. package/src/contentful/blocks/accordion/Accordion.stories.mocks.tsx +128 -128
  127. package/src/contentful/blocks/accordion/Accordion.stories.tsx +98 -98
  128. package/src/contentful/blocks/accordion/index.test.tsx +218 -0
  129. package/src/contentful/blocks/accordion/index.tsx +114 -112
  130. package/src/contentful/blocks/accordion/types.ts +34 -34
  131. package/src/contentful/blocks/address-input-banner/index.test.tsx +132 -0
  132. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  133. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  134. package/src/contentful/blocks/anchored-bottom-banner/index.test.tsx +287 -0
  135. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +181 -181
  136. package/src/contentful/blocks/anchored-bottom-banner/types.ts +13 -13
  137. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.mocks.tsx +144 -144
  138. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.tsx +157 -156
  139. package/src/contentful/blocks/blogs-grid/index.test.tsx +355 -0
  140. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  141. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  142. package/src/contentful/blocks/blogs-grid-base/index.test.tsx +274 -0
  143. package/src/contentful/blocks/blogs-grid-base/index.tsx +119 -119
  144. package/src/contentful/blocks/blogs-grid-base/types.ts +36 -36
  145. package/src/contentful/blocks/breadcrumbs/BreadcrumbNavigation.stories.tsx +147 -147
  146. package/src/contentful/blocks/breadcrumbs/index.test.tsx +281 -0
  147. package/src/contentful/blocks/breadcrumbs/index.tsx +95 -95
  148. package/src/contentful/blocks/breadcrumbs/types.ts +8 -8
  149. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  150. package/src/contentful/blocks/button/index.test.tsx +339 -0
  151. package/src/contentful/blocks/button/index.tsx +131 -131
  152. package/src/contentful/blocks/button/types.ts +39 -39
  153. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  154. package/src/contentful/blocks/callout/index.test.tsx +539 -0
  155. package/src/contentful/blocks/callout/index.tsx +277 -277
  156. package/src/contentful/blocks/callout/types.ts +78 -78
  157. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  158. package/src/contentful/blocks/cards/blog-card/index.test.tsx +218 -0
  159. package/src/contentful/blocks/cards/blog-card/index.tsx +129 -129
  160. package/src/contentful/blocks/cards/blog-card/types.ts +34 -34
  161. package/src/contentful/blocks/cards/floating-image-card/index.test.tsx +201 -0
  162. package/src/contentful/blocks/cards/floating-image-card/index.tsx +119 -119
  163. package/src/contentful/blocks/cards/floating-image-card/types.ts +30 -30
  164. package/src/contentful/blocks/cards/full-image-card/index.test.tsx +216 -0
  165. package/src/contentful/blocks/cards/full-image-card/index.tsx +130 -130
  166. package/src/contentful/blocks/cards/full-image-card/types.ts +29 -29
  167. package/src/contentful/blocks/cards/index.test.tsx +39 -0
  168. package/src/contentful/blocks/cards/index.tsx +13 -13
  169. package/src/contentful/blocks/cards/product-card/index.test.tsx +263 -0
  170. package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
  171. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  172. package/src/contentful/blocks/cards/simple-card/index.test.tsx +364 -0
  173. package/src/contentful/blocks/cards/simple-card/index.tsx +325 -325
  174. package/src/contentful/blocks/cards/simple-card/types.ts +71 -71
  175. package/src/contentful/blocks/cards/testimonial-card/index.test.tsx +180 -0
  176. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  177. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  178. package/src/contentful/blocks/cards/types.ts +1 -1
  179. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  180. package/src/contentful/blocks/carousel/helper.test.tsx +539 -0
  181. package/src/contentful/blocks/carousel/helper.tsx +494 -494
  182. package/src/contentful/blocks/carousel/index.test.tsx +308 -0
  183. package/src/contentful/blocks/carousel/index.tsx +87 -87
  184. package/src/contentful/blocks/carousel/types.test.ts +16 -0
  185. package/src/contentful/blocks/carousel/types.ts +145 -145
  186. package/src/contentful/blocks/cart-retention-banner/index.test.tsx +409 -0
  187. package/src/contentful/blocks/cart-retention-banner/index.tsx +109 -109
  188. package/src/contentful/blocks/cart-retention-banner/types.ts +11 -11
  189. package/src/contentful/blocks/comparison-table/index.test.tsx +114 -0
  190. package/src/contentful/blocks/comparison-table/index.tsx +29 -29
  191. package/src/contentful/blocks/comparison-table/types.ts +6 -6
  192. package/src/contentful/blocks/cookiebanner/index.test.tsx +277 -0
  193. package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
  194. package/src/contentful/blocks/cookiebanner/type.ts +7 -7
  195. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  196. package/src/contentful/blocks/cta-callout/index.test.tsx +244 -0
  197. package/src/contentful/blocks/cta-callout/index.tsx +73 -73
  198. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  199. package/src/contentful/blocks/dynamic-tabs/index.test.tsx +240 -0
  200. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  201. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  202. package/src/contentful/blocks/email-input-block/index.test.tsx +213 -0
  203. package/src/contentful/blocks/email-input-block/index.tsx +121 -116
  204. package/src/contentful/blocks/email-input-block/types.ts +16 -16
  205. package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
  206. package/src/contentful/blocks/find-kinetic/index.test.tsx +269 -0
  207. package/src/contentful/blocks/find-kinetic/index.tsx +138 -130
  208. package/src/contentful/blocks/find-kinetic/types.ts +20 -19
  209. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  210. package/src/contentful/blocks/floating-banner/index.test.tsx +246 -0
  211. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  212. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  213. package/src/contentful/blocks/footer/Footer.stories.tsx +317 -317
  214. package/src/contentful/blocks/footer/index.test.tsx +302 -0
  215. package/src/contentful/blocks/footer/index.tsx +91 -91
  216. package/src/contentful/blocks/footer/types.ts +13 -13
  217. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  218. package/src/contentful/blocks/image-promo-bar/helper.test.tsx +61 -0
  219. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  220. package/src/contentful/blocks/image-promo-bar/index.test.tsx +467 -0
  221. package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
  222. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  223. package/src/contentful/blocks/image-promo-bar/vimeo-embed.test.tsx +142 -0
  224. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  225. package/src/contentful/blocks/image-promo-bar/youtube-embed.test.tsx +104 -0
  226. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  227. package/src/contentful/blocks/modal/constants.ts +53 -53
  228. package/src/contentful/blocks/modal/index.test.tsx +209 -0
  229. package/src/contentful/blocks/modal/index.tsx +108 -108
  230. package/src/contentful/blocks/modal/types.ts +12 -12
  231. package/src/contentful/blocks/navigation/Navigation.stories.mocks.tsx +78 -0
  232. package/src/contentful/blocks/navigation/Navigation.stories.tsx +138 -0
  233. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.test.tsx +208 -0
  234. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +141 -139
  235. package/src/contentful/blocks/navigation/index.test.tsx +924 -0
  236. package/src/contentful/blocks/navigation/index.tsx +569 -568
  237. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.test.tsx +131 -0
  238. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  239. package/src/contentful/blocks/navigation/types.ts +71 -71
  240. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  241. package/src/contentful/blocks/primary-hero/index.test.tsx +286 -0
  242. package/src/contentful/blocks/primary-hero/index.tsx +239 -236
  243. package/src/contentful/blocks/primary-hero/types.ts +37 -37
  244. package/src/contentful/blocks/search-block/index.test.tsx +268 -0
  245. package/src/contentful/blocks/search-block/index.tsx +90 -90
  246. package/src/contentful/blocks/search-block/types.ts +15 -15
  247. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  248. package/src/contentful/blocks/shape-background-wrapper/index.test.tsx +284 -0
  249. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  250. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  251. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  252. package/src/contentful/blocks/text/index.test.tsx +36 -0
  253. package/src/contentful/blocks/text/index.tsx +12 -12
  254. package/src/contentful/blocks/text/types.ts +1 -1
  255. package/src/contentful/index.test.ts +45 -0
  256. package/src/contentful/index.ts +105 -105
  257. package/src/global-mocks/contentful/to-document.ts +25 -0
  258. package/src/global-mocks/cookie.ts +48 -0
  259. package/src/global-mocks/cx.ts +37 -0
  260. package/src/global-mocks/index.ts +89 -0
  261. package/src/global-mocks/speed-card-bg.ts +27 -0
  262. package/src/global-mocks/utm.ts +49 -0
  263. package/src/hooks/contentful/use-contentful-rich-text.test.tsx +1758 -0
  264. package/src/hooks/contentful/use-contentful-rich-text.tsx +309 -309
  265. package/src/hooks/contentful/use-processed-check-list.test.tsx +277 -0
  266. package/src/hooks/contentful/use-processed-check-list.ts +63 -63
  267. package/src/hooks/use-body-scroll-lock.test.ts +134 -0
  268. package/src/hooks/use-body-scroll-lock.ts +34 -34
  269. package/src/hooks/use-carousel-swipe.test.ts +393 -0
  270. package/src/hooks/use-carousel-swipe.ts +264 -264
  271. package/src/hooks/use-outside-click.test.ts +142 -0
  272. package/src/hooks/use-outside-click.ts +17 -17
  273. package/src/index.ts +107 -107
  274. package/src/next/index.test.ts +7 -0
  275. package/src/next/index.ts +5 -5
  276. package/src/setupTests.ts +52 -46
  277. package/src/stories/DocsTemplate.tsx +24 -24
  278. package/src/styles/globals.css +343 -343
  279. package/src/types/global.d.ts +9 -9
  280. package/src/types/micro-components.ts +99 -99
  281. package/src/types/utm.ts +49 -49
  282. package/src/utils/contentful/to-document.test.ts +85 -0
  283. package/src/utils/contentful/to-document.ts +24 -24
  284. package/src/utils/cookie.test.ts +180 -0
  285. package/src/utils/cookie.ts +84 -84
  286. package/src/utils/cx.test.ts +90 -0
  287. package/src/utils/cx.ts +49 -49
  288. package/src/utils/index.test.ts +115 -0
  289. package/src/utils/index.ts +41 -41
  290. package/src/utils/speed-card-bg.test.ts +46 -0
  291. package/src/utils/speed-card-bg.ts +24 -24
  292. package/src/utils/utm.test.ts +359 -0
  293. package/src/utils/utm.ts +221 -221
@@ -0,0 +1,467 @@
1
+ import React from "react";
2
+ import { ImagePromoBar } from "./index";
3
+ import { ImagePromoBarProps } from "./types";
4
+
5
+ import { fireEvent, render, screen } from "@testing-library/react";
6
+
7
+ jest.mock("../button", () => ({
8
+ __esModule: true,
9
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
10
+ Button: ({ buttonLabel, ...rest }: any) => (
11
+ <button data-testid="cta-button">{buttonLabel || "CTA"}</button>
12
+ ),
13
+ default: ({ buttonLabel }: any) => (
14
+ <button data-testid="cta-button">{buttonLabel || "CTA"}</button>
15
+ ),
16
+ }));
17
+
18
+ jest.mock("./helper", () => ({
19
+ PlayButton: ({ isHovered }: any) => (
20
+ <div data-testid="play-button" data-hovered={isHovered} />
21
+ ),
22
+ }));
23
+
24
+ jest.mock("./vimeo-embed", () => ({
25
+ VimeoEmbed: ({ link, autoplay }: any) => (
26
+ <div data-testid="vimeo-embed" data-link={link} data-autoplay={autoplay} />
27
+ ),
28
+ }));
29
+
30
+ jest.mock("./youtube-embed", () => ({
31
+ YoutubeEmbed: ({ link, autoplay }: any) => (
32
+ <div
33
+ data-testid="youtube-embed"
34
+ data-link={link}
35
+ data-autoplay={autoplay}
36
+ />
37
+ ),
38
+ }));
39
+
40
+ jest.mock("@shared/components/checklist", () => ({
41
+ Checklist: ({ items }: any) => (
42
+ <ul data-testid="checklist">
43
+ {items.map((i: string, idx: number) => (
44
+ <li key={idx}>{i}</li>
45
+ ))}
46
+ </ul>
47
+ ),
48
+ }));
49
+
50
+ jest.mock("@shared/components/image", () => ({
51
+ Image: ({ src, alt }: any) => <img data-testid="image" src={src} alt={alt} />,
52
+ }));
53
+
54
+ jest.mock("@shared/components/link", () => ({
55
+ Link: ({ children, href }: any) => (
56
+ <a data-testid="image-link" href={href}>
57
+ {children}
58
+ </a>
59
+ ),
60
+ }));
61
+
62
+ jest.mock("@shared/components/next-image", () => ({
63
+ NextImage: ({ src, alt }: any) => (
64
+ <img data-testid="next-image" src={src} alt={alt} />
65
+ ),
66
+ }));
67
+
68
+ jest.mock("@shared/components/text", () => ({
69
+ Text: ({ as: Tag = "span", children, className }: any) => (
70
+ <Tag className={className}>{children}</Tag>
71
+ ),
72
+ }));
73
+
74
+ jest.mock("@shared/utils", () => ({
75
+ cx: (...args: any[]) => args.filter(Boolean).join(" "),
76
+ }));
77
+
78
+ const defaultProps: ImagePromoBarProps = {
79
+ brow: "",
80
+ enableHeading: false,
81
+ title: "",
82
+ subTitle: "",
83
+ image: "",
84
+ imageWidth: 660,
85
+ imageHeight: 660,
86
+ mediaPosition: true,
87
+ description: null,
88
+ openDescriptionLinksOnANewTab: false,
89
+ checklist: [],
90
+ disclaimer: null,
91
+ imageLinks: [],
92
+ cta: null,
93
+ secondaryCta: null,
94
+ ctaDisclaimer: null,
95
+ videoLink: { link: "", image: "", videoPopup: false },
96
+ maxWidth: true,
97
+ color: "light",
98
+ };
99
+
100
+ describe("ImagePromoBar", () => {
101
+ describe("Text content", () => {
102
+ it("renders brow text when provided", () => {
103
+ render(<ImagePromoBar {...defaultProps} brow="New" />);
104
+ expect(screen.getByText("New")).toBeInTheDocument();
105
+ });
106
+
107
+ it("renders title as h2 when enableHeading is false", () => {
108
+ render(<ImagePromoBar {...defaultProps} title="Title" />);
109
+ expect(screen.getByRole("heading", { level: 2 })).toHaveTextContent(
110
+ "Title"
111
+ );
112
+ });
113
+
114
+ it("renders title as h1 when enableHeading is true", () => {
115
+ render(
116
+ <ImagePromoBar {...defaultProps} title="Title" enableHeading={true} />
117
+ );
118
+ expect(screen.getByRole("heading", { level: 1 })).toHaveTextContent(
119
+ "Title"
120
+ );
121
+ });
122
+
123
+ it("renders subTitle as h3 when enableHeading is false", () => {
124
+ render(<ImagePromoBar {...defaultProps} subTitle="Sub" />);
125
+ expect(screen.getByRole("heading", { level: 3 })).toHaveTextContent(
126
+ "Sub"
127
+ );
128
+ });
129
+
130
+ it("renders subTitle as h2 when enableHeading is true", () => {
131
+ render(
132
+ <ImagePromoBar {...defaultProps} subTitle="Sub" enableHeading={true} />
133
+ );
134
+ expect(screen.getByText("Sub").tagName).toBe("H2");
135
+ });
136
+
137
+ it("renders description", () => {
138
+ render(<ImagePromoBar {...defaultProps} description="Desc text" />);
139
+ expect(screen.getByText("Desc text")).toBeInTheDocument();
140
+ });
141
+
142
+ it("renders ctaDisclaimer and disclaimer", () => {
143
+ render(
144
+ <ImagePromoBar
145
+ {...defaultProps}
146
+ ctaDisclaimer="Discl1"
147
+ disclaimer="Discl2"
148
+ />
149
+ );
150
+ expect(screen.getByText("Discl1")).toBeInTheDocument();
151
+ expect(screen.getByText("Discl2")).toBeInTheDocument();
152
+ });
153
+ });
154
+
155
+ describe("Color and layout", () => {
156
+ it("applies text-white for light color", () => {
157
+ const { container } = render(
158
+ <ImagePromoBar {...defaultProps} color="light" />
159
+ );
160
+ expect(container.querySelector(".text-white")).toBeInTheDocument();
161
+ });
162
+
163
+ it("applies text-text for dark color", () => {
164
+ const { container } = render(
165
+ <ImagePromoBar {...defaultProps} color="dark" />
166
+ );
167
+ expect(container.querySelector(".text-text")).toBeInTheDocument();
168
+ });
169
+
170
+ it("applies max-w-120 when maxWidth is true", () => {
171
+ const { container } = render(
172
+ <ImagePromoBar {...defaultProps} maxWidth={true} />
173
+ );
174
+ expect(container.querySelector(".max-w-120")).toBeInTheDocument();
175
+ });
176
+
177
+ it("does not apply max-w-120 when maxWidth is false", () => {
178
+ const { container } = render(
179
+ <ImagePromoBar {...defaultProps} maxWidth={false} />
180
+ );
181
+ expect(container.querySelector(".max-w-120")).not.toBeInTheDocument();
182
+ });
183
+
184
+ it("applies lg:flex-row-reverse when mediaPosition is true", () => {
185
+ const { container } = render(
186
+ <ImagePromoBar {...defaultProps} mediaPosition={true} />
187
+ );
188
+ expect(container.innerHTML).toContain("lg:flex-row-reverse");
189
+ });
190
+
191
+ it("applies lg:flex-row when mediaPosition is false", () => {
192
+ const { container } = render(
193
+ <ImagePromoBar {...defaultProps} mediaPosition={false} />
194
+ );
195
+ expect(container.innerHTML).toContain("lg:flex-row");
196
+ expect(container.innerHTML).not.toContain("lg:flex-row-reverse");
197
+ });
198
+ });
199
+
200
+ describe("Checklist", () => {
201
+ it("renders checklist when items exist", () => {
202
+ render(
203
+ <ImagePromoBar {...defaultProps} checklist={["Item 1", "Item 2"]} />
204
+ );
205
+ expect(screen.getByTestId("checklist")).toBeInTheDocument();
206
+ });
207
+
208
+ it("does not render checklist when empty", () => {
209
+ render(<ImagePromoBar {...defaultProps} checklist={[]} />);
210
+ expect(screen.queryByTestId("checklist")).not.toBeInTheDocument();
211
+ });
212
+ });
213
+
214
+ describe("Image links", () => {
215
+ it("renders image links when provided", () => {
216
+ const imageLinks = [
217
+ { url: "https://a.com", image: "https://img.test/a.png" },
218
+ { url: "https://b.com", image: "https://img.test/b.png" },
219
+ ];
220
+ render(<ImagePromoBar {...defaultProps} imageLinks={imageLinks} />);
221
+ const links = screen.getAllByTestId("image-link");
222
+ expect(links).toHaveLength(2);
223
+ expect(links[0]).toHaveAttribute("href", "https://a.com");
224
+ });
225
+ });
226
+
227
+ describe("CTAs", () => {
228
+ it("renders primary CTA", () => {
229
+ render(<ImagePromoBar {...defaultProps} cta={{ buttonLabel: "Buy" }} />);
230
+ expect(screen.getByTestId("cta-button")).toBeInTheDocument();
231
+ });
232
+
233
+ it("renders both CTAs", () => {
234
+ render(
235
+ <ImagePromoBar
236
+ {...defaultProps}
237
+ cta={{ buttonLabel: "Primary" }}
238
+ secondaryCta={{ buttonLabel: "Secondary" }}
239
+ />
240
+ );
241
+ expect(screen.getAllByTestId("cta-button")).toHaveLength(2);
242
+ });
243
+
244
+ it("does not render CTA section when neither provided", () => {
245
+ const { container } = render(<ImagePromoBar {...defaultProps} />);
246
+ expect(container.querySelector(".primary-cta")).not.toBeInTheDocument();
247
+ });
248
+ });
249
+
250
+ describe("Image", () => {
251
+ it("renders NextImage when image is provided", () => {
252
+ render(
253
+ <ImagePromoBar {...defaultProps} image="https://img.test/hero.jpg" />
254
+ );
255
+ const img = screen.getByTestId("next-image");
256
+ expect(img).toHaveAttribute("src", "https://img.test/hero.jpg");
257
+ });
258
+
259
+ it("does not render image section when image is empty", () => {
260
+ render(<ImagePromoBar {...defaultProps} image="" />);
261
+ expect(screen.queryByTestId("next-image")).not.toBeInTheDocument();
262
+ });
263
+ });
264
+
265
+ describe("Video", () => {
266
+ it("renders video section with play button when videoLink has link and image", () => {
267
+ render(
268
+ <ImagePromoBar
269
+ {...defaultProps}
270
+ videoLink={{
271
+ link: "https://youtube.com/watch?v=123",
272
+ image: "thumb.jpg",
273
+ }}
274
+ />
275
+ );
276
+ expect(screen.getByTestId("play-button")).toBeInTheDocument();
277
+ expect(screen.getByAltText("Video preview")).toBeInTheDocument();
278
+ });
279
+
280
+ it("shows youtube embed on click for youtube link", () => {
281
+ render(
282
+ <ImagePromoBar
283
+ {...defaultProps}
284
+ videoLink={{
285
+ link: "https://youtube.com/watch?v=123",
286
+ image: "thumb.jpg",
287
+ videoPopup: false,
288
+ }}
289
+ />
290
+ );
291
+ fireEvent.click(
292
+ screen.getByTestId("play-button").closest(".video-section")!
293
+ );
294
+ expect(screen.getByTestId("youtube-embed")).toBeInTheDocument();
295
+ });
296
+
297
+ it("shows vimeo embed on click for vimeo link", () => {
298
+ render(
299
+ <ImagePromoBar
300
+ {...defaultProps}
301
+ videoLink={{
302
+ link: "https://vimeo.com/123",
303
+ image: "thumb.jpg",
304
+ videoPopup: false,
305
+ }}
306
+ />
307
+ );
308
+ fireEvent.click(
309
+ screen.getByTestId("play-button").closest(".video-section")!
310
+ );
311
+ expect(screen.getByTestId("vimeo-embed")).toBeInTheDocument();
312
+ });
313
+
314
+ it("shows popup overlay when videoPopup is true and playing", () => {
315
+ const { container } = render(
316
+ <ImagePromoBar
317
+ {...defaultProps}
318
+ videoLink={{
319
+ link: "https://youtube.com/watch?v=123",
320
+ image: "thumb.jpg",
321
+ videoPopup: true,
322
+ }}
323
+ />
324
+ );
325
+ fireEvent.click(
326
+ screen.getByTestId("play-button").closest(".video-section")!
327
+ );
328
+ expect(container.querySelector(".fixed")).toBeInTheDocument();
329
+ });
330
+
331
+ it("closes popup on overlay click", () => {
332
+ const { container } = render(
333
+ <ImagePromoBar
334
+ {...defaultProps}
335
+ videoLink={{
336
+ link: "https://youtube.com/watch?v=123",
337
+ image: "thumb.jpg",
338
+ videoPopup: true,
339
+ }}
340
+ />
341
+ );
342
+ fireEvent.click(
343
+ screen.getByTestId("play-button").closest(".video-section")!
344
+ );
345
+ fireEvent.click(container.querySelector(".fixed")!);
346
+ expect(container.querySelector(".fixed")).not.toBeInTheDocument();
347
+ });
348
+
349
+ it("does not close popup when clicking inside video container", () => {
350
+ const { container } = render(
351
+ <ImagePromoBar
352
+ {...defaultProps}
353
+ videoLink={{
354
+ link: "https://youtube.com/watch?v=123",
355
+ image: "thumb.jpg",
356
+ videoPopup: true,
357
+ }}
358
+ />
359
+ );
360
+ fireEvent.click(
361
+ screen.getByTestId("play-button").closest(".video-section")!
362
+ );
363
+ const inner = container.querySelector(".max-w-6xl");
364
+ fireEvent.click(inner!);
365
+ expect(container.querySelector(".fixed")).toBeInTheDocument();
366
+ });
367
+
368
+ it("handles hover state on video section", () => {
369
+ render(
370
+ <ImagePromoBar
371
+ {...defaultProps}
372
+ videoLink={{
373
+ link: "https://youtube.com/watch?v=123",
374
+ image: "thumb.jpg",
375
+ }}
376
+ />
377
+ );
378
+ const videoSection = screen
379
+ .getByTestId("play-button")
380
+ .closest(".video-section")!;
381
+ fireEvent.mouseEnter(videoSection);
382
+ expect(screen.getByTestId("play-button")).toHaveAttribute(
383
+ "data-hovered",
384
+ "true"
385
+ );
386
+ fireEvent.mouseLeave(videoSection);
387
+ expect(screen.getByTestId("play-button")).toHaveAttribute(
388
+ "data-hovered",
389
+ "false"
390
+ );
391
+ });
392
+
393
+ it("does not render video section when videoLink.link is empty", () => {
394
+ const { container } = render(
395
+ <ImagePromoBar
396
+ {...defaultProps}
397
+ videoLink={{ link: "", image: "thumb.jpg" }}
398
+ />
399
+ );
400
+ expect(container.querySelector(".video-section")).not.toBeInTheDocument();
401
+ });
402
+
403
+ it("does not render video preview image when videoLink.image is missing", () => {
404
+ render(
405
+ <ImagePromoBar
406
+ {...defaultProps}
407
+ videoLink={{ link: "https://youtube.com/watch?v=123" }}
408
+ />
409
+ );
410
+ expect(screen.queryByAltText("Video preview")).not.toBeInTheDocument();
411
+ });
412
+
413
+ it("handles videoLink being undefined", () => {
414
+ const { container } = render(
415
+ <ImagePromoBar {...defaultProps} videoLink={undefined as any} />
416
+ );
417
+ expect(container.querySelector(".video-section")).not.toBeInTheDocument();
418
+ });
419
+
420
+ it("handles videoLink with no videoPopup property (defaults to false)", () => {
421
+ const { container } = render(
422
+ <ImagePromoBar
423
+ {...defaultProps}
424
+ videoLink={{
425
+ link: "https://youtube.com/watch?v=123",
426
+ image: "thumb.jpg",
427
+ }}
428
+ />
429
+ );
430
+ fireEvent.click(container.querySelector(".video-section")!);
431
+ // Should show inline embed, not popup
432
+ expect(container.querySelector(".fixed")).not.toBeInTheDocument();
433
+ expect(screen.getByTestId("youtube-embed")).toBeInTheDocument();
434
+ });
435
+
436
+ it("does not render inline embed when videoPopup is true", () => {
437
+ const { container } = render(
438
+ <ImagePromoBar
439
+ {...defaultProps}
440
+ videoLink={{
441
+ link: "https://youtube.com/watch?v=123",
442
+ image: "thumb.jpg",
443
+ videoPopup: true,
444
+ }}
445
+ />
446
+ );
447
+ fireEvent.click(container.querySelector(".video-section")!);
448
+ // Popup renders, inline section hidden via the !videoPopup && isPlaying condition
449
+ expect(container.querySelector(".fixed")).toBeInTheDocument();
450
+ });
451
+
452
+ it("does not render contentVideo embed when videoLink has no image", () => {
453
+ const { container } = render(
454
+ <ImagePromoBar
455
+ {...defaultProps}
456
+ videoLink={{
457
+ link: "https://youtube.com/watch?v=123",
458
+ videoPopup: false,
459
+ }}
460
+ />
461
+ );
462
+ fireEvent.click(container.querySelector(".video-section")!);
463
+ // contentVideo is null when videoImage is falsy
464
+ expect(screen.queryByTestId("youtube-embed")).not.toBeInTheDocument();
465
+ });
466
+ });
467
+ });