@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
@@ -1,309 +1,309 @@
1
- import { useMemo, type ReactNode } from "react";
2
-
3
- import {
4
- documentToReactComponents,
5
- type Options,
6
- } from "@contentful/rich-text-react-renderer";
7
- import {
8
- BLOCKS,
9
- INLINES,
10
- MARKS,
11
- type Document,
12
- } from "@contentful/rich-text-types";
13
- import { Checklist } from "@shared/components/checklist";
14
- import { Link } from "@shared/components/link";
15
- import { MaterialIcon } from "@shared/components/material-icon";
16
- import { Text } from "@shared/components/text";
17
- import { toDocument } from "@shared/utils/contentful/to-document";
18
-
19
- const defaultOptions: Options = {
20
- renderMark: {
21
- [MARKS.BOLD]: text => <strong className="label3">{text}</strong>,
22
- [MARKS.ITALIC]: text => <em>{text}</em>,
23
- [MARKS.UNDERLINE]: text => <u>{text}</u>,
24
- [MARKS.CODE]: text => <code>{text}</code>,
25
- },
26
- renderNode: {
27
- [BLOCKS.PARAGRAPH]: (_node, children) => (
28
- <div className="body3 mb-4">{children}</div>
29
- ),
30
- [BLOCKS.HEADING_1]: (node, children) => {
31
- return (
32
- <Text as="h1" className={"heading2 md:heading1"}>
33
- {children}
34
- </Text>
35
- );
36
- },
37
- [BLOCKS.HEADING_2]: (node, children) => (
38
- <Text as="h2" className={"heading6 md:heading5"}>
39
- {children}
40
- </Text>
41
- ),
42
- [BLOCKS.HEADING_3]: (node, children) => (
43
- <Text as="h3" className={"heading6 md:heading5"}>
44
- {children}
45
- </Text>
46
- ),
47
- [BLOCKS.HEADING_4]: (node, children) => (
48
- <Text as="h3" className={"headingClass"}>
49
- {children}
50
- </Text>
51
- ),
52
- [BLOCKS.HEADING_5]: (node, children) => (
53
- <Text as="h3" className={"heading6 md:heading5"}>
54
- {children}
55
- </Text>
56
- ),
57
- [BLOCKS.HEADING_6]: (node, children) => (
58
- <Text as="h3" className={"heading6"}>
59
- {children}
60
- </Text>
61
- ),
62
- [BLOCKS.QUOTE]: (_node, children) => <blockquote>{children}</blockquote>,
63
- [BLOCKS.UL_LIST]: (_node, children) => <ul>{children}</ul>,
64
- [BLOCKS.OL_LIST]: (_node, children) => <ol>{children}</ol>,
65
- [BLOCKS.LIST_ITEM]: (_node, children) => <li>{children}</li>,
66
-
67
- [INLINES.HYPERLINK]: (node, children) => {
68
- const url = (node as any)?.data?.uri as string;
69
- const external = /^https?:\/\//.test(url);
70
- return (
71
- <a
72
- href={url}
73
- target={external ? "_blank" : undefined}
74
- rel={external ? "noopener noreferrer" : undefined}
75
- >
76
- {children}
77
- </a>
78
- );
79
- },
80
- [BLOCKS.EMBEDDED_ASSET]: node => {
81
- const target = (node as any)?.data?.target;
82
- const fields = target?.fields;
83
- const url = target?.url || fields?.file?.url;
84
- const alt = fields?.title || fields?.description || "Embedded asset";
85
- if (!url) return null;
86
- const src = url.startsWith("//") ? `https:${url}` : url;
87
- return <img src={src} alt={alt} style={{ maxWidth: "100%" }} />;
88
- },
89
-
90
- [BLOCKS.EMBEDDED_ENTRY]: node => {
91
- const entry = (node as any)?.data?.target;
92
- const ctid = entry?.sys?.contentType?.sys?.id;
93
- switch (ctid) {
94
- case "callout":
95
- return (
96
- <aside
97
- style={{ border: "1px solid #ddd", padding: 12, borderRadius: 8 }}
98
- >
99
- <strong>{entry.fields.title}</strong>
100
- <div>{entry.fields.body}</div>
101
- </aside>
102
- );
103
- default:
104
- return null;
105
- }
106
- },
107
-
108
- [INLINES.EMBEDDED_ENTRY]: node => {
109
- const entry = (node as any)?.data?.target;
110
- const ctid = entry?.sys?.contentType?.sys?.id;
111
- switch (ctid) {
112
- case "componentCheckList":
113
- return (
114
- <span
115
- style={{
116
- display: "inline-flex",
117
- alignItems: "center",
118
- gap: "4px",
119
- padding: "2px 6px",
120
- backgroundColor: "#f0fdf4",
121
- border: "1px solid #bbf7d0",
122
- borderRadius: "4px",
123
- fontSize: "0.9em",
124
- }}
125
- >
126
- ✅ {entry.fields.title}
127
- </span>
128
- );
129
- default:
130
- return <span>{entry?.fields?.title ?? ""}</span>;
131
- }
132
- },
133
- },
134
- };
135
- /**
136
- * Utility function to render Contentful Rich Text.
137
- * Safe to use inside .map() loops.
138
- */
139
- export function renderContentfulRichText(
140
- doc: Document | null | undefined,
141
- target?: boolean,
142
- className: string = "body1",
143
- linkClassName: string = "body1 font-bold",
144
- options?: Options
145
- ): ReactNode | null {
146
- if (!doc || !Array.isArray(doc.content)) return null;
147
-
148
- const merged: Options = {
149
- ...defaultOptions,
150
- ...options,
151
- renderNode: {
152
- ...defaultOptions.renderNode,
153
- ...options?.renderNode,
154
- // Logic for links based on the isTargetBlank flag
155
- [BLOCKS.PARAGRAPH]: (_node, children) => (
156
- <div className={className}>{children}</div>
157
- ),
158
- [INLINES.HYPERLINK]: (node, children) => {
159
- const url = (node as any)?.data?.uri as string;
160
- const external = /^https?:\/\//.test(url);
161
-
162
- // Priority: 1. Manual flag from Contentful, 2. Regex check for external URLs
163
- const shouldOpenBlank = target ?? external;
164
-
165
- return (
166
- <Link
167
- href={url}
168
- target={shouldOpenBlank ? "_blank" : "_self"}
169
- rel={shouldOpenBlank ? "noopener noreferrer" : undefined}
170
- variant="default"
171
- className={linkClassName}
172
- >
173
- {children}
174
- </Link>
175
- );
176
- },
177
- },
178
- };
179
-
180
- // Directly return the rendered components without useMemo
181
- return documentToReactComponents(doc, merged);
182
- }
183
-
184
- export function useContentfulRichText(
185
- doc: Document | null | undefined,
186
- options?: Options
187
- ): ReactNode | null {
188
- return useMemo(() => {
189
- if (!doc || !Array.isArray(doc.content)) return null;
190
- const merged: Options = {
191
- ...defaultOptions,
192
- renderMark: { ...defaultOptions.renderMark, ...options?.renderMark },
193
- renderNode: { ...defaultOptions.renderNode, ...options?.renderNode },
194
- };
195
- return documentToReactComponents(doc, merged);
196
- }, [doc, options]);
197
- }
198
- // 1. Remove useMemo and React Hook imports from this specific function's logic
199
- export function renderContentfulRichTextTable(
200
- doc: Document | null | undefined,
201
- links?: any
202
- ) {
203
- if (!doc || !Array.isArray(doc.content)) return null;
204
-
205
- const options: Options = {
206
- ...defaultOptions, // Spread defaultOptions so <p> and <strong> still work!
207
- renderMark: {
208
- ...defaultOptions.renderMark,
209
- [MARKS.BOLD]: text => (
210
- <strong className="label4 md:label2">{text}</strong>
211
- ),
212
- },
213
- renderNode: {
214
- ...defaultOptions.renderNode,
215
- [BLOCKS.PARAGRAPH]: (_node, children) => <>{children}</>,
216
- [BLOCKS.TABLE]: (node: any, children) => (
217
- <div className="comparison-table-wrapper w-full overflow-x-auto border-none md:overflow-hidden">
218
- <table
219
- className={`responsive-table w-full table-fixed border-collapse ${
220
- node.content[0]?.content?.length > 2
221
- ? "min-w-[100.1%]"
222
- : "min-w-full"
223
- }`}
224
- >
225
- <tbody>{children}</tbody>
226
- </table>
227
- </div>
228
- ),
229
- [BLOCKS.TABLE_ROW]: (node, children) => (
230
- <tr className="border-b border-gray-200 last:border-0">{children}</tr>
231
- ),
232
- [BLOCKS.TABLE_HEADER_CELL]: (node: any, children) => {
233
- const isScrollable = node.parent?.content.length > 2;
234
- return (
235
- <th
236
- className={`label4 break-words py-4 text-center md:label2 ${isScrollable ? "sticky left-0 z-20 w-[50vw] min-w-[50vw] first:z-30 first:w-[50vw] first:min-w-[50vw] first:border-r" : "w-1/4 first:w-1/2"} `}
237
- >
238
- {children}
239
- </th>
240
- );
241
- },
242
- [BLOCKS.TABLE_CELL]: (node: any, children) => {
243
- const isScrollable = node.parent?.content.length > 2;
244
-
245
- // Logic to check for "yes" or "no"
246
- // children[0].props.children is usually where the text sits if it's in a <p> tag
247
- const rawText = node.content[0]?.content[0]?.value
248
- ?.toLowerCase()
249
- .trim();
250
-
251
- const renderContent = () => {
252
- if (rawText === "yes") {
253
- return (
254
- <MaterialIcon name="check_circle" color="#24A76A" fill={1} />
255
- );
256
- }
257
- if (rawText === "no") {
258
- return <MaterialIcon name="cancel" color="#CECECE" fill={1} />;
259
- }
260
- return children;
261
- };
262
-
263
- return (
264
- <td
265
- className={`rt-table-cell footnote break-words bg-white py-2 md:py-4 text-center align-top leading-5 text-text md:body2 first:text-left md:leading-7 ${
266
- isScrollable
267
- ? "w-[50vw] min-w-[50vw] first:sticky first:left-0 first:z-10 first:w-[50vw] first:min-w-[50vw] first:border-r"
268
- : "w-1/4 first:w-1/2"
269
- } `}
270
- >
271
- <> {renderContent()}</>
272
- </td>
273
- );
274
- },
275
- [INLINES.EMBEDDED_ENTRY]: node => {
276
- const entryId = node.data.target.sys.id;
277
-
278
- const entry = links?.entries?.inline?.find(
279
- (e: any) => e.sys.id === entryId
280
- );
281
- if (!entry) return null;
282
-
283
- if (entry.__typename === "ComponentCheckList") {
284
- // FIX: Don't use a Hook here.
285
- // Map the items manually or use a non-hook helper function.
286
- const items =
287
- entry.list?.items?.map((item: any) =>
288
- renderContentfulRichText(
289
- toDocument(item?.checkListTitle ?? ""),
290
- true,
291
- ""
292
- )
293
- ) || [];
294
- return (
295
- <Checklist
296
- items={items}
297
- listIconName="disc"
298
- listItemClassName="items-baseline footnote md:body2 leading-5 md:leading-7 text-text"
299
- />
300
- );
301
- }
302
-
303
- return <span>{entry.title || ""}</span>;
304
- },
305
- },
306
- };
307
-
308
- return documentToReactComponents(doc, options);
309
- }
1
+ import { useMemo, type ReactNode } from "react";
2
+
3
+ import {
4
+ documentToReactComponents,
5
+ type Options,
6
+ } from "@contentful/rich-text-react-renderer";
7
+ import {
8
+ BLOCKS,
9
+ INLINES,
10
+ MARKS,
11
+ type Document,
12
+ } from "@contentful/rich-text-types";
13
+ import { Checklist } from "@shared/components/checklist";
14
+ import { Link } from "@shared/components/link";
15
+ import { MaterialIcon } from "@shared/components/material-icon";
16
+ import { Text } from "@shared/components/text";
17
+ import { toDocument } from "@shared/utils/contentful/to-document";
18
+
19
+ const defaultOptions: Options = {
20
+ renderMark: {
21
+ [MARKS.BOLD]: text => <strong className="label3">{text}</strong>,
22
+ [MARKS.ITALIC]: text => <em>{text}</em>,
23
+ [MARKS.UNDERLINE]: text => <u>{text}</u>,
24
+ [MARKS.CODE]: text => <code>{text}</code>,
25
+ },
26
+ renderNode: {
27
+ [BLOCKS.PARAGRAPH]: (_node, children) => (
28
+ <div className="body3 mb-4">{children}</div>
29
+ ),
30
+ [BLOCKS.HEADING_1]: (node, children) => {
31
+ return (
32
+ <Text as="h1" className={"heading2 md:heading1"}>
33
+ {children}
34
+ </Text>
35
+ );
36
+ },
37
+ [BLOCKS.HEADING_2]: (node, children) => (
38
+ <Text as="h2" className={"heading6 md:heading5"}>
39
+ {children}
40
+ </Text>
41
+ ),
42
+ [BLOCKS.HEADING_3]: (node, children) => (
43
+ <Text as="h3" className={"heading6 md:heading5"}>
44
+ {children}
45
+ </Text>
46
+ ),
47
+ [BLOCKS.HEADING_4]: (node, children) => (
48
+ <Text as="h3" className={"headingClass"}>
49
+ {children}
50
+ </Text>
51
+ ),
52
+ [BLOCKS.HEADING_5]: (node, children) => (
53
+ <Text as="h3" className={"heading6 md:heading5"}>
54
+ {children}
55
+ </Text>
56
+ ),
57
+ [BLOCKS.HEADING_6]: (node, children) => (
58
+ <Text as="h3" className={"heading6"}>
59
+ {children}
60
+ </Text>
61
+ ),
62
+ [BLOCKS.QUOTE]: (_node, children) => <blockquote>{children}</blockquote>,
63
+ [BLOCKS.UL_LIST]: (_node, children) => <ul>{children}</ul>,
64
+ [BLOCKS.OL_LIST]: (_node, children) => <ol>{children}</ol>,
65
+ [BLOCKS.LIST_ITEM]: (_node, children) => <li>{children}</li>,
66
+
67
+ [INLINES.HYPERLINK]: (node, children) => {
68
+ const url = (node as any)?.data?.uri as string;
69
+ const external = /^https?:\/\//.test(url);
70
+ return (
71
+ <a
72
+ href={url}
73
+ target={external ? "_blank" : undefined}
74
+ rel={external ? "noopener noreferrer" : undefined}
75
+ >
76
+ {children}
77
+ </a>
78
+ );
79
+ },
80
+ [BLOCKS.EMBEDDED_ASSET]: node => {
81
+ const target = (node as any)?.data?.target;
82
+ const fields = target?.fields;
83
+ const url = target?.url || fields?.file?.url;
84
+ const alt = fields?.title || fields?.description || "Embedded asset";
85
+ if (!url) return null;
86
+ const src = url.startsWith("//") ? `https:${url}` : url;
87
+ return <img src={src} alt={alt} style={{ maxWidth: "100%" }} />;
88
+ },
89
+
90
+ [BLOCKS.EMBEDDED_ENTRY]: node => {
91
+ const entry = (node as any)?.data?.target;
92
+ const ctid = entry?.sys?.contentType?.sys?.id;
93
+ switch (ctid) {
94
+ case "callout":
95
+ return (
96
+ <aside
97
+ style={{ border: "1px solid #ddd", padding: 12, borderRadius: 8 }}
98
+ >
99
+ <strong>{entry.fields.title}</strong>
100
+ <div>{entry.fields.body}</div>
101
+ </aside>
102
+ );
103
+ default:
104
+ return null;
105
+ }
106
+ },
107
+
108
+ [INLINES.EMBEDDED_ENTRY]: node => {
109
+ const entry = (node as any)?.data?.target;
110
+ const ctid = entry?.sys?.contentType?.sys?.id;
111
+ switch (ctid) {
112
+ case "componentCheckList":
113
+ return (
114
+ <span
115
+ style={{
116
+ display: "inline-flex",
117
+ alignItems: "center",
118
+ gap: "4px",
119
+ padding: "2px 6px",
120
+ backgroundColor: "#f0fdf4",
121
+ border: "1px solid #bbf7d0",
122
+ borderRadius: "4px",
123
+ fontSize: "0.9em",
124
+ }}
125
+ >
126
+ ✅ {entry.fields.title}
127
+ </span>
128
+ );
129
+ default:
130
+ return <span>{entry?.fields?.title ?? ""}</span>;
131
+ }
132
+ },
133
+ },
134
+ };
135
+ /**
136
+ * Utility function to render Contentful Rich Text.
137
+ * Safe to use inside .map() loops.
138
+ */
139
+ export function renderContentfulRichText(
140
+ doc: Document | null | undefined,
141
+ target?: boolean,
142
+ className: string = "body1",
143
+ linkClassName: string = "body1 font-bold",
144
+ options?: Options
145
+ ): ReactNode | null {
146
+ if (!doc || !Array.isArray(doc.content)) return null;
147
+
148
+ const merged: Options = {
149
+ ...defaultOptions,
150
+ ...options,
151
+ renderNode: {
152
+ ...defaultOptions.renderNode,
153
+ ...options?.renderNode,
154
+ // Logic for links based on the isTargetBlank flag
155
+ [BLOCKS.PARAGRAPH]: (_node, children) => (
156
+ <div className={className}>{children}</div>
157
+ ),
158
+ [INLINES.HYPERLINK]: (node, children) => {
159
+ const url = (node as any)?.data?.uri as string;
160
+ const external = /^https?:\/\//.test(url);
161
+
162
+ // Priority: 1. Manual flag from Contentful, 2. Regex check for external URLs
163
+ const shouldOpenBlank = target ?? external;
164
+
165
+ return (
166
+ <Link
167
+ href={url}
168
+ target={shouldOpenBlank ? "_blank" : "_self"}
169
+ rel={shouldOpenBlank ? "noopener noreferrer" : undefined}
170
+ variant="default"
171
+ className={linkClassName}
172
+ >
173
+ {children}
174
+ </Link>
175
+ );
176
+ },
177
+ },
178
+ };
179
+
180
+ // Directly return the rendered components without useMemo
181
+ return documentToReactComponents(doc, merged);
182
+ }
183
+
184
+ export function useContentfulRichText(
185
+ doc: Document | null | undefined,
186
+ options?: Options
187
+ ): ReactNode | null {
188
+ return useMemo(() => {
189
+ if (!doc || !Array.isArray(doc.content)) return null;
190
+ const merged: Options = {
191
+ ...defaultOptions,
192
+ renderMark: { ...defaultOptions.renderMark, ...options?.renderMark },
193
+ renderNode: { ...defaultOptions.renderNode, ...options?.renderNode },
194
+ };
195
+ return documentToReactComponents(doc, merged);
196
+ }, [doc, options]);
197
+ }
198
+ // 1. Remove useMemo and React Hook imports from this specific function's logic
199
+ export function renderContentfulRichTextTable(
200
+ doc: Document | null | undefined,
201
+ links?: any
202
+ ) {
203
+ if (!doc || !Array.isArray(doc.content)) return null;
204
+
205
+ const options: Options = {
206
+ ...defaultOptions, // Spread defaultOptions so <p> and <strong> still work!
207
+ renderMark: {
208
+ ...defaultOptions.renderMark,
209
+ [MARKS.BOLD]: text => (
210
+ <strong className="label4 md:label2">{text}</strong>
211
+ ),
212
+ },
213
+ renderNode: {
214
+ ...defaultOptions.renderNode,
215
+ [BLOCKS.PARAGRAPH]: (_node, children) => <>{children}</>,
216
+ [BLOCKS.TABLE]: (node: any, children) => (
217
+ <div className="comparison-table-wrapper w-full overflow-x-auto border-none md:overflow-hidden">
218
+ <table
219
+ className={`responsive-table w-full table-fixed border-collapse ${
220
+ node.content[0]?.content?.length > 2
221
+ ? "min-w-[100.1%]"
222
+ : "min-w-full"
223
+ }`}
224
+ >
225
+ <tbody>{children}</tbody>
226
+ </table>
227
+ </div>
228
+ ),
229
+ [BLOCKS.TABLE_ROW]: (node, children) => (
230
+ <tr className="border-b border-gray-200 last:border-0">{children}</tr>
231
+ ),
232
+ [BLOCKS.TABLE_HEADER_CELL]: (node: any, children) => {
233
+ const isScrollable = node.parent?.content.length > 2;
234
+ return (
235
+ <th
236
+ className={`label4 break-words py-4 text-center md:label2 ${isScrollable ? "sticky left-0 z-20 w-[50vw] min-w-[50vw] first:z-30 first:w-[50vw] first:min-w-[50vw] first:border-r" : "w-1/4 first:w-1/2"} `}
237
+ >
238
+ {children}
239
+ </th>
240
+ );
241
+ },
242
+ [BLOCKS.TABLE_CELL]: (node: any, children) => {
243
+ const isScrollable = node.parent?.content.length > 2;
244
+
245
+ // Logic to check for "yes" or "no"
246
+ // children[0].props.children is usually where the text sits if it's in a <p> tag
247
+ const rawText = node.content[0]?.content[0]?.value
248
+ ?.toLowerCase()
249
+ .trim();
250
+
251
+ const renderContent = () => {
252
+ if (rawText === "yes") {
253
+ return (
254
+ <MaterialIcon name="check_circle" color="#24A76A" fill={1} />
255
+ );
256
+ }
257
+ if (rawText === "no") {
258
+ return <MaterialIcon name="cancel" color="#CECECE" fill={1} />;
259
+ }
260
+ return children;
261
+ };
262
+
263
+ return (
264
+ <td
265
+ className={`rt-table-cell footnote break-words bg-white py-2 text-center align-top leading-5 text-text md:body2 first:text-left md:py-4 md:leading-7 ${
266
+ isScrollable
267
+ ? "w-[50vw] min-w-[50vw] first:sticky first:left-0 first:z-10 first:w-[50vw] first:min-w-[50vw] first:border-r"
268
+ : "w-1/4 first:w-1/2"
269
+ } `}
270
+ >
271
+ <> {renderContent()}</>
272
+ </td>
273
+ );
274
+ },
275
+ [INLINES.EMBEDDED_ENTRY]: node => {
276
+ const entryId = node.data.target.sys.id;
277
+
278
+ const entry = links?.entries?.inline?.find(
279
+ (e: any) => e.sys.id === entryId
280
+ );
281
+ if (!entry) return null;
282
+
283
+ if (entry.__typename === "ComponentCheckList") {
284
+ // FIX: Don't use a Hook here.
285
+ // Map the items manually or use a non-hook helper function.
286
+ const items =
287
+ entry.list?.items?.map((item: any) =>
288
+ renderContentfulRichText(
289
+ toDocument(item?.checkListTitle ?? ""),
290
+ true,
291
+ ""
292
+ )
293
+ ) || [];
294
+ return (
295
+ <Checklist
296
+ items={items}
297
+ listIconName="disc"
298
+ listItemClassName="items-baseline footnote md:body2 leading-5 md:leading-7 text-text"
299
+ />
300
+ );
301
+ }
302
+
303
+ return <span>{entry.title || ""}</span>;
304
+ },
305
+ },
306
+ };
307
+
308
+ return documentToReactComponents(doc, options);
309
+ }