@windstream/react-shared-components 0.1.93 → 0.1.94

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (298) hide show
  1. package/README.md +635 -635
  2. package/dist/contentful/index.esm.js +3 -3
  3. package/dist/contentful/index.esm.js.map +1 -1
  4. package/dist/contentful/index.js +3 -3
  5. package/dist/contentful/index.js.map +1 -1
  6. package/dist/core.d.ts +6 -6
  7. package/dist/index.d.ts +2 -2
  8. package/dist/index.esm.js +13 -5
  9. package/dist/index.esm.js.map +1 -1
  10. package/dist/index.js +13 -5
  11. package/dist/index.js.map +1 -1
  12. package/dist/next/index.esm.js +2 -2
  13. package/dist/next/index.esm.js.map +1 -1
  14. package/dist/next/index.js +2 -2
  15. package/dist/next/index.js.map +1 -1
  16. package/dist/styles.css +1 -1
  17. package/dist/utils/index.esm.js +1 -1
  18. package/dist/utils/index.esm.js.map +1 -1
  19. package/dist/utils/index.js +1 -1
  20. package/dist/utils/index.js.map +1 -1
  21. package/package.json +191 -185
  22. package/src/components/accordion/Accordion.stories.tsx +230 -230
  23. package/src/components/accordion/index.test.tsx +270 -0
  24. package/src/components/accordion/index.tsx +70 -70
  25. package/src/components/accordion/types.ts +12 -12
  26. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  27. package/src/components/alert-card/index.test.tsx +152 -0
  28. package/src/components/alert-card/index.tsx +41 -41
  29. package/src/components/alert-card/types.ts +13 -13
  30. package/src/components/animation-wrapper/index.test.tsx +424 -0
  31. package/src/components/animation-wrapper/index.tsx +129 -129
  32. package/src/components/animation-wrapper/types.ts +11 -11
  33. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  34. package/src/components/brand-button/helpers.ts +35 -35
  35. package/src/components/brand-button/index.test.tsx +292 -0
  36. package/src/components/brand-button/index.tsx +120 -120
  37. package/src/components/brand-button/types.ts +38 -38
  38. package/src/components/button/Button.stories.tsx +108 -108
  39. package/src/components/button/index.test.tsx +91 -0
  40. package/src/components/button/index.tsx +27 -27
  41. package/src/components/button/types.ts +14 -14
  42. package/src/components/call-button/CallButton.stories.tsx +324 -324
  43. package/src/components/call-button/index.test.tsx +260 -0
  44. package/src/components/call-button/index.tsx +106 -106
  45. package/src/components/call-button/types.ts +16 -16
  46. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  47. package/src/components/checkbox/index.test.tsx +252 -0
  48. package/src/components/checkbox/index.tsx +197 -197
  49. package/src/components/checkbox/types.ts +27 -27
  50. package/src/components/checklist/Checklist.stories.tsx +150 -150
  51. package/src/components/checklist/index.test.tsx +231 -0
  52. package/src/components/checklist/index.tsx +96 -61
  53. package/src/components/checklist/types.ts +23 -17
  54. package/src/components/collapse/Collapse.stories.tsx +255 -255
  55. package/src/components/collapse/index.test.tsx +277 -0
  56. package/src/components/collapse/index.tsx +47 -46
  57. package/src/components/collapse/types.ts +6 -6
  58. package/src/components/divider/Divider.stories.tsx +205 -205
  59. package/src/components/divider/index.test.tsx +53 -0
  60. package/src/components/divider/index.tsx +22 -22
  61. package/src/components/divider/type.ts +3 -3
  62. package/src/components/image/Image.stories.tsx +113 -113
  63. package/src/components/image/index.test.tsx +174 -0
  64. package/src/components/image/index.tsx +25 -25
  65. package/src/components/image/types.ts +40 -40
  66. package/src/components/input/Input.stories.tsx +325 -325
  67. package/src/components/input/index.test.tsx +348 -0
  68. package/src/components/input/index.tsx +177 -177
  69. package/src/components/input/types.ts +37 -37
  70. package/src/components/link/Link.stories.tsx +163 -163
  71. package/src/components/link/index.test.tsx +199 -0
  72. package/src/components/link/index.tsx +116 -116
  73. package/src/components/link/types.ts +25 -25
  74. package/src/components/list/List.stories.tsx +272 -272
  75. package/src/components/list/index.test.tsx +166 -0
  76. package/src/components/list/index.tsx +88 -88
  77. package/src/components/list/list-item/index.tsx +38 -38
  78. package/src/components/list/list-item/types.ts +13 -13
  79. package/src/components/list/types.ts +29 -29
  80. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  81. package/src/components/material-icon/constants.ts +99 -99
  82. package/src/components/material-icon/index.test.tsx +130 -0
  83. package/src/components/material-icon/index.tsx +47 -47
  84. package/src/components/material-icon/types.ts +31 -31
  85. package/src/components/modal/Modal.stories.tsx +171 -171
  86. package/src/components/modal/index.test.tsx +310 -0
  87. package/src/components/modal/index.tsx +164 -164
  88. package/src/components/modal/types.ts +24 -24
  89. package/src/components/next-image/index.test.tsx +406 -0
  90. package/src/components/next-image/index.tsx +74 -74
  91. package/src/components/next-image/types.ts +1 -1
  92. package/src/components/pagination/index.test.tsx +521 -0
  93. package/src/components/pagination/index.tsx +91 -91
  94. package/src/components/pagination/types.ts +6 -6
  95. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  96. package/src/components/radio-button/index.test.tsx +151 -0
  97. package/src/components/radio-button/index.tsx +75 -75
  98. package/src/components/radio-button/types.ts +21 -21
  99. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  100. package/src/components/see-more/index.test.tsx +96 -0
  101. package/src/components/see-more/index.tsx +44 -44
  102. package/src/components/see-more/types.ts +4 -4
  103. package/src/components/select/Select.stories.tsx +411 -411
  104. package/src/components/select/index.test.tsx +256 -0
  105. package/src/components/select/index.tsx +155 -155
  106. package/src/components/select/types.ts +36 -36
  107. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  108. package/src/components/select-plan-button/index.test.tsx +173 -0
  109. package/src/components/select-plan-button/index.tsx +63 -63
  110. package/src/components/select-plan-button/types.ts +17 -17
  111. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  112. package/src/components/skeleton/index.test.tsx +74 -0
  113. package/src/components/skeleton/index.tsx +61 -61
  114. package/src/components/skeleton/types.ts +4 -4
  115. package/src/components/spinner/Spinner.stories.tsx +335 -335
  116. package/src/components/spinner/index.test.tsx +76 -0
  117. package/src/components/spinner/index.tsx +44 -44
  118. package/src/components/spinner/types.ts +5 -5
  119. package/src/components/text/Text.stories.tsx +321 -321
  120. package/src/components/text/index.test.tsx +65 -0
  121. package/src/components/text/index.tsx +25 -25
  122. package/src/components/text/types.ts +45 -45
  123. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  124. package/src/components/tooltip/index.test.tsx +50 -0
  125. package/src/components/tooltip/index.tsx +74 -74
  126. package/src/components/tooltip/types.ts +7 -7
  127. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  128. package/src/components/view-cart-button/index.test.tsx +57 -0
  129. package/src/components/view-cart-button/index.tsx +42 -42
  130. package/src/components/view-cart-button/types.ts +5 -5
  131. package/src/contentful/blocks/accordion/Accordion.stories.mocks.tsx +128 -128
  132. package/src/contentful/blocks/accordion/Accordion.stories.tsx +98 -98
  133. package/src/contentful/blocks/accordion/index.test.tsx +218 -0
  134. package/src/contentful/blocks/accordion/index.tsx +114 -112
  135. package/src/contentful/blocks/accordion/types.ts +34 -34
  136. package/src/contentful/blocks/address-input-banner/index.test.tsx +132 -0
  137. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  138. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  139. package/src/contentful/blocks/anchored-bottom-banner/index.test.tsx +287 -0
  140. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +181 -181
  141. package/src/contentful/blocks/anchored-bottom-banner/types.ts +13 -13
  142. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.mocks.tsx +144 -144
  143. package/src/contentful/blocks/blogs-grid/BlogGrid.stories.tsx +157 -156
  144. package/src/contentful/blocks/blogs-grid/index.test.tsx +355 -0
  145. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  146. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  147. package/src/contentful/blocks/blogs-grid-base/index.test.tsx +274 -0
  148. package/src/contentful/blocks/blogs-grid-base/index.tsx +119 -119
  149. package/src/contentful/blocks/blogs-grid-base/types.ts +36 -36
  150. package/src/contentful/blocks/breadcrumbs/BreadcrumbNavigation.stories.tsx +147 -147
  151. package/src/contentful/blocks/breadcrumbs/index.test.tsx +281 -0
  152. package/src/contentful/blocks/breadcrumbs/index.tsx +95 -95
  153. package/src/contentful/blocks/breadcrumbs/types.ts +8 -8
  154. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  155. package/src/contentful/blocks/button/index.test.tsx +339 -0
  156. package/src/contentful/blocks/button/index.tsx +131 -131
  157. package/src/contentful/blocks/button/types.ts +39 -39
  158. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  159. package/src/contentful/blocks/callout/index.test.tsx +539 -0
  160. package/src/contentful/blocks/callout/index.tsx +277 -277
  161. package/src/contentful/blocks/callout/types.ts +78 -78
  162. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  163. package/src/contentful/blocks/cards/blog-card/index.test.tsx +218 -0
  164. package/src/contentful/blocks/cards/blog-card/index.tsx +129 -129
  165. package/src/contentful/blocks/cards/blog-card/types.ts +34 -34
  166. package/src/contentful/blocks/cards/floating-image-card/index.test.tsx +201 -0
  167. package/src/contentful/blocks/cards/floating-image-card/index.tsx +119 -119
  168. package/src/contentful/blocks/cards/floating-image-card/types.ts +30 -30
  169. package/src/contentful/blocks/cards/full-image-card/index.test.tsx +216 -0
  170. package/src/contentful/blocks/cards/full-image-card/index.tsx +130 -130
  171. package/src/contentful/blocks/cards/full-image-card/types.ts +29 -29
  172. package/src/contentful/blocks/cards/index.test.tsx +39 -0
  173. package/src/contentful/blocks/cards/index.tsx +13 -13
  174. package/src/contentful/blocks/cards/product-card/index.test.tsx +263 -0
  175. package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
  176. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  177. package/src/contentful/blocks/cards/simple-card/index.test.tsx +364 -0
  178. package/src/contentful/blocks/cards/simple-card/index.tsx +325 -325
  179. package/src/contentful/blocks/cards/simple-card/types.ts +71 -71
  180. package/src/contentful/blocks/cards/testimonial-card/index.test.tsx +180 -0
  181. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  182. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  183. package/src/contentful/blocks/cards/types.ts +1 -1
  184. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  185. package/src/contentful/blocks/carousel/helper.test.tsx +539 -0
  186. package/src/contentful/blocks/carousel/helper.tsx +494 -494
  187. package/src/contentful/blocks/carousel/index.test.tsx +308 -0
  188. package/src/contentful/blocks/carousel/index.tsx +87 -87
  189. package/src/contentful/blocks/carousel/types.test.ts +16 -0
  190. package/src/contentful/blocks/carousel/types.ts +145 -145
  191. package/src/contentful/blocks/cart-retention-banner/index.test.tsx +409 -0
  192. package/src/contentful/blocks/cart-retention-banner/index.tsx +109 -109
  193. package/src/contentful/blocks/cart-retention-banner/types.ts +11 -11
  194. package/src/contentful/blocks/comparison-table/index.test.tsx +114 -0
  195. package/src/contentful/blocks/comparison-table/index.tsx +29 -29
  196. package/src/contentful/blocks/comparison-table/types.ts +6 -6
  197. package/src/contentful/blocks/cookiebanner/index.test.tsx +277 -0
  198. package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
  199. package/src/contentful/blocks/cookiebanner/type.ts +7 -7
  200. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  201. package/src/contentful/blocks/cta-callout/index.test.tsx +244 -0
  202. package/src/contentful/blocks/cta-callout/index.tsx +73 -73
  203. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  204. package/src/contentful/blocks/dynamic-tabs/index.test.tsx +240 -0
  205. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  206. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  207. package/src/contentful/blocks/email-input-block/index.test.tsx +213 -0
  208. package/src/contentful/blocks/email-input-block/index.tsx +121 -116
  209. package/src/contentful/blocks/email-input-block/types.ts +16 -16
  210. package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
  211. package/src/contentful/blocks/find-kinetic/index.test.tsx +269 -0
  212. package/src/contentful/blocks/find-kinetic/index.tsx +138 -138
  213. package/src/contentful/blocks/find-kinetic/types.ts +20 -20
  214. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  215. package/src/contentful/blocks/floating-banner/index.test.tsx +246 -0
  216. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  217. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  218. package/src/contentful/blocks/footer/Footer.stories.tsx +317 -317
  219. package/src/contentful/blocks/footer/index.test.tsx +302 -0
  220. package/src/contentful/blocks/footer/index.tsx +91 -91
  221. package/src/contentful/blocks/footer/types.ts +13 -13
  222. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  223. package/src/contentful/blocks/image-promo-bar/helper.test.tsx +61 -0
  224. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  225. package/src/contentful/blocks/image-promo-bar/index.test.tsx +467 -0
  226. package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
  227. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  228. package/src/contentful/blocks/image-promo-bar/vimeo-embed.test.tsx +142 -0
  229. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  230. package/src/contentful/blocks/image-promo-bar/youtube-embed.test.tsx +104 -0
  231. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  232. package/src/contentful/blocks/modal/constants.ts +53 -53
  233. package/src/contentful/blocks/modal/index.test.tsx +209 -0
  234. package/src/contentful/blocks/modal/index.tsx +108 -108
  235. package/src/contentful/blocks/modal/types.ts +12 -12
  236. package/src/contentful/blocks/navigation/Navigation.stories.mocks.tsx +78 -78
  237. package/src/contentful/blocks/navigation/Navigation.stories.tsx +138 -138
  238. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.test.tsx +208 -0
  239. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +141 -141
  240. package/src/contentful/blocks/navigation/index.test.tsx +924 -0
  241. package/src/contentful/blocks/navigation/index.tsx +569 -569
  242. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.test.tsx +131 -0
  243. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  244. package/src/contentful/blocks/navigation/types.ts +71 -71
  245. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  246. package/src/contentful/blocks/primary-hero/index.test.tsx +286 -0
  247. package/src/contentful/blocks/primary-hero/index.tsx +239 -236
  248. package/src/contentful/blocks/primary-hero/types.ts +37 -37
  249. package/src/contentful/blocks/search-block/index.test.tsx +268 -0
  250. package/src/contentful/blocks/search-block/index.tsx +90 -90
  251. package/src/contentful/blocks/search-block/types.ts +15 -15
  252. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  253. package/src/contentful/blocks/shape-background-wrapper/index.test.tsx +284 -0
  254. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  255. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  256. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  257. package/src/contentful/blocks/text/index.test.tsx +36 -0
  258. package/src/contentful/blocks/text/index.tsx +12 -12
  259. package/src/contentful/blocks/text/types.ts +1 -1
  260. package/src/contentful/index.test.ts +45 -0
  261. package/src/contentful/index.ts +105 -105
  262. package/src/global-mocks/contentful/to-document.ts +25 -0
  263. package/src/global-mocks/cookie.ts +48 -0
  264. package/src/global-mocks/cx.ts +37 -0
  265. package/src/global-mocks/index.ts +89 -0
  266. package/src/global-mocks/speed-card-bg.ts +27 -0
  267. package/src/global-mocks/utm.ts +49 -0
  268. package/src/hooks/contentful/use-contentful-rich-text.test.tsx +1758 -0
  269. package/src/hooks/contentful/use-contentful-rich-text.tsx +309 -309
  270. package/src/hooks/contentful/use-processed-check-list.test.tsx +277 -0
  271. package/src/hooks/contentful/use-processed-check-list.ts +63 -63
  272. package/src/hooks/use-body-scroll-lock.test.ts +134 -0
  273. package/src/hooks/use-body-scroll-lock.ts +34 -34
  274. package/src/hooks/use-carousel-swipe.test.ts +393 -0
  275. package/src/hooks/use-carousel-swipe.ts +264 -264
  276. package/src/hooks/use-outside-click.test.ts +142 -0
  277. package/src/hooks/use-outside-click.ts +17 -17
  278. package/src/index.ts +107 -107
  279. package/src/next/index.test.ts +7 -0
  280. package/src/next/index.ts +5 -5
  281. package/src/setupTests.ts +52 -46
  282. package/src/stories/DocsTemplate.tsx +24 -24
  283. package/src/styles/globals.css +343 -343
  284. package/src/types/global.d.ts +9 -9
  285. package/src/types/micro-components.ts +99 -99
  286. package/src/types/utm.ts +49 -49
  287. package/src/utils/contentful/to-document.test.ts +85 -0
  288. package/src/utils/contentful/to-document.ts +24 -24
  289. package/src/utils/cookie.test.ts +180 -0
  290. package/src/utils/cookie.ts +84 -84
  291. package/src/utils/cx.test.ts +90 -0
  292. package/src/utils/cx.ts +49 -49
  293. package/src/utils/index.test.ts +115 -0
  294. package/src/utils/index.ts +41 -41
  295. package/src/utils/speed-card-bg.test.ts +46 -0
  296. package/src/utils/speed-card-bg.ts +24 -24
  297. package/src/utils/utm.test.ts +359 -0
  298. package/src/utils/utm.ts +221 -221
@@ -0,0 +1,209 @@
1
+ import React from "react";
2
+ import { Modal } from "./index";
3
+
4
+ import { fireEvent, render, screen } from "@testing-library/react";
5
+
6
+ jest.mock("framer-motion", () => ({
7
+ motion: {
8
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
9
+ div: ({ children, className, ...rest }: any) => (
10
+ <div className={className}>{children}</div>
11
+ ),
12
+ },
13
+ }));
14
+
15
+ jest.mock("@radix-ui/react-dialog", () => ({
16
+ Root: ({ open, onOpenChange, children }: any) => (
17
+ <div
18
+ data-testid="dialog-root"
19
+ data-open={open}
20
+ onClick={() => open && onOpenChange?.()}
21
+ >
22
+ {children}
23
+ </div>
24
+ ),
25
+ Portal: ({ children }: any) => (
26
+ <div data-testid="dialog-portal">{children}</div>
27
+ ),
28
+ Overlay: () => <div data-testid="dialog-overlay" />,
29
+ Content: ({ children, className, style, onOpenAutoFocus }: any) => {
30
+ // Call onOpenAutoFocus to cover that branch
31
+ if (onOpenAutoFocus) {
32
+ const mockEvent = { preventDefault: jest.fn() };
33
+ onOpenAutoFocus(mockEvent);
34
+ }
35
+ return (
36
+ <div data-testid="dialog-content" className={className} style={style}>
37
+ {children}
38
+ </div>
39
+ );
40
+ },
41
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
42
+ Close: ({ children, asChild }: any) => (
43
+ <div data-testid="dialog-close">{children}</div>
44
+ ),
45
+ }));
46
+
47
+ jest.mock("@shared/components/material-icon", () => ({
48
+ MaterialIcon: ({ name }: any) => (
49
+ <span data-testid="material-icon">{name}</span>
50
+ ),
51
+ }));
52
+
53
+ jest.mock("@shared/components/text", () => ({
54
+ Text: ({ as: Tag = "span", children, className }: any) => (
55
+ <Tag className={className}>{children}</Tag>
56
+ ),
57
+ }));
58
+
59
+ jest.mock("@shared/contentful/blocks/button", () => ({
60
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
61
+ Button: ({ children, buttonClassName, showButtonAs }: any) => (
62
+ <button data-testid="close-button" className={buttonClassName}>
63
+ {children}
64
+ </button>
65
+ ),
66
+ }));
67
+
68
+ jest.mock("@shared/utils", () => ({
69
+ cx: (...args: any[]) => args.filter(Boolean).join(" "),
70
+ }));
71
+
72
+ describe("Modal", () => {
73
+ const defaultProps = { isOpen: true, onRequestClose: jest.fn() };
74
+
75
+ beforeEach(() => jest.clearAllMocks());
76
+
77
+ describe("Rendering", () => {
78
+ it("renders dialog root with open state", () => {
79
+ render(<Modal {...defaultProps} />);
80
+ expect(screen.getByTestId("dialog-root")).toHaveAttribute(
81
+ "data-open",
82
+ "true"
83
+ );
84
+ });
85
+
86
+ it("renders close button with MaterialIcon", () => {
87
+ render(<Modal {...defaultProps} />);
88
+ expect(screen.getByTestId("close-button")).toBeInTheDocument();
89
+ expect(screen.getByText("close")).toBeInTheDocument();
90
+ });
91
+
92
+ it("renders title as h2 when provided", () => {
93
+ render(<Modal {...defaultProps} title="My Title" />);
94
+ expect(screen.getByRole("heading", { level: 2 })).toHaveTextContent(
95
+ "My Title"
96
+ );
97
+ });
98
+
99
+ it("does not render title when not provided", () => {
100
+ render(<Modal {...defaultProps} />);
101
+ expect(screen.queryByRole("heading")).not.toBeInTheDocument();
102
+ });
103
+
104
+ it("renders description when provided", () => {
105
+ render(<Modal {...defaultProps} description="Some desc" />);
106
+ expect(screen.getByText("Some desc")).toBeInTheDocument();
107
+ });
108
+
109
+ it("does not render description when not provided", () => {
110
+ const { container } = render(<Modal {...defaultProps} />);
111
+ expect(container.querySelector(".mb-8")).not.toBeInTheDocument();
112
+ });
113
+
114
+ it("renders content", () => {
115
+ render(
116
+ <Modal
117
+ {...defaultProps}
118
+ content={<div data-testid="content">Content</div>}
119
+ />
120
+ );
121
+ expect(screen.getByTestId("content")).toBeInTheDocument();
122
+ });
123
+
124
+ it("renders children", () => {
125
+ render(
126
+ <Modal {...defaultProps}>
127
+ <span data-testid="child">Child</span>
128
+ </Modal>
129
+ );
130
+ expect(screen.getByTestId("child")).toBeInTheDocument();
131
+ });
132
+
133
+ it("does not render children placeholder when not provided", () => {
134
+ const { container } = render(<Modal {...defaultProps} />);
135
+ expect(container.innerHTML).not.toContain("null");
136
+ });
137
+ });
138
+
139
+ describe("Size", () => {
140
+ it("applies lg maxWidth by default", () => {
141
+ render(<Modal {...defaultProps} />);
142
+ expect(screen.getByTestId("dialog-content").style.maxWidth).toBe(
143
+ "1024px"
144
+ );
145
+ });
146
+
147
+ it("applies sm maxWidth", () => {
148
+ render(<Modal {...defaultProps} size="sm" />);
149
+ expect(screen.getByTestId("dialog-content").style.maxWidth).toBe("640px");
150
+ });
151
+
152
+ it("applies xl maxWidth", () => {
153
+ render(<Modal {...defaultProps} size="xl" />);
154
+ expect(screen.getByTestId("dialog-content").style.maxWidth).toBe(
155
+ "1280px"
156
+ );
157
+ });
158
+ });
159
+
160
+ describe("Props", () => {
161
+ it("applies bodyClassName to content", () => {
162
+ render(<Modal {...defaultProps} bodyClassName="custom-body" />);
163
+ expect(screen.getByTestId("dialog-content").className).toContain(
164
+ "custom-body"
165
+ );
166
+ });
167
+
168
+ it("applies type and index as data attributes", () => {
169
+ render(<Modal {...defaultProps} type="tabmodal" index={2} />);
170
+ const section = screen
171
+ .getByTestId("dialog-content")
172
+ .querySelector("[data-section-type]");
173
+ expect(section).toHaveAttribute("data-section-type", "tabmodal");
174
+ expect(section).toHaveAttribute("data-section-index", "2");
175
+ });
176
+
177
+ it("defaults type to modal and index to 0", () => {
178
+ render(<Modal {...defaultProps} />);
179
+ const section = screen
180
+ .getByTestId("dialog-content")
181
+ .querySelector("[data-section-type]");
182
+ expect(section).toHaveAttribute("data-section-type", "modal");
183
+ expect(section).toHaveAttribute("data-section-index", "0");
184
+ });
185
+ });
186
+
187
+ describe("Close behavior", () => {
188
+ it("calls onRequestClose when dialog onOpenChange triggers while open", () => {
189
+ render(<Modal {...defaultProps} />);
190
+ fireEvent.click(screen.getByTestId("dialog-root"));
191
+ expect(defaultProps.onRequestClose).toHaveBeenCalled();
192
+ });
193
+
194
+ it("does not call onRequestClose when isOpen is false", () => {
195
+ render(
196
+ <Modal isOpen={false} onRequestClose={defaultProps.onRequestClose} />
197
+ );
198
+ fireEvent.click(screen.getByTestId("dialog-root"));
199
+ expect(defaultProps.onRequestClose).not.toHaveBeenCalled();
200
+ });
201
+
202
+ it("handles missing onRequestClose gracefully", () => {
203
+ expect(() => {
204
+ render(<Modal isOpen={true} />);
205
+ fireEvent.click(screen.getByTestId("dialog-root"));
206
+ }).not.toThrow();
207
+ });
208
+ });
209
+ });
@@ -1,108 +1,108 @@
1
- "use client";
2
-
3
- import React from "react";
4
- import {
5
- backdropAnimationVariants,
6
- contentAnimationVariants,
7
- sizeToPixel,
8
- } from "./constants";
9
- import { ModalProps } from "./types";
10
- import { motion } from "framer-motion";
11
-
12
- import * as Dialog from "@radix-ui/react-dialog";
13
- import { MaterialIcon } from "@shared/components/material-icon";
14
- import { Text } from "@shared/components/text";
15
- import { Button } from "@shared/contentful/blocks/button";
16
- import { cx } from "@shared/utils";
17
-
18
- type ExtendedModalProps = ModalProps & {
19
- type?: string;
20
- index?: number;
21
- };
22
- export const Modal: React.FC<ExtendedModalProps> = props => {
23
- const {
24
- isOpen,
25
- onRequestClose,
26
- size,
27
- title,
28
- content,
29
- description,
30
- children,
31
- bodyClassName,
32
- type = "modal",
33
- index = 0,
34
- } = props;
35
-
36
- return (
37
- <Dialog.Root
38
- open={isOpen}
39
- onOpenChange={() => isOpen && onRequestClose?.()}
40
- >
41
- <Dialog.Portal>
42
- <motion.div
43
- initial="closed"
44
- animate={isOpen ? "open" : "closed"}
45
- variants={backdropAnimationVariants}
46
- transition={{ type: "tween", duration: 0.2, ease: "easeInOut" }}
47
- className="fixed inset-0 z-[1000] bg-scrim-bg-modal"
48
- >
49
- <Dialog.Overlay />
50
- </motion.div>
51
- <div className="pointer-events-none fixed inset-0 z-[1001] flex items-center justify-center">
52
- <motion.div
53
- initial="closed"
54
- animate={isOpen ? "open" : "closed"}
55
- variants={contentAnimationVariants}
56
- transition={{ type: "tween", duration: 0.3, ease: "easeInOut" }}
57
- className="pointer-events-auto w-[calc(100vw-10px)] focus:outline-none sm:w-[calc(100vw-90px)]"
58
- >
59
- <Dialog.Content
60
- onOpenAutoFocus={e => e.preventDefault()}
61
- className={cx(
62
- "relative mx-auto w-full rounded-modal bg-bg p-[25px] shadow-drop focus:outline-none",
63
- bodyClassName
64
- )}
65
- style={{ maxWidth: sizeToPixel[size || "lg"] }}
66
- >
67
- <div data-section-type={type} data-section-index={index}>
68
- <Dialog.Close asChild={true}>
69
- <Button
70
- showButtonAs="unstyled"
71
- buttonClassName="absolute right-0 top-0 mr-2 mt-2 bg-bg"
72
- >
73
- <MaterialIcon name="close" />
74
- </Button>
75
- </Dialog.Close>
76
- <div className="custom-modal-body max-h-[85vh] overflow-visible">
77
- {title ? (
78
- <Text
79
- className={cx(
80
- "mb-3 mt-5 w-full pr-8 text-center sm:pr-0",
81
- "heading5 md:pt-0",
82
- "mb-4 md:mb-[60px]"
83
- )}
84
- as="h2"
85
- >
86
- {title}
87
- </Text>
88
- ) : null}
89
-
90
- {description ? (
91
- <Text as="div" className="mb-8 sm:mb-5">
92
- {description}
93
- </Text>
94
- ) : null}
95
-
96
- <div className="flex flex-col gap-4 sm:gap-5">
97
- {content}
98
- {children ? children : null}
99
- </div>
100
- </div>
101
- </div>
102
- </Dialog.Content>
103
- </motion.div>
104
- </div>
105
- </Dialog.Portal>
106
- </Dialog.Root>
107
- );
108
- };
1
+ "use client";
2
+
3
+ import React from "react";
4
+ import {
5
+ backdropAnimationVariants,
6
+ contentAnimationVariants,
7
+ sizeToPixel,
8
+ } from "./constants";
9
+ import { ModalProps } from "./types";
10
+ import { motion } from "framer-motion";
11
+
12
+ import * as Dialog from "@radix-ui/react-dialog";
13
+ import { MaterialIcon } from "@shared/components/material-icon";
14
+ import { Text } from "@shared/components/text";
15
+ import { Button } from "@shared/contentful/blocks/button";
16
+ import { cx } from "@shared/utils";
17
+
18
+ type ExtendedModalProps = ModalProps & {
19
+ type?: string;
20
+ index?: number;
21
+ };
22
+ export const Modal: React.FC<ExtendedModalProps> = props => {
23
+ const {
24
+ isOpen,
25
+ onRequestClose,
26
+ size,
27
+ title,
28
+ content,
29
+ description,
30
+ children,
31
+ bodyClassName,
32
+ type = "modal",
33
+ index = 0,
34
+ } = props;
35
+
36
+ return (
37
+ <Dialog.Root
38
+ open={isOpen}
39
+ onOpenChange={() => isOpen && onRequestClose?.()}
40
+ >
41
+ <Dialog.Portal>
42
+ <motion.div
43
+ initial="closed"
44
+ animate={isOpen ? "open" : "closed"}
45
+ variants={backdropAnimationVariants}
46
+ transition={{ type: "tween", duration: 0.2, ease: "easeInOut" }}
47
+ className="fixed inset-0 z-[1000] bg-scrim-bg-modal"
48
+ >
49
+ <Dialog.Overlay />
50
+ </motion.div>
51
+ <div className="pointer-events-none fixed inset-0 z-[1001] flex items-center justify-center">
52
+ <motion.div
53
+ initial="closed"
54
+ animate={isOpen ? "open" : "closed"}
55
+ variants={contentAnimationVariants}
56
+ transition={{ type: "tween", duration: 0.3, ease: "easeInOut" }}
57
+ className="pointer-events-auto w-[calc(100vw-10px)] focus:outline-none sm:w-[calc(100vw-90px)]"
58
+ >
59
+ <Dialog.Content
60
+ onOpenAutoFocus={e => e.preventDefault()}
61
+ className={cx(
62
+ "relative mx-auto w-full rounded-modal bg-bg p-[25px] shadow-drop focus:outline-none",
63
+ bodyClassName
64
+ )}
65
+ style={{ maxWidth: sizeToPixel[size || "lg"] }}
66
+ >
67
+ <div data-section-type={type} data-section-index={index}>
68
+ <Dialog.Close asChild={true}>
69
+ <Button
70
+ showButtonAs="unstyled"
71
+ buttonClassName="absolute right-0 top-0 mr-2 mt-2 bg-bg"
72
+ >
73
+ <MaterialIcon name="close" />
74
+ </Button>
75
+ </Dialog.Close>
76
+ <div className="custom-modal-body max-h-[85vh] overflow-visible">
77
+ {title ? (
78
+ <Text
79
+ className={cx(
80
+ "mb-3 mt-5 w-full pr-8 text-center sm:pr-0",
81
+ "heading5 md:pt-0",
82
+ "mb-4 md:mb-[60px]"
83
+ )}
84
+ as="h2"
85
+ >
86
+ {title}
87
+ </Text>
88
+ ) : null}
89
+
90
+ {description ? (
91
+ <Text as="div" className="mb-8 sm:mb-5">
92
+ {description}
93
+ </Text>
94
+ ) : null}
95
+
96
+ <div className="flex flex-col gap-4 sm:gap-5">
97
+ {content}
98
+ {children ? children : null}
99
+ </div>
100
+ </div>
101
+ </div>
102
+ </Dialog.Content>
103
+ </motion.div>
104
+ </div>
105
+ </Dialog.Portal>
106
+ </Dialog.Root>
107
+ );
108
+ };
@@ -1,12 +1,12 @@
1
- import React from "react";
2
-
3
- export type ModalProps = {
4
- isOpen: boolean;
5
- onRequestClose?: () => void;
6
- size?: "lg" | "md" | "sm" | "xl" | "xs";
7
- title?: string;
8
- description?: React.ReactNode;
9
- content?: React.ReactNode;
10
- children?: React.ReactNode;
11
- bodyClassName?: string;
12
- };
1
+ import React from "react";
2
+
3
+ export type ModalProps = {
4
+ isOpen: boolean;
5
+ onRequestClose?: () => void;
6
+ size?: "lg" | "md" | "sm" | "xl" | "xs";
7
+ title?: string;
8
+ description?: React.ReactNode;
9
+ content?: React.ReactNode;
10
+ children?: React.ReactNode;
11
+ bodyClassName?: string;
12
+ };
@@ -1,78 +1,78 @@
1
- export const PRIMARY_LINKS: any = [
2
- {
3
- __typename: "ComponentButton",
4
- buttonLabel: "Home",
5
- href: "/",
6
- preserveQueryParameters: false,
7
- },
8
- {
9
- __typename: "ComponentButton",
10
- buttonLabel: "Products",
11
- href: "/products",
12
- preserveQueryParameters: false,
13
- },
14
- {
15
- __typename: "ComponentButtonGroup",
16
- title: "Services",
17
- items: {
18
- items: [
19
- {
20
- __typename: "ComponentButton",
21
- buttonLabel: "Internet",
22
- href: "/internet",
23
- },
24
- {
25
- __typename: "ComponentButton",
26
- buttonLabel: "Phone",
27
- href: "/phone",
28
- },
29
- ],
30
- },
31
- },
32
- ];
33
-
34
- export const UTILITY_LINKS: any = [
35
- {
36
- __typename: "ComponentButton",
37
- buttonLabel: "Residential",
38
- href: "/residential",
39
- },
40
- {
41
- __typename: "ComponentButton",
42
- buttonLabel: "Business",
43
- href: "/business",
44
- },
45
-
46
- ];
47
-
48
- export const ACCOUNT_LINKS: any = [
49
- {
50
- __typename: "ComponentButtonGroup",
51
- title: "My Account",
52
- items: {
53
- items: [
54
- {
55
- __typename: "ComponentButton",
56
- buttonLabel: "Profile",
57
- href: "/profile",
58
- },
59
- ],
60
- },
61
- },
62
- ];
63
-
64
- export const SUPPORT_LINKS: any = [
65
- {
66
- __typename: "ComponentButtonGroup",
67
- title: "Support",
68
- items: {
69
- items: [
70
- {
71
- __typename: "ComponentButton",
72
- buttonLabel: "Help",
73
- href: "/help",
74
- },
75
- ],
76
- },
77
- },
78
- ];
1
+ export const PRIMARY_LINKS: any = [
2
+ {
3
+ __typename: "ComponentButton",
4
+ buttonLabel: "Home",
5
+ href: "/",
6
+ preserveQueryParameters: false,
7
+ },
8
+ {
9
+ __typename: "ComponentButton",
10
+ buttonLabel: "Products",
11
+ href: "/products",
12
+ preserveQueryParameters: false,
13
+ },
14
+ {
15
+ __typename: "ComponentButtonGroup",
16
+ title: "Services",
17
+ items: {
18
+ items: [
19
+ {
20
+ __typename: "ComponentButton",
21
+ buttonLabel: "Internet",
22
+ href: "/internet",
23
+ },
24
+ {
25
+ __typename: "ComponentButton",
26
+ buttonLabel: "Phone",
27
+ href: "/phone",
28
+ },
29
+ ],
30
+ },
31
+ },
32
+ ];
33
+
34
+ export const UTILITY_LINKS: any = [
35
+ {
36
+ __typename: "ComponentButton",
37
+ buttonLabel: "Residential",
38
+ href: "/residential",
39
+ },
40
+ {
41
+ __typename: "ComponentButton",
42
+ buttonLabel: "Business",
43
+ href: "/business",
44
+ },
45
+
46
+ ];
47
+
48
+ export const ACCOUNT_LINKS: any = [
49
+ {
50
+ __typename: "ComponentButtonGroup",
51
+ title: "My Account",
52
+ items: {
53
+ items: [
54
+ {
55
+ __typename: "ComponentButton",
56
+ buttonLabel: "Profile",
57
+ href: "/profile",
58
+ },
59
+ ],
60
+ },
61
+ },
62
+ ];
63
+
64
+ export const SUPPORT_LINKS: any = [
65
+ {
66
+ __typename: "ComponentButtonGroup",
67
+ title: "Support",
68
+ items: {
69
+ items: [
70
+ {
71
+ __typename: "ComponentButton",
72
+ buttonLabel: "Help",
73
+ href: "/help",
74
+ },
75
+ ],
76
+ },
77
+ },
78
+ ];