@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,277 @@
1
+ import {
2
+ label1BoldOptions,
3
+ useProcessedChecklist,
4
+ } from "./use-processed-check-list";
5
+
6
+ import { MARKS } from "@contentful/rich-text-types";
7
+ import { renderHook } from "@testing-library/react";
8
+
9
+ // Mock renderContentfulRichText to return predictable output
10
+ jest.mock("./use-contentful-rich-text", () => ({
11
+ renderContentfulRichText: jest.fn(
12
+ (doc: any, _target?: boolean, className?: string) => {
13
+ // Return a simple string representation based on the doc
14
+ if (!doc) return null;
15
+ // Extract text from the doc content if available
16
+ const text = doc?.content?.[0]?.content?.[0]?.value;
17
+ return text ? `rendered:${text}:${className ?? ""}` : null;
18
+ }
19
+ ),
20
+ }));
21
+
22
+ jest.mock("@shared/utils/contentful/to-document", () => ({
23
+ toDocument: jest.fn((v: any) => {
24
+ if (!v) return null;
25
+ if (typeof v === "string") {
26
+ return {
27
+ nodeType: "document",
28
+ data: {},
29
+ content: [
30
+ {
31
+ nodeType: "paragraph",
32
+ data: {},
33
+ content: [{ nodeType: "text", value: v, marks: [], data: {} }],
34
+ },
35
+ ],
36
+ };
37
+ }
38
+ return v;
39
+ }),
40
+ }));
41
+
42
+ const makeChecklistData = (items: any[]) => ({
43
+ list: {
44
+ items,
45
+ },
46
+ });
47
+
48
+ describe("useProcessedChecklist", () => {
49
+ describe("Full items mode (titlesOnly=false)", () => {
50
+ it("returns empty array when checklistData is undefined", () => {
51
+ const { result } = renderHook(() => useProcessedChecklist(undefined));
52
+ expect(result.current).toEqual([]);
53
+ });
54
+
55
+ it("returns empty array when checklistData has no list", () => {
56
+ const { result } = renderHook(() => useProcessedChecklist({}));
57
+ expect(result.current).toEqual([]);
58
+ });
59
+
60
+ it("returns empty array when list.items is empty", () => {
61
+ const data = makeChecklistData([]);
62
+ const { result } = renderHook(() => useProcessedChecklist(data));
63
+ expect(result.current).toEqual([]);
64
+ });
65
+
66
+ it("returns ProcessedChecklistItem array with title, iconUrl, anchorId", () => {
67
+ const data = makeChecklistData([
68
+ {
69
+ checkListTitle: "Item 1",
70
+ icon: { url: "https://cdn.com/icon1.png" },
71
+ anchorId: "anchor-1",
72
+ },
73
+ {
74
+ checkListTitle: "Item 2",
75
+ icon: { url: "https://cdn.com/icon2.png" },
76
+ anchorId: "anchor-2",
77
+ },
78
+ ]);
79
+ const { result } = renderHook(() => useProcessedChecklist(data));
80
+ expect(result.current).toHaveLength(2);
81
+ expect(result.current[0]).toEqual({
82
+ title: "rendered:Item 1:",
83
+ iconUrl: "https://cdn.com/icon1.png",
84
+ anchorId: "anchor-1",
85
+ });
86
+ expect(result.current[1]).toEqual({
87
+ title: "rendered:Item 2:",
88
+ iconUrl: "https://cdn.com/icon2.png",
89
+ anchorId: "anchor-2",
90
+ });
91
+ });
92
+
93
+ it("sets iconUrl to undefined when icon is missing", () => {
94
+ const data = makeChecklistData([
95
+ { checkListTitle: "No icon", icon: undefined },
96
+ ]);
97
+ const { result } = renderHook(() => useProcessedChecklist(data));
98
+ expect(result.current[0].iconUrl).toBeUndefined();
99
+ });
100
+
101
+ it("sets anchorId to undefined when anchorId is missing", () => {
102
+ const data = makeChecklistData([{ checkListTitle: "No anchor" }]);
103
+ const { result } = renderHook(() => useProcessedChecklist(data));
104
+ expect(result.current[0].anchorId).toBeUndefined();
105
+ });
106
+
107
+ it("uses empty string for title when checkListTitle is null", () => {
108
+ const data = makeChecklistData([{ checkListTitle: null }]);
109
+ const { result } = renderHook(() => useProcessedChecklist(data));
110
+ // toDocument("") returns null, renderContentfulRichText returns null, falls back to ""
111
+ expect(result.current[0].title).toBe("");
112
+ });
113
+
114
+ it("handles item with undefined checkListTitle", () => {
115
+ const data = makeChecklistData([{}]);
116
+ const { result } = renderHook(() => useProcessedChecklist(data));
117
+ expect(result.current[0].title).toBe("");
118
+ });
119
+
120
+ it("handles null item in items array", () => {
121
+ const data = makeChecklistData([null]);
122
+ const { result } = renderHook(() => useProcessedChecklist(data));
123
+ expect(result.current[0].title).toBe("");
124
+ expect(result.current[0].iconUrl).toBeUndefined();
125
+ expect(result.current[0].anchorId).toBeUndefined();
126
+ });
127
+
128
+ it("handles undefined item in items array", () => {
129
+ const data = makeChecklistData([undefined]);
130
+ const { result } = renderHook(() => useProcessedChecklist(data));
131
+ expect(result.current[0].title).toBe("");
132
+ });
133
+
134
+ it("passes richTextClassName to renderContentfulRichText in full mode", () => {
135
+ const data = makeChecklistData([{ checkListTitle: "Styled" }]);
136
+ const { result } = renderHook(() =>
137
+ useProcessedChecklist(data, false, undefined, "full-class")
138
+ );
139
+ expect(result.current[0].title).toBe("rendered:Styled:full-class");
140
+ });
141
+
142
+ it("passes richTextOptions in full mode", () => {
143
+ const data = makeChecklistData([{ checkListTitle: "Opts" }]);
144
+ const customOpts = { renderMark: { [MARKS.BOLD]: (t: any) => t } };
145
+ const { result } = renderHook(() =>
146
+ useProcessedChecklist(data, false, customOpts, "opts-class")
147
+ );
148
+ expect(result.current[0].title).toBe("rendered:Opts:opts-class");
149
+ });
150
+
151
+ it("handles item with icon but no url", () => {
152
+ const data = makeChecklistData([{ checkListTitle: "No url", icon: {} }]);
153
+ const { result } = renderHook(() => useProcessedChecklist(data));
154
+ expect(result.current[0].iconUrl).toBeUndefined();
155
+ });
156
+ });
157
+
158
+ describe("Titles only mode (titlesOnly=true)", () => {
159
+ it("returns string array of rendered titles", () => {
160
+ const data = makeChecklistData([
161
+ { checkListTitle: "Title A" },
162
+ { checkListTitle: "Title B" },
163
+ ]);
164
+ const { result } = renderHook(() => useProcessedChecklist(data, true));
165
+ expect(result.current).toHaveLength(2);
166
+ expect(result.current[0]).toBe("rendered:Title A:");
167
+ expect(result.current[1]).toBe("rendered:Title B:");
168
+ });
169
+
170
+ it("returns empty string for null checkListTitle", () => {
171
+ const data = makeChecklistData([{ checkListTitle: null }]);
172
+ const { result } = renderHook(() => useProcessedChecklist(data, true));
173
+ expect(result.current[0]).toBe("");
174
+ });
175
+
176
+ it("returns empty array when no items", () => {
177
+ const { result } = renderHook(() =>
178
+ useProcessedChecklist(undefined, true)
179
+ );
180
+ expect(result.current).toEqual([]);
181
+ });
182
+
183
+ it("handles null item in items array in titlesOnly mode", () => {
184
+ const data = makeChecklistData([null]);
185
+ const { result } = renderHook(() => useProcessedChecklist(data, true));
186
+ expect(result.current[0]).toBe("");
187
+ });
188
+
189
+ it("handles undefined item in items array in titlesOnly mode", () => {
190
+ const data = makeChecklistData([undefined]);
191
+ const { result } = renderHook(() => useProcessedChecklist(data, true));
192
+ expect(result.current[0]).toBe("");
193
+ });
194
+
195
+ it("passes richTextOptions in titlesOnly mode", () => {
196
+ const data = makeChecklistData([{ checkListTitle: "Opt" }]);
197
+ const customOpts = { renderMark: {} };
198
+ const { result } = renderHook(() =>
199
+ useProcessedChecklist(data, true, customOpts, "title-class")
200
+ );
201
+ expect(result.current[0]).toBe("rendered:Opt:title-class");
202
+ });
203
+ });
204
+
205
+ describe("richTextOptions and richTextClassName", () => {
206
+ it("passes richTextClassName to renderContentfulRichText", () => {
207
+ const data = makeChecklistData([{ checkListTitle: "Styled" }]);
208
+ const { result } = renderHook(() =>
209
+ useProcessedChecklist(data, true, undefined, "custom-class")
210
+ );
211
+ expect(result.current[0]).toBe("rendered:Styled:custom-class");
212
+ });
213
+
214
+ it("passes richTextOptions through", () => {
215
+ const data = makeChecklistData([{ checkListTitle: "Options" }]);
216
+ const customOpts = { renderMark: {} };
217
+ const { result } = renderHook(() =>
218
+ useProcessedChecklist(data, false, customOpts)
219
+ );
220
+ // Should still produce results (richTextOptions is passed but our mock ignores it)
221
+ expect(result.current).toHaveLength(1);
222
+ });
223
+ });
224
+
225
+ describe("Memoization", () => {
226
+ it("returns same reference when inputs don't change", () => {
227
+ const data = makeChecklistData([{ checkListTitle: "Memo" }]);
228
+ const { result, rerender } = renderHook(
229
+ ({ d }) => useProcessedChecklist(d),
230
+ { initialProps: { d: data } }
231
+ );
232
+ const first = result.current;
233
+ rerender({ d: data });
234
+ expect(result.current).toBe(first);
235
+ });
236
+
237
+ it("recomputes when checklistData changes", () => {
238
+ const data1 = makeChecklistData([{ checkListTitle: "V1" }]);
239
+ const data2 = makeChecklistData([{ checkListTitle: "V2" }]);
240
+ const { result, rerender } = renderHook(
241
+ ({ d }) => useProcessedChecklist(d),
242
+ { initialProps: { d: data1 } }
243
+ );
244
+ const first = result.current;
245
+ rerender({ d: data2 });
246
+ expect(result.current).not.toBe(first);
247
+ });
248
+
249
+ it("recomputes when titlesOnly changes", () => {
250
+ const data = makeChecklistData([{ checkListTitle: "Toggle" }]);
251
+ const { result, rerender } = renderHook(
252
+ ({ t }) => useProcessedChecklist(data, t),
253
+ { initialProps: { t: false as boolean } }
254
+ );
255
+ const first = result.current;
256
+ rerender({ t: true });
257
+ expect(result.current).not.toBe(first);
258
+ });
259
+ });
260
+ });
261
+
262
+ describe("label1BoldOptions", () => {
263
+ it("exports BOLD mark renderer that wraps in span with label1 class", () => {
264
+ const renderer = label1BoldOptions.renderMark[MARKS.BOLD];
265
+ expect(renderer).toBeDefined();
266
+ });
267
+
268
+ it("renders bold text in a span with label1 className", () => {
269
+ const renderer = label1BoldOptions.renderMark[MARKS.BOLD];
270
+ const result = renderer("Bold text");
271
+ // createElement returns a React element with type 'span' and className 'label1'
272
+ expect(result).toBeDefined();
273
+ expect((result as any).type).toBe("span");
274
+ expect((result as any).props.className).toBe("label1");
275
+ expect((result as any).props.children).toBe("Bold text");
276
+ });
277
+ });
@@ -1,63 +1,63 @@
1
- import { createElement, useMemo, type ReactNode } from "react";
2
- import { renderContentfulRichText } from "./use-contentful-rich-text";
3
-
4
- import { type Options } from "@contentful/rich-text-react-renderer";
5
- import { MARKS } from "@contentful/rich-text-types";
6
- import { toDocument } from "@shared/utils/contentful/to-document";
7
-
8
- export const label1BoldOptions = {
9
- renderMark: {
10
- [MARKS.BOLD]: (text: ReactNode) =>
11
- createElement("span", { className: "label1" }, text),
12
- },
13
- };
14
-
15
- export interface ProcessedChecklistItem {
16
- title: string;
17
- iconUrl?: string;
18
- anchorId?: string;
19
- }
20
-
21
- // titlesOnly as a boolean parameter with a conditional return type
22
- export function useProcessedChecklist<T extends boolean = false>(
23
- checklistData?: any,
24
- titlesOnly?: T,
25
- richTextOptions?: Options,
26
- richTextClassName?: string
27
- ): T extends true ? string[] : ProcessedChecklistItem[] {
28
- return useMemo(() => {
29
- const rawItems: any[] = checklistData?.list?.items ?? [];
30
-
31
- if (titlesOnly) {
32
- return rawItems.map(
33
- item =>
34
- renderContentfulRichText(
35
- toDocument(item?.checkListTitle ?? ""),
36
- undefined,
37
- richTextClassName,
38
- undefined,
39
- richTextOptions
40
- ) ?? ""
41
- ) as any;
42
- }
43
-
44
- return rawItems.map(item => ({
45
- title:
46
- renderContentfulRichText(
47
- toDocument(item?.checkListTitle ?? ""),
48
- undefined,
49
- richTextClassName,
50
- undefined,
51
- richTextOptions
52
- ) ?? "",
53
- iconUrl: item?.icon?.url ?? undefined,
54
- anchorId: item?.anchorId ?? undefined,
55
- })) as any;
56
- }, [checklistData, titlesOnly, richTextOptions, richTextClassName]) as any;
57
- }
58
-
59
- // Usage:
60
- // Full items
61
- //const items = useProcessedChecklist(checklistData); // ProcessedChecklistItem[]
62
- // Titles only
63
- //const titles = useProcessedChecklist(checklistData, true); // string[]
1
+ import { createElement, useMemo, type ReactNode } from "react";
2
+ import { renderContentfulRichText } from "./use-contentful-rich-text";
3
+
4
+ import { type Options } from "@contentful/rich-text-react-renderer";
5
+ import { MARKS } from "@contentful/rich-text-types";
6
+ import { toDocument } from "@shared/utils/contentful/to-document";
7
+
8
+ export const label1BoldOptions = {
9
+ renderMark: {
10
+ [MARKS.BOLD]: (text: ReactNode) =>
11
+ createElement("span", { className: "label1" }, text),
12
+ },
13
+ };
14
+
15
+ export interface ProcessedChecklistItem {
16
+ title: string;
17
+ iconUrl?: string;
18
+ anchorId?: string;
19
+ }
20
+
21
+ // titlesOnly as a boolean parameter with a conditional return type
22
+ export function useProcessedChecklist<T extends boolean = false>(
23
+ checklistData?: any,
24
+ titlesOnly?: T,
25
+ richTextOptions?: Options,
26
+ richTextClassName?: string
27
+ ): T extends true ? string[] : ProcessedChecklistItem[] {
28
+ return useMemo(() => {
29
+ const rawItems: any[] = checklistData?.list?.items ?? [];
30
+
31
+ if (titlesOnly) {
32
+ return rawItems.map(
33
+ item =>
34
+ renderContentfulRichText(
35
+ toDocument(item?.checkListTitle ?? ""),
36
+ undefined,
37
+ richTextClassName,
38
+ undefined,
39
+ richTextOptions
40
+ ) ?? ""
41
+ ) as any;
42
+ }
43
+
44
+ return rawItems.map(item => ({
45
+ title:
46
+ renderContentfulRichText(
47
+ toDocument(item?.checkListTitle ?? ""),
48
+ undefined,
49
+ richTextClassName,
50
+ undefined,
51
+ richTextOptions
52
+ ) ?? "",
53
+ iconUrl: item?.icon?.url ?? undefined,
54
+ anchorId: item?.anchorId ?? undefined,
55
+ })) as any;
56
+ }, [checklistData, titlesOnly, richTextOptions, richTextClassName]) as any;
57
+ }
58
+
59
+ // Usage:
60
+ // Full items
61
+ //const items = useProcessedChecklist(checklistData); // ProcessedChecklistItem[]
62
+ // Titles only
63
+ //const titles = useProcessedChecklist(checklistData, true); // string[]
@@ -0,0 +1,134 @@
1
+ import useBodyScrollLock from "./use-body-scroll-lock";
2
+
3
+ import { cleanup, renderHook } from "@testing-library/react";
4
+
5
+ describe("useBodyScrollLock", () => {
6
+ beforeEach(() => {
7
+ cleanup();
8
+ document.body.style.overflow = "";
9
+ document.body.style.marginRight = "";
10
+ });
11
+
12
+ afterEach(() => {
13
+ cleanup();
14
+ document.body.style.overflow = "";
15
+ document.body.style.marginRight = "";
16
+ });
17
+
18
+ describe("Scroll locking (isOpen=true)", () => {
19
+ it("sets body overflow to hidden when isOpen is true", () => {
20
+ renderHook(() => useBodyScrollLock(true, false));
21
+ expect(document.body.style.overflow).toBe("hidden");
22
+ });
23
+
24
+ it("sets marginRight based on scrollbar width when isOpen is true", () => {
25
+ renderHook(() => useBodyScrollLock(true, false));
26
+ const scrollbarWidth =
27
+ window.innerWidth - document.documentElement.clientWidth;
28
+ expect(document.body.style.marginRight).toBe(`${scrollbarWidth}px`);
29
+ });
30
+ });
31
+
32
+ describe("Scroll unlocking (isOpen=false)", () => {
33
+ it("resets overflow to unset when isOpen=false and hideScrollOnIsOpenFalse=true", () => {
34
+ renderHook(() => useBodyScrollLock(false, true));
35
+ expect(document.body.style.overflow).toBe("unset");
36
+ });
37
+
38
+ it("preserves existing overflow when isOpen=false and hideScrollOnIsOpenFalse=false", () => {
39
+ document.body.style.overflow = "auto";
40
+ renderHook(() => useBodyScrollLock(false, false));
41
+ expect(document.body.style.overflow).toBe("auto");
42
+ });
43
+
44
+ it("attempts to set marginRight to unset when isOpen is false", () => {
45
+ const calls: string[] = [];
46
+ const original = Object.getOwnPropertyDescriptor(
47
+ CSSStyleDeclaration.prototype,
48
+ "marginRight"
49
+ );
50
+ Object.defineProperty(document.body.style, "marginRight", {
51
+ set(val: string) {
52
+ calls.push(val);
53
+ },
54
+ get() {
55
+ return "";
56
+ },
57
+ configurable: true,
58
+ });
59
+ renderHook(() => useBodyScrollLock(false, false));
60
+ expect(calls).toContain("unset");
61
+ // Restore
62
+ if (original) {
63
+ Object.defineProperty(
64
+ CSSStyleDeclaration.prototype,
65
+ "marginRight",
66
+ original
67
+ );
68
+ }
69
+ });
70
+
71
+ it("attempts to set marginRight to unset for isOpen false regardless of hideScroll", () => {
72
+ const calls: string[] = [];
73
+ Object.defineProperty(document.body.style, "marginRight", {
74
+ set(val: string) {
75
+ calls.push(val);
76
+ },
77
+ get() {
78
+ return "";
79
+ },
80
+ configurable: true,
81
+ });
82
+ renderHook(() => useBodyScrollLock(false, true));
83
+ expect(calls).toContain("unset");
84
+ });
85
+ });
86
+
87
+ describe("Cleanup on unmount", () => {
88
+ it("resets body overflow to unset on unmount", () => {
89
+ const { unmount } = renderHook(() => useBodyScrollLock(true, false));
90
+ expect(document.body.style.overflow).toBe("hidden");
91
+ unmount();
92
+ expect(document.body.style.overflow).toBe("unset");
93
+ });
94
+
95
+ it("resets overflow on unmount even when isOpen was false", () => {
96
+ const { unmount } = renderHook(() => useBodyScrollLock(false, true));
97
+ unmount();
98
+ expect(document.body.style.overflow).toBe("unset");
99
+ });
100
+ });
101
+
102
+ describe("Dependency changes", () => {
103
+ it("locks scroll when isOpen changes from false to true", () => {
104
+ const { rerender } = renderHook(
105
+ ({ isOpen, hide }) => useBodyScrollLock(isOpen, hide),
106
+ { initialProps: { isOpen: false, hide: false } }
107
+ );
108
+ expect(document.body.style.overflow).not.toBe("hidden");
109
+ rerender({ isOpen: true, hide: false });
110
+ expect(document.body.style.overflow).toBe("hidden");
111
+ });
112
+
113
+ it("unlocks scroll when isOpen changes from true to false with hideScroll=true", () => {
114
+ const { rerender } = renderHook(
115
+ ({ isOpen, hide }) => useBodyScrollLock(isOpen, hide),
116
+ { initialProps: { isOpen: true, hide: true } }
117
+ );
118
+ expect(document.body.style.overflow).toBe("hidden");
119
+ rerender({ isOpen: false, hide: true });
120
+ expect(document.body.style.overflow).toBe("unset");
121
+ });
122
+
123
+ it("runs cleanup then effect when dependencies change", () => {
124
+ const { rerender } = renderHook(
125
+ ({ isOpen, hide }) => useBodyScrollLock(isOpen, hide),
126
+ { initialProps: { isOpen: true, hide: false } }
127
+ );
128
+ expect(document.body.style.overflow).toBe("hidden");
129
+ // Change hideScrollOnIsOpenFalse - triggers cleanup + re-run
130
+ rerender({ isOpen: true, hide: true });
131
+ expect(document.body.style.overflow).toBe("hidden");
132
+ });
133
+ });
134
+ });
@@ -1,34 +1,34 @@
1
- "use client";
2
-
3
- import { useEffect } from "react";
4
-
5
- /**
6
- * A custom hook that locks the body scroll based on the `isOpen` and `hideScrollOnIsOpenFalse` flags.
7
- *
8
- * @param {boolean} isOpen - Determines if the scroll should be locked (hidden).
9
- * @param {boolean} hideScrollOnIsOpenFalse - Determines if the scroll should be reset when `isOpen` is false.
10
- */
11
- function useBodyScrollLock(isOpen: boolean, hideScrollOnIsOpenFalse: boolean) {
12
- useEffect(() => {
13
- if (typeof document === "undefined") return;
14
-
15
- // To avoid content shifting when the scrollbar appears/disappears
16
- const scrollbarWidth =
17
- window.innerWidth - document.documentElement.clientWidth;
18
- document.body.style.marginRight = isOpen ? `${scrollbarWidth}px` : "unset";
19
-
20
- // Set the body overflow style based on the `isOpen` and `hideScrollOnIsOpenFalse` values.
21
- document.body.style.overflow = isOpen
22
- ? "hidden"
23
- : hideScrollOnIsOpenFalse
24
- ? "unset"
25
- : document.body.style.overflow;
26
-
27
- // Cleanup function to reset body overflow when the component unmounts or dependencies change.
28
- return () => {
29
- document.body.style.overflow = "unset";
30
- };
31
- }, [isOpen, hideScrollOnIsOpenFalse]);
32
- }
33
-
34
- export default useBodyScrollLock;
1
+ "use client";
2
+
3
+ import { useEffect } from "react";
4
+
5
+ /**
6
+ * A custom hook that locks the body scroll based on the `isOpen` and `hideScrollOnIsOpenFalse` flags.
7
+ *
8
+ * @param {boolean} isOpen - Determines if the scroll should be locked (hidden).
9
+ * @param {boolean} hideScrollOnIsOpenFalse - Determines if the scroll should be reset when `isOpen` is false.
10
+ */
11
+ function useBodyScrollLock(isOpen: boolean, hideScrollOnIsOpenFalse: boolean) {
12
+ useEffect(() => {
13
+ if (typeof document === "undefined") return;
14
+
15
+ // To avoid content shifting when the scrollbar appears/disappears
16
+ const scrollbarWidth =
17
+ window.innerWidth - document.documentElement.clientWidth;
18
+ document.body.style.marginRight = isOpen ? `${scrollbarWidth}px` : "unset";
19
+
20
+ // Set the body overflow style based on the `isOpen` and `hideScrollOnIsOpenFalse` values.
21
+ document.body.style.overflow = isOpen
22
+ ? "hidden"
23
+ : hideScrollOnIsOpenFalse
24
+ ? "unset"
25
+ : document.body.style.overflow;
26
+
27
+ // Cleanup function to reset body overflow when the component unmounts or dependencies change.
28
+ return () => {
29
+ document.body.style.overflow = "unset";
30
+ };
31
+ }, [isOpen, hideScrollOnIsOpenFalse]);
32
+ }
33
+
34
+ export default useBodyScrollLock;