@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,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;