@windstream/react-shared-components 0.1.90 → 0.1.91

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 (210) 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/index.d.ts +2 -2
  7. package/dist/index.esm.js +13 -5
  8. package/dist/index.esm.js.map +1 -1
  9. package/dist/index.js +13 -5
  10. package/dist/index.js.map +1 -1
  11. package/dist/next/index.esm.js +2 -2
  12. package/dist/next/index.esm.js.map +1 -1
  13. package/dist/next/index.js +2 -2
  14. package/dist/next/index.js.map +1 -1
  15. package/dist/styles.css +1 -1
  16. package/dist/utils/index.esm.js +1 -1
  17. package/dist/utils/index.esm.js.map +1 -1
  18. package/dist/utils/index.js +1 -1
  19. package/dist/utils/index.js.map +1 -1
  20. package/package.json +185 -185
  21. package/src/components/accordion/Accordion.stories.tsx +230 -230
  22. package/src/components/accordion/index.tsx +70 -70
  23. package/src/components/accordion/types.ts +12 -12
  24. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  25. package/src/components/alert-card/index.tsx +41 -41
  26. package/src/components/alert-card/types.ts +13 -13
  27. package/src/components/animation-wrapper/index.tsx +129 -129
  28. package/src/components/animation-wrapper/types.ts +11 -11
  29. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  30. package/src/components/brand-button/helpers.ts +35 -35
  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.tsx +27 -27
  35. package/src/components/button/types.ts +14 -14
  36. package/src/components/call-button/CallButton.stories.tsx +324 -324
  37. package/src/components/call-button/index.tsx +106 -106
  38. package/src/components/call-button/types.ts +16 -16
  39. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  40. package/src/components/checkbox/index.tsx +197 -197
  41. package/src/components/checkbox/types.ts +27 -27
  42. package/src/components/checklist/Checklist.stories.tsx +150 -150
  43. package/src/components/checklist/index.tsx +61 -61
  44. package/src/components/checklist/types.ts +17 -17
  45. package/src/components/collapse/Collapse.stories.tsx +255 -255
  46. package/src/components/collapse/index.tsx +46 -46
  47. package/src/components/collapse/types.ts +6 -6
  48. package/src/components/divider/Divider.stories.tsx +205 -205
  49. package/src/components/divider/index.tsx +22 -22
  50. package/src/components/divider/type.ts +3 -3
  51. package/src/components/image/Image.stories.tsx +113 -113
  52. package/src/components/image/index.tsx +25 -25
  53. package/src/components/image/types.ts +40 -40
  54. package/src/components/input/Input.stories.tsx +325 -325
  55. package/src/components/input/index.tsx +177 -177
  56. package/src/components/input/types.ts +37 -37
  57. package/src/components/link/Link.stories.tsx +163 -163
  58. package/src/components/link/index.tsx +116 -116
  59. package/src/components/link/types.ts +25 -25
  60. package/src/components/list/List.stories.tsx +272 -272
  61. package/src/components/list/index.tsx +88 -88
  62. package/src/components/list/list-item/index.tsx +38 -38
  63. package/src/components/list/list-item/types.ts +13 -13
  64. package/src/components/list/types.ts +29 -29
  65. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  66. package/src/components/material-icon/constants.ts +99 -99
  67. package/src/components/material-icon/index.tsx +47 -47
  68. package/src/components/material-icon/types.ts +31 -31
  69. package/src/components/modal/Modal.stories.tsx +171 -171
  70. package/src/components/modal/index.tsx +164 -164
  71. package/src/components/modal/types.ts +24 -24
  72. package/src/components/next-image/index.tsx +74 -74
  73. package/src/components/next-image/types.ts +1 -1
  74. package/src/components/pagination/index.tsx +91 -91
  75. package/src/components/pagination/types.ts +6 -6
  76. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  77. package/src/components/radio-button/index.tsx +75 -75
  78. package/src/components/radio-button/types.ts +21 -21
  79. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  80. package/src/components/see-more/index.tsx +44 -44
  81. package/src/components/see-more/types.ts +4 -4
  82. package/src/components/select/Select.stories.tsx +411 -411
  83. package/src/components/select/index.tsx +155 -155
  84. package/src/components/select/types.ts +36 -36
  85. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  86. package/src/components/select-plan-button/index.tsx +63 -63
  87. package/src/components/select-plan-button/types.ts +17 -17
  88. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  89. package/src/components/skeleton/index.tsx +61 -61
  90. package/src/components/skeleton/types.ts +4 -4
  91. package/src/components/spinner/Spinner.stories.tsx +335 -335
  92. package/src/components/spinner/index.tsx +44 -44
  93. package/src/components/spinner/types.ts +5 -5
  94. package/src/components/text/Text.stories.tsx +321 -321
  95. package/src/components/text/index.tsx +25 -25
  96. package/src/components/text/types.ts +45 -45
  97. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  98. package/src/components/tooltip/index.tsx +74 -74
  99. package/src/components/tooltip/types.ts +7 -7
  100. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  101. package/src/components/view-cart-button/index.tsx +42 -42
  102. package/src/components/view-cart-button/types.ts +5 -5
  103. package/src/contentful/blocks/accordion/Accordion.stories.mocks.tsx +128 -128
  104. package/src/contentful/blocks/accordion/Accordion.stories.tsx +98 -98
  105. package/src/contentful/blocks/accordion/index.tsx +112 -112
  106. package/src/contentful/blocks/accordion/types.ts +34 -34
  107. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  108. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  109. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +181 -181
  110. package/src/contentful/blocks/anchored-bottom-banner/types.ts +13 -13
  111. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.mocks.tsx +144 -144
  112. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.tsx +156 -156
  113. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  114. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  115. package/src/contentful/blocks/blogs-grid-base/index.tsx +119 -119
  116. package/src/contentful/blocks/blogs-grid-base/types.ts +36 -36
  117. package/src/contentful/blocks/breadcrumbs/BreadcrumbNavigation.stories.tsx +147 -147
  118. package/src/contentful/blocks/breadcrumbs/index.tsx +95 -95
  119. package/src/contentful/blocks/breadcrumbs/types.ts +8 -8
  120. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  121. package/src/contentful/blocks/button/index.tsx +131 -131
  122. package/src/contentful/blocks/button/types.ts +39 -39
  123. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  124. package/src/contentful/blocks/callout/index.tsx +277 -279
  125. package/src/contentful/blocks/callout/types.ts +78 -78
  126. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  127. package/src/contentful/blocks/cards/blog-card/index.tsx +129 -129
  128. package/src/contentful/blocks/cards/blog-card/types.ts +34 -34
  129. package/src/contentful/blocks/cards/floating-image-card/index.tsx +119 -119
  130. package/src/contentful/blocks/cards/floating-image-card/types.ts +30 -30
  131. package/src/contentful/blocks/cards/full-image-card/index.tsx +130 -130
  132. package/src/contentful/blocks/cards/full-image-card/types.ts +29 -29
  133. package/src/contentful/blocks/cards/index.tsx +13 -13
  134. package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
  135. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  136. package/src/contentful/blocks/cards/simple-card/index.tsx +325 -325
  137. package/src/contentful/blocks/cards/simple-card/types.ts +71 -71
  138. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  139. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  140. package/src/contentful/blocks/cards/types.ts +1 -1
  141. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  142. package/src/contentful/blocks/carousel/helper.tsx +494 -494
  143. package/src/contentful/blocks/carousel/index.tsx +87 -87
  144. package/src/contentful/blocks/carousel/types.ts +145 -145
  145. package/src/contentful/blocks/cart-retention-banner/index.tsx +109 -109
  146. package/src/contentful/blocks/cart-retention-banner/types.ts +11 -11
  147. package/src/contentful/blocks/comparison-table/index.tsx +29 -29
  148. package/src/contentful/blocks/comparison-table/types.ts +6 -6
  149. package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
  150. package/src/contentful/blocks/cookiebanner/type.ts +7 -7
  151. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  152. package/src/contentful/blocks/cta-callout/index.tsx +73 -71
  153. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  154. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  155. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  156. package/src/contentful/blocks/email-input-block/index.tsx +116 -116
  157. package/src/contentful/blocks/email-input-block/types.ts +16 -16
  158. package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
  159. package/src/contentful/blocks/find-kinetic/index.tsx +130 -130
  160. package/src/contentful/blocks/find-kinetic/types.ts +19 -19
  161. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  162. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  163. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  164. package/src/contentful/blocks/footer/Footer.stories.tsx +317 -317
  165. package/src/contentful/blocks/footer/index.tsx +91 -91
  166. package/src/contentful/blocks/footer/types.ts +13 -13
  167. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  168. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  169. package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
  170. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  171. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  172. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  173. package/src/contentful/blocks/modal/constants.ts +53 -53
  174. package/src/contentful/blocks/modal/index.tsx +107 -107
  175. package/src/contentful/blocks/modal/types.ts +12 -12
  176. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +139 -139
  177. package/src/contentful/blocks/navigation/index.tsx +568 -568
  178. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  179. package/src/contentful/blocks/navigation/types.ts +71 -71
  180. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  181. package/src/contentful/blocks/primary-hero/index.tsx +236 -236
  182. package/src/contentful/blocks/primary-hero/types.ts +37 -37
  183. package/src/contentful/blocks/search-block/index.tsx +90 -90
  184. package/src/contentful/blocks/search-block/types.ts +15 -15
  185. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  186. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  187. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  188. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  189. package/src/contentful/blocks/text/index.tsx +12 -12
  190. package/src/contentful/blocks/text/types.ts +1 -1
  191. package/src/contentful/index.ts +105 -105
  192. package/src/hooks/contentful/use-contentful-rich-text.tsx +309 -309
  193. package/src/hooks/contentful/use-processed-check-list.ts +63 -63
  194. package/src/hooks/use-body-scroll-lock.ts +34 -34
  195. package/src/hooks/use-carousel-swipe.ts +264 -264
  196. package/src/hooks/use-outside-click.ts +17 -17
  197. package/src/index.ts +107 -107
  198. package/src/next/index.ts +5 -5
  199. package/src/setupTests.ts +46 -46
  200. package/src/stories/DocsTemplate.tsx +24 -24
  201. package/src/styles/globals.css +343 -343
  202. package/src/types/global.d.ts +9 -9
  203. package/src/types/micro-components.ts +99 -99
  204. package/src/types/utm.ts +49 -49
  205. package/src/utils/contentful/to-document.ts +24 -24
  206. package/src/utils/cookie.ts +84 -84
  207. package/src/utils/cx.ts +49 -49
  208. package/src/utils/index.ts +41 -41
  209. package/src/utils/speed-card-bg.ts +24 -24
  210. 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 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
+ }