@windstream/react-shared-components 0.1.71 → 0.1.72

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 (208) hide show
  1. package/README.md +635 -635
  2. package/dist/contentful/index.d.ts +106 -30
  3. package/dist/contentful/index.esm.js +4 -2
  4. package/dist/contentful/index.esm.js.map +1 -1
  5. package/dist/contentful/index.js +4 -2
  6. package/dist/contentful/index.js.map +1 -1
  7. package/dist/core.d.ts +5 -5
  8. package/dist/index.d.ts +4 -4
  9. package/dist/index.esm.js +7 -5
  10. package/dist/index.esm.js.map +1 -1
  11. package/dist/index.js +7 -5
  12. package/dist/index.js.map +1 -1
  13. package/dist/next/index.esm.js +3 -1
  14. package/dist/next/index.esm.js.map +1 -1
  15. package/dist/next/index.js +3 -1
  16. package/dist/next/index.js.map +1 -1
  17. package/dist/styles.css +1 -1
  18. package/dist/utils/index.d.ts +15 -3
  19. package/dist/utils/index.esm.js +1 -1
  20. package/dist/utils/index.esm.js.map +1 -1
  21. package/dist/utils/index.js +1 -1
  22. package/dist/utils/index.js.map +1 -1
  23. package/package.json +185 -185
  24. package/src/components/accordion/Accordion.stories.tsx +230 -230
  25. package/src/components/accordion/index.tsx +70 -70
  26. package/src/components/accordion/types.ts +12 -12
  27. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  28. package/src/components/alert-card/index.tsx +41 -41
  29. package/src/components/alert-card/types.ts +13 -13
  30. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  31. package/src/components/brand-button/helpers.ts +35 -35
  32. package/src/components/brand-button/index.tsx +120 -120
  33. package/src/components/brand-button/types.ts +38 -38
  34. package/src/components/button/Button.stories.tsx +108 -108
  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.tsx +106 -106
  39. package/src/components/call-button/types.ts +16 -16
  40. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  41. package/src/components/checkbox/index.tsx +197 -197
  42. package/src/components/checkbox/types.ts +27 -27
  43. package/src/components/checklist/Checklist.stories.tsx +150 -150
  44. package/src/components/checklist/index.tsx +61 -61
  45. package/src/components/checklist/types.ts +17 -17
  46. package/src/components/collapse/Collapse.stories.tsx +255 -255
  47. package/src/components/collapse/index.tsx +46 -46
  48. package/src/components/collapse/types.ts +6 -6
  49. package/src/components/divider/Divider.stories.tsx +205 -205
  50. package/src/components/divider/index.tsx +22 -22
  51. package/src/components/divider/type.ts +3 -3
  52. package/src/components/image/Image.stories.tsx +113 -113
  53. package/src/components/image/index.tsx +25 -25
  54. package/src/components/image/types.ts +40 -40
  55. package/src/components/input/Input.stories.tsx +325 -325
  56. package/src/components/input/index.tsx +177 -177
  57. package/src/components/input/types.ts +37 -37
  58. package/src/components/link/Link.stories.tsx +163 -163
  59. package/src/components/link/index.tsx +116 -116
  60. package/src/components/link/types.ts +25 -25
  61. package/src/components/list/List.stories.tsx +272 -272
  62. package/src/components/list/index.tsx +88 -88
  63. package/src/components/list/list-item/index.tsx +38 -38
  64. package/src/components/list/list-item/types.ts +13 -13
  65. package/src/components/list/types.ts +29 -29
  66. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  67. package/src/components/material-icon/constants.ts +99 -99
  68. package/src/components/material-icon/index.tsx +47 -47
  69. package/src/components/material-icon/types.ts +31 -31
  70. package/src/components/modal/Modal.stories.tsx +171 -171
  71. package/src/components/modal/index.tsx +164 -164
  72. package/src/components/modal/types.ts +24 -24
  73. package/src/components/next-image/index.tsx +74 -72
  74. package/src/components/next-image/types.ts +1 -1
  75. package/src/components/pagination/index.tsx +91 -91
  76. package/src/components/pagination/types.ts +6 -6
  77. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  78. package/src/components/radio-button/index.tsx +75 -75
  79. package/src/components/radio-button/types.ts +21 -21
  80. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  81. package/src/components/see-more/index.tsx +44 -44
  82. package/src/components/see-more/types.ts +4 -4
  83. package/src/components/select/Select.stories.tsx +411 -411
  84. package/src/components/select/index.tsx +155 -155
  85. package/src/components/select/types.ts +36 -36
  86. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  87. package/src/components/select-plan-button/index.tsx +63 -63
  88. package/src/components/select-plan-button/types.ts +17 -17
  89. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  90. package/src/components/skeleton/index.tsx +61 -61
  91. package/src/components/skeleton/types.ts +4 -4
  92. package/src/components/spinner/Spinner.stories.tsx +335 -335
  93. package/src/components/spinner/index.tsx +44 -44
  94. package/src/components/spinner/types.ts +5 -5
  95. package/src/components/text/Text.stories.tsx +321 -321
  96. package/src/components/text/index.tsx +25 -25
  97. package/src/components/text/types.ts +45 -45
  98. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  99. package/src/components/tooltip/index.tsx +74 -74
  100. package/src/components/tooltip/types.ts +7 -7
  101. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  102. package/src/components/view-cart-button/index.tsx +42 -42
  103. package/src/components/view-cart-button/types.ts +5 -5
  104. package/src/contentful/blocks/accordion/Accordion.stories.mocks.tsx +128 -128
  105. package/src/contentful/blocks/accordion/Accordion.stories.tsx +98 -106
  106. package/src/contentful/blocks/accordion/index.tsx +112 -112
  107. package/src/contentful/blocks/accordion/types.ts +34 -34
  108. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  109. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  110. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +181 -70
  111. package/src/contentful/blocks/anchored-bottom-banner/types.ts +13 -10
  112. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  113. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  114. package/src/contentful/blocks/blogs-grid-base/index.tsx +119 -119
  115. package/src/contentful/blocks/blogs-grid-base/types.ts +36 -36
  116. package/src/contentful/blocks/breadcrumbs/index.tsx +81 -81
  117. package/src/contentful/blocks/breadcrumbs/types.ts +6 -6
  118. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  119. package/src/contentful/blocks/button/index.tsx +131 -131
  120. package/src/contentful/blocks/button/types.ts +39 -39
  121. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  122. package/src/contentful/blocks/callout/index.tsx +252 -88
  123. package/src/contentful/blocks/callout/types.ts +68 -15
  124. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  125. package/src/contentful/blocks/cards/blog-card/index.tsx +129 -129
  126. package/src/contentful/blocks/cards/blog-card/types.ts +34 -34
  127. package/src/contentful/blocks/cards/floating-image-card/index.tsx +119 -0
  128. package/src/contentful/blocks/cards/floating-image-card/types.ts +30 -0
  129. package/src/contentful/blocks/cards/full-image-card/index.tsx +130 -0
  130. package/src/contentful/blocks/cards/full-image-card/types.ts +29 -0
  131. package/src/contentful/blocks/cards/index.tsx +13 -13
  132. package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
  133. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  134. package/src/contentful/blocks/cards/simple-card/index.tsx +325 -89
  135. package/src/contentful/blocks/cards/simple-card/types.ts +71 -28
  136. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  137. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  138. package/src/contentful/blocks/cards/types.ts +1 -1
  139. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  140. package/src/contentful/blocks/carousel/helper.tsx +440 -440
  141. package/src/contentful/blocks/carousel/index.tsx +85 -85
  142. package/src/contentful/blocks/carousel/types.ts +144 -144
  143. package/src/contentful/blocks/cart-retention-banner/index.tsx +105 -105
  144. package/src/contentful/blocks/cart-retention-banner/types.ts +11 -11
  145. package/src/contentful/blocks/comparison-table/index.tsx +29 -27
  146. package/src/contentful/blocks/comparison-table/types.ts +6 -6
  147. package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
  148. package/src/contentful/blocks/cookiebanner/type.ts +7 -7
  149. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  150. package/src/contentful/blocks/cta-callout/index.tsx +71 -71
  151. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  152. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  153. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  154. package/src/contentful/blocks/email-input-block/index.tsx +116 -116
  155. package/src/contentful/blocks/email-input-block/types.ts +16 -16
  156. package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
  157. package/src/contentful/blocks/find-kinetic/index.tsx +130 -130
  158. package/src/contentful/blocks/find-kinetic/types.ts +19 -19
  159. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  160. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  161. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  162. package/src/contentful/blocks/footer/Footer.stories.tsx +317 -204
  163. package/src/contentful/blocks/footer/index.tsx +91 -91
  164. package/src/contentful/blocks/footer/types.ts +13 -13
  165. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  166. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  167. package/src/contentful/blocks/image-promo-bar/index.tsx +246 -246
  168. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  169. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  170. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  171. package/src/contentful/blocks/modal/constants.ts +53 -53
  172. package/src/contentful/blocks/modal/index.tsx +107 -107
  173. package/src/contentful/blocks/modal/types.ts +12 -12
  174. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +124 -124
  175. package/src/contentful/blocks/navigation/index.tsx +462 -462
  176. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  177. package/src/contentful/blocks/navigation/types.ts +64 -64
  178. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  179. package/src/contentful/blocks/primary-hero/index.tsx +236 -236
  180. package/src/contentful/blocks/primary-hero/types.ts +37 -37
  181. package/src/contentful/blocks/search-block/index.tsx +90 -90
  182. package/src/contentful/blocks/search-block/types.ts +15 -15
  183. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  184. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  185. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  186. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  187. package/src/contentful/blocks/text/index.tsx +12 -12
  188. package/src/contentful/blocks/text/types.ts +1 -1
  189. package/src/contentful/index.ts +105 -105
  190. package/src/hooks/contentful/use-contentful-rich-text.tsx +309 -307
  191. package/src/hooks/contentful/use-processed-check-list.ts +63 -63
  192. package/src/hooks/use-body-scroll-lock.ts +34 -34
  193. package/src/hooks/use-carousel-swipe.ts +264 -264
  194. package/src/hooks/use-outside-click.ts +17 -17
  195. package/src/index.ts +101 -101
  196. package/src/next/index.ts +5 -5
  197. package/src/setupTests.ts +46 -46
  198. package/src/stories/DocsTemplate.tsx +24 -24
  199. package/src/styles/globals.css +343 -343
  200. package/src/types/global.d.ts +9 -9
  201. package/src/types/micro-components.ts +99 -99
  202. package/src/types/utm.ts +49 -49
  203. package/src/utils/contentful/to-document.ts +24 -24
  204. package/src/utils/cookie.ts +84 -84
  205. package/src/utils/cx.ts +49 -49
  206. package/src/utils/index.ts +41 -38
  207. package/src/utils/speed-card-bg.ts +24 -0
  208. package/src/utils/utm.ts +221 -221
@@ -1,462 +1,462 @@
1
- import React, { FormEvent } from "react";
2
- import { DesktopLinkGroups } from "./desktop-link-groups.tsx";
3
- import { MobileLinkGroups } from "./mobile-link-groups.tsx";
4
- import { NavigationProps } from "./types";
5
-
6
- import { CallButton } from "@shared/components/call-button";
7
- import { Input } from "@shared/components/input";
8
- import { Link } from "@shared/components/link";
9
- import { MaterialIcon } from "@shared/components/material-icon";
10
- import { NextImage } from "@shared/components/next-image";
11
- import { Text } from "@shared/components/text";
12
- import { Button as ContentfulButton } from "@shared/contentful/blocks/button";
13
- import { cx } from "@shared/utils";
14
-
15
- export const Navigation: React.FC<NavigationProps> = props => {
16
- const {
17
- primaryNavigationLinks,
18
- utilityNavigationLinks,
19
- checkPlansJSX,
20
- primaryNavigationLogo,
21
- accountNavigationLinks,
22
- supportNavigationLinks,
23
- searchBarIcon,
24
- invocaPhoneNumberLink,
25
- invocaPhoneNumberDisplayText,
26
- callNowCtaText,
27
- displayCartIcon,
28
- cartHref,
29
- cartHasRetention,
30
- cartIconAriaLabel = "Cart",
31
- onCallClickDesktop,
32
- onCallClickMobile,
33
- onCartClick,
34
- onSearch = () => {},
35
- utilityNavActiveIndex,
36
- primaryNavigationLogoWidth = 76.5,
37
- primaryNavigationLogoHeight = 24,
38
- hideMobileCallButton = false,
39
- } = props;
40
- return (
41
- <div className="component-container">
42
- <nav className={`menu-container z-[1000]`}>
43
- <div className="utility-container hidden lg:block lg:border-b lg:px-2">
44
- <div className="mx-auto flex max-w-120 justify-between">
45
- <ul className="flex gap-5" aria-label="Utility Navigation">
46
- {utilityNavigationLinks?.map((links, index) => {
47
- return (
48
- <li key={`main-menu-items-${index}`}>
49
- <ContentfulButton
50
- linkClassName={cx(
51
- "footnote flex items-center w-full h-11 text-text",
52
- // If utilityNavActiveIndex is not provided, default to making the second link active (for existing business app). If utilityNavActiveIndex is provided, use it to determine which link is active.
53
- typeof utilityNavActiveIndex !== "number"
54
- ? index === 1 && "label4"
55
- : utilityNavActiveIndex === index && "label4"
56
- )}
57
- linkVariant="unstyled"
58
- {...(Object.fromEntries(
59
- Object.entries(links).filter(([_, v]) => v !== null)
60
- ) as any)}
61
- />
62
- </li>
63
- );
64
- })}
65
- </ul>
66
- <div className="flex items-center gap-10">
67
- <CallButton
68
- className="border-none"
69
- href={invocaPhoneNumberLink}
70
- prefix={callNowCtaText}
71
- onClick={onCallClickDesktop}
72
- >
73
- <Text className="body3 text-text">
74
- {invocaPhoneNumberDisplayText}
75
- </Text>
76
- </CallButton>
77
- {displayCartIcon ? (
78
- <Link
79
- href={cartHref || "#"}
80
- className="relative inline-flex cursor-pointer"
81
- aria-label={cartIconAriaLabel}
82
- onClick={onCartClick}
83
- >
84
- <MaterialIcon name="shopping_cart" />
85
- {cartHasRetention ? (
86
- <span className="absolute -right-2 -top-1 h-2.5 w-2.5 rounded-full bg-icon-brand" />
87
- ) : null}
88
- </Link>
89
- ) : null}
90
- {accountNavigationLinks?.map((links, index) => {
91
- return (
92
- <DesktopLinkGroups
93
- key={`my-account-${index}`}
94
- anchorName={`my-account-${index}`}
95
- link={links}
96
- />
97
- );
98
- })}
99
- </div>
100
- </div>
101
- </div>
102
- <div className="main-nav-container" aria-label="Main Navigation">
103
- <div className="mobile-nav-section flex h-14 items-center justify-between border border-b px-5 py-[10px] lg:hidden">
104
- <div>
105
- <Link href="/" className="flex">
106
- <NextImage
107
- src={
108
- typeof primaryNavigationLogo === "string"
109
- ? primaryNavigationLogo
110
- : primaryNavigationLogo?.url || ""
111
- }
112
- alt="Kinetic business logo"
113
- width={primaryNavigationLogoWidth}
114
- height={primaryNavigationLogoHeight}
115
- />
116
- </Link>
117
- </div>
118
- <div className="flex items-center gap-6">
119
- {hideMobileCallButton ? null : (
120
- <CallButton
121
- href={invocaPhoneNumberLink}
122
- onClick={onCallClickMobile}
123
- >
124
- <Text as="span" className="footnote">
125
- {invocaPhoneNumberDisplayText}
126
- </Text>
127
- </CallButton>
128
- )}
129
- {displayCartIcon ? (
130
- <Link
131
- href={cartHref || "#"}
132
- className="relative inline-flex cursor-pointer"
133
- aria-label={cartIconAriaLabel}
134
- onClick={onCartClick}
135
- >
136
- <MaterialIcon name="shopping_cart" />
137
- {cartHasRetention ? (
138
- <span className="absolute -right-2 -top-1 h-2.5 w-2.5 rounded-full bg-icon-brand" />
139
- ) : null}
140
- </Link>
141
- ) : null}
142
- <MobileMenu {...props} />
143
- </div>
144
- </div>
145
-
146
- <div className="desktop-nav-section hidden lg:block lg:border-b lg:px-2">
147
- <div className="mx-auto flex h-14 max-w-120 items-center justify-between">
148
- <div className="flex h-full">
149
- <Link href="/" className="flex">
150
- <NextImage
151
- src={
152
- typeof primaryNavigationLogo === "string"
153
- ? primaryNavigationLogo
154
- : primaryNavigationLogo?.url || ""
155
- }
156
- alt="Kinetic business logo"
157
- width={primaryNavigationLogoWidth}
158
- height={primaryNavigationLogoHeight}
159
- className="mr-[64px]"
160
- />
161
- </Link>
162
-
163
- <div className="flex h-full items-center gap-5">
164
- {primaryNavigationLinks?.map((links, index) => {
165
- return (
166
- <DesktopLinkGroups
167
- key={`main-menu-${index}`}
168
- anchorName={`main-menu-${index}`}
169
- link={links}
170
- />
171
- );
172
- })}
173
- </div>
174
- </div>
175
- <div className="flex h-full items-center gap-10">
176
- <DesktopSearchInput
177
- searchBarIconURL={
178
- typeof searchBarIcon === "string"
179
- ? searchBarIcon
180
- : searchBarIcon?.url || ""
181
- }
182
- onSearch={onSearch}
183
- />
184
- {supportNavigationLinks?.map((links, index) => {
185
- return (
186
- <DesktopLinkGroups
187
- key={`support-menu-${index}`}
188
- anchorName={`support-menu-${index}`}
189
- link={links}
190
- />
191
- );
192
- })}
193
- </div>
194
- </div>
195
- </div>
196
- </div>
197
- </nav>
198
-
199
- {checkPlansJSX && <div className="md:hidden">{checkPlansJSX}</div>}
200
- </div>
201
- );
202
- };
203
-
204
- const MobileMenu = (props: NavigationProps) => {
205
- const {
206
- primaryNavigationLinks,
207
- utilityNavigationLinks,
208
- supportNavigationLinks,
209
- accountNavigationLinks,
210
- } = props;
211
- const [isOpen, setIsOpen] = React.useState(false);
212
-
213
- React.useEffect(() => {
214
- if (typeof window === "undefined") return;
215
-
216
- if (isOpen) document.body.style.overflowY = "hidden";
217
- else document.body.style.overflowY = "unset";
218
-
219
- const element = document.getElementById("drawer-items");
220
- if (!element) return;
221
-
222
- const focusableEls = element.querySelectorAll(".focus-item");
223
- const firstFocusableEl = focusableEls[0];
224
- const lastFocusableEl = focusableEls[focusableEls.length - 1];
225
-
226
- const handleKeyDown = (e: {
227
- key: string;
228
- keyCode: number;
229
- shiftKey: any;
230
- preventDefault: () => void;
231
- }) => {
232
- const isTabPressed = e.key === "Tab" || e.keyCode === 9;
233
-
234
- if (!isTabPressed) return;
235
-
236
- if (e.shiftKey) {
237
- if (document.activeElement === firstFocusableEl) {
238
- (lastFocusableEl as HTMLButtonElement).focus?.();
239
- e.preventDefault();
240
- }
241
- } else {
242
- if (document.activeElement === lastFocusableEl) {
243
- (firstFocusableEl as HTMLButtonElement).focus?.();
244
- e.preventDefault();
245
- }
246
- }
247
- };
248
-
249
- window.addEventListener("keydown", handleKeyDown);
250
-
251
- return () => {
252
- document.body.style.overflowY = "unset";
253
- window.removeEventListener("keydown", handleKeyDown);
254
- };
255
- }, [isOpen]);
256
-
257
- const closeMenu = () => {
258
- setIsOpen(false);
259
- };
260
-
261
- return (
262
- <div>
263
- <ContentfulButton
264
- showButtonAs="unstyled"
265
- buttonClassName="flex"
266
- onClick={() => setIsOpen(true)}
267
- >
268
- <MaterialIcon name="menu" />
269
- </ContentfulButton>
270
- {isOpen ? (
271
- <div className="fixed bottom-0 left-0 right-0 top-0 z-[90] h-full w-full bg-scrim-bg-modal"></div>
272
- ) : null}
273
-
274
- <div
275
- className={cx(
276
- "fixed bottom-0 right-0 top-0",
277
- "z-[100] h-full bg-bg px-0 py-4",
278
- "transition-all duration-300 ease-in-out",
279
- "block",
280
- isOpen ? "right-0" : "-right-96"
281
- )}
282
- id="mobile-menu-overlay"
283
- >
284
- <div id="drawer-items" className="flex h-full flex-col gap-3">
285
- <div className="flex items-center justify-between px-4">
286
- <div>
287
- <CallButton
288
- className="border-none"
289
- href={props.invocaPhoneNumberLink}
290
- onClick={props.onCallClickMobile}
291
- >
292
- {props.invocaPhoneNumberDisplayText}
293
- </CallButton>
294
- </div>
295
- <div>
296
- <ContentfulButton
297
- showButtonAs="unstyled"
298
- buttonClassName="focus-item flex"
299
- onClick={closeMenu}
300
- >
301
- <MaterialIcon name="close" />
302
- </ContentfulButton>
303
- </div>
304
- </div>
305
- <MobileSearchInput
306
- closeMenu={closeMenu}
307
- isMenuOpen={isOpen}
308
- searchBarIconURL={
309
- typeof props.searchBarIcon === "string"
310
- ? props.searchBarIcon
311
- : props.searchBarIcon?.url || ""
312
- }
313
- onSearch={props.onSearch || (() => {})}
314
- />
315
- <div
316
- className="flex-grow overflow-y-auto"
317
- onClick={event => {
318
- // Close the drawer when a link (anchor) is clicked.
319
- // Group-toggle buttons render as <button> and won't match,
320
- // so they continue to expand/collapse normally.
321
- // Required for Pages Router where MobileMenu persists across
322
- // client-side navigations and its local `isOpen` state would
323
- // otherwise stay true on the new route.
324
- const target = event.target as HTMLElement | null;
325
- if (target?.closest("a")) {
326
- closeMenu();
327
- }
328
- }}
329
- >
330
- <ul className="mt-2 flex flex-col gap-2">
331
- {[
332
- ...(primaryNavigationLinks || []),
333
- ...(supportNavigationLinks || []),
334
- ...(accountNavigationLinks || []),
335
- ].map((links, index) => (
336
- <li key={`main-menu-items-${index}`}>
337
- <MobileLinkGroups link={links} />
338
- </li>
339
- ))}
340
- </ul>
341
-
342
- <ul className="mt-2 flex gap-5 bg-bg-fill-info px-4">
343
- {utilityNavigationLinks?.map((link, index) => {
344
- return (
345
- <li key={`utility-menu-items-${index}`}>
346
- <ContentfulButton
347
- key={`utility-submenu-link-btn-${link.anchorId}`}
348
- {...(Object.fromEntries(
349
- Object.entries(link).filter(([_, v]) => v !== null)
350
- ) as any)}
351
- linkClassName={cx(
352
- "footnote flex items-center w-full h-11 text-text-link",
353
- typeof props.utilityNavActiveIndex !== "number"
354
- ? index === 1 && "label4"
355
- : props.utilityNavActiveIndex === index && "label4"
356
- )}
357
- linkVariant="unstyled"
358
- />
359
- </li>
360
- );
361
- })}
362
- </ul>
363
- </div>
364
- </div>
365
- </div>
366
- </div>
367
- );
368
- };
369
-
370
- const MobileSearchInput = (props: {
371
- closeMenu: () => void;
372
- onSearch: (query: string) => void;
373
- isMenuOpen: boolean;
374
- searchBarIconURL: string;
375
- }) => {
376
- const { closeMenu, onSearch, isMenuOpen, searchBarIconURL } = props;
377
- const [searchValue, setSearchValue] = React.useState("");
378
- const searchInputRef = React.useRef<HTMLInputElement>(null);
379
-
380
- const redirectToSearchResults = (e: FormEvent<HTMLFormElement> | any) => {
381
- closeMenu();
382
- e.preventDefault();
383
- onSearch(searchValue);
384
- };
385
-
386
- React.useEffect(() => {
387
- if (!isMenuOpen) {
388
- setSearchValue("");
389
- }
390
- }, [isMenuOpen]);
391
-
392
- return (
393
- <form
394
- name="searchForm"
395
- className="flex border-b border-t transition-colors focus-within:border-border-focus"
396
- onSubmit={redirectToSearchResults}
397
- >
398
- <NextImage
399
- src={searchBarIconURL}
400
- width={32}
401
- height={32}
402
- alt="Search icon"
403
- role="button"
404
- className="ml-2"
405
- onClick={redirectToSearchResults}
406
- />
407
- <div className="flex-grow">
408
- <Input
409
- ref={searchInputRef}
410
- className={"body3 h-[34px] rounded-none px-3 text-text"}
411
- name="search"
412
- placeholder="Search..."
413
- value={searchValue}
414
- onChange={e => setSearchValue(e.target.value)}
415
- autoComplete="off"
416
- containerClassName="h-[46px] px-4 pl-0 rounded-none flex-grow border-none"
417
- />
418
- </div>
419
- </form>
420
- );
421
- };
422
-
423
- const DesktopSearchInput = (props: {
424
- searchBarIconURL: string;
425
- onSearch: (query: string) => void;
426
- }) => {
427
- const { searchBarIconURL, onSearch } = props;
428
- const [searchValue, setSearchValue] = React.useState("");
429
- const searchInputRef = React.useRef<HTMLInputElement>(null);
430
-
431
- const redirectToSearchResults = (e: FormEvent<HTMLFormElement> | any) => {
432
- e.preventDefault();
433
- onSearch(searchValue);
434
- };
435
-
436
- return (
437
- <form
438
- name="searchForm"
439
- className="flex h-9 w-60 rounded-input-xl border px-1 transition-colors focus-within:border-border-focus"
440
- onSubmit={redirectToSearchResults}
441
- >
442
- <NextImage
443
- src={searchBarIconURL}
444
- width={32}
445
- height={32}
446
- alt="Search icon"
447
- role="button"
448
- onClick={redirectToSearchResults}
449
- />
450
- <Input
451
- ref={searchInputRef}
452
- className={"body3 rounded-full px-3 text-text"}
453
- name="search"
454
- placeholder="Search..."
455
- value={searchValue}
456
- onChange={e => setSearchValue(e.target.value)}
457
- autoComplete="off"
458
- containerClassName="px-0 h-full border-none rounded-full"
459
- />
460
- </form>
461
- );
462
- };
1
+ import React, { FormEvent } from "react";
2
+ import { DesktopLinkGroups } from "./desktop-link-groups.tsx";
3
+ import { MobileLinkGroups } from "./mobile-link-groups.tsx";
4
+ import { NavigationProps } from "./types";
5
+
6
+ import { CallButton } from "@shared/components/call-button";
7
+ import { Input } from "@shared/components/input";
8
+ import { Link } from "@shared/components/link";
9
+ import { MaterialIcon } from "@shared/components/material-icon";
10
+ import { NextImage } from "@shared/components/next-image";
11
+ import { Text } from "@shared/components/text";
12
+ import { Button as ContentfulButton } from "@shared/contentful/blocks/button";
13
+ import { cx } from "@shared/utils";
14
+
15
+ export const Navigation: React.FC<NavigationProps> = props => {
16
+ const {
17
+ primaryNavigationLinks,
18
+ utilityNavigationLinks,
19
+ checkPlansJSX,
20
+ primaryNavigationLogo,
21
+ accountNavigationLinks,
22
+ supportNavigationLinks,
23
+ searchBarIcon,
24
+ invocaPhoneNumberLink,
25
+ invocaPhoneNumberDisplayText,
26
+ callNowCtaText,
27
+ displayCartIcon,
28
+ cartHref,
29
+ cartHasRetention,
30
+ cartIconAriaLabel = "Cart",
31
+ onCallClickDesktop,
32
+ onCallClickMobile,
33
+ onCartClick,
34
+ onSearch = () => {},
35
+ utilityNavActiveIndex,
36
+ primaryNavigationLogoWidth = 76.5,
37
+ primaryNavigationLogoHeight = 24,
38
+ hideMobileCallButton = false,
39
+ } = props;
40
+ return (
41
+ <div className="component-container">
42
+ <nav className={`menu-container z-[1000]`}>
43
+ <div className="utility-container hidden lg:block lg:border-b lg:px-2">
44
+ <div className="mx-auto flex max-w-120 justify-between">
45
+ <ul className="flex gap-5" aria-label="Utility Navigation">
46
+ {utilityNavigationLinks?.map((links, index) => {
47
+ return (
48
+ <li key={`main-menu-items-${index}`}>
49
+ <ContentfulButton
50
+ linkClassName={cx(
51
+ "footnote flex items-center w-full h-11 text-text",
52
+ // If utilityNavActiveIndex is not provided, default to making the second link active (for existing business app). If utilityNavActiveIndex is provided, use it to determine which link is active.
53
+ typeof utilityNavActiveIndex !== "number"
54
+ ? index === 1 && "label4"
55
+ : utilityNavActiveIndex === index && "label4"
56
+ )}
57
+ linkVariant="unstyled"
58
+ {...(Object.fromEntries(
59
+ Object.entries(links).filter(([_, v]) => v !== null)
60
+ ) as any)}
61
+ />
62
+ </li>
63
+ );
64
+ })}
65
+ </ul>
66
+ <div className="flex items-center gap-10">
67
+ <CallButton
68
+ className="border-none"
69
+ href={invocaPhoneNumberLink}
70
+ prefix={callNowCtaText}
71
+ onClick={onCallClickDesktop}
72
+ >
73
+ <Text className="body3 text-text">
74
+ {invocaPhoneNumberDisplayText}
75
+ </Text>
76
+ </CallButton>
77
+ {displayCartIcon ? (
78
+ <Link
79
+ href={cartHref || "#"}
80
+ className="relative inline-flex cursor-pointer"
81
+ aria-label={cartIconAriaLabel}
82
+ onClick={onCartClick}
83
+ >
84
+ <MaterialIcon name="shopping_cart" />
85
+ {cartHasRetention ? (
86
+ <span className="absolute -right-2 -top-1 h-2.5 w-2.5 rounded-full bg-icon-brand" />
87
+ ) : null}
88
+ </Link>
89
+ ) : null}
90
+ {accountNavigationLinks?.map((links, index) => {
91
+ return (
92
+ <DesktopLinkGroups
93
+ key={`my-account-${index}`}
94
+ anchorName={`my-account-${index}`}
95
+ link={links}
96
+ />
97
+ );
98
+ })}
99
+ </div>
100
+ </div>
101
+ </div>
102
+ <div className="main-nav-container" aria-label="Main Navigation">
103
+ <div className="mobile-nav-section flex h-14 items-center justify-between border border-b px-5 py-[10px] lg:hidden">
104
+ <div>
105
+ <Link href="/" className="flex">
106
+ <NextImage
107
+ src={
108
+ typeof primaryNavigationLogo === "string"
109
+ ? primaryNavigationLogo
110
+ : primaryNavigationLogo?.url || ""
111
+ }
112
+ alt="Kinetic business logo"
113
+ width={primaryNavigationLogoWidth}
114
+ height={primaryNavigationLogoHeight}
115
+ />
116
+ </Link>
117
+ </div>
118
+ <div className="flex items-center gap-6">
119
+ {hideMobileCallButton ? null : (
120
+ <CallButton
121
+ href={invocaPhoneNumberLink}
122
+ onClick={onCallClickMobile}
123
+ >
124
+ <Text as="span" className="footnote">
125
+ {invocaPhoneNumberDisplayText}
126
+ </Text>
127
+ </CallButton>
128
+ )}
129
+ {displayCartIcon ? (
130
+ <Link
131
+ href={cartHref || "#"}
132
+ className="relative inline-flex cursor-pointer"
133
+ aria-label={cartIconAriaLabel}
134
+ onClick={onCartClick}
135
+ >
136
+ <MaterialIcon name="shopping_cart" />
137
+ {cartHasRetention ? (
138
+ <span className="absolute -right-2 -top-1 h-2.5 w-2.5 rounded-full bg-icon-brand" />
139
+ ) : null}
140
+ </Link>
141
+ ) : null}
142
+ <MobileMenu {...props} />
143
+ </div>
144
+ </div>
145
+
146
+ <div className="desktop-nav-section hidden lg:block lg:border-b lg:px-2">
147
+ <div className="mx-auto flex h-14 max-w-120 items-center justify-between">
148
+ <div className="flex h-full">
149
+ <Link href="/" className="flex">
150
+ <NextImage
151
+ src={
152
+ typeof primaryNavigationLogo === "string"
153
+ ? primaryNavigationLogo
154
+ : primaryNavigationLogo?.url || ""
155
+ }
156
+ alt="Kinetic business logo"
157
+ width={primaryNavigationLogoWidth}
158
+ height={primaryNavigationLogoHeight}
159
+ className="mr-[64px]"
160
+ />
161
+ </Link>
162
+
163
+ <div className="flex h-full items-center gap-5">
164
+ {primaryNavigationLinks?.map((links, index) => {
165
+ return (
166
+ <DesktopLinkGroups
167
+ key={`main-menu-${index}`}
168
+ anchorName={`main-menu-${index}`}
169
+ link={links}
170
+ />
171
+ );
172
+ })}
173
+ </div>
174
+ </div>
175
+ <div className="flex h-full items-center gap-10">
176
+ <DesktopSearchInput
177
+ searchBarIconURL={
178
+ typeof searchBarIcon === "string"
179
+ ? searchBarIcon
180
+ : searchBarIcon?.url || ""
181
+ }
182
+ onSearch={onSearch}
183
+ />
184
+ {supportNavigationLinks?.map((links, index) => {
185
+ return (
186
+ <DesktopLinkGroups
187
+ key={`support-menu-${index}`}
188
+ anchorName={`support-menu-${index}`}
189
+ link={links}
190
+ />
191
+ );
192
+ })}
193
+ </div>
194
+ </div>
195
+ </div>
196
+ </div>
197
+ </nav>
198
+
199
+ {checkPlansJSX && <div className="md:hidden">{checkPlansJSX}</div>}
200
+ </div>
201
+ );
202
+ };
203
+
204
+ const MobileMenu = (props: NavigationProps) => {
205
+ const {
206
+ primaryNavigationLinks,
207
+ utilityNavigationLinks,
208
+ supportNavigationLinks,
209
+ accountNavigationLinks,
210
+ } = props;
211
+ const [isOpen, setIsOpen] = React.useState(false);
212
+
213
+ React.useEffect(() => {
214
+ if (typeof window === "undefined") return;
215
+
216
+ if (isOpen) document.body.style.overflowY = "hidden";
217
+ else document.body.style.overflowY = "unset";
218
+
219
+ const element = document.getElementById("drawer-items");
220
+ if (!element) return;
221
+
222
+ const focusableEls = element.querySelectorAll(".focus-item");
223
+ const firstFocusableEl = focusableEls[0];
224
+ const lastFocusableEl = focusableEls[focusableEls.length - 1];
225
+
226
+ const handleKeyDown = (e: {
227
+ key: string;
228
+ keyCode: number;
229
+ shiftKey: any;
230
+ preventDefault: () => void;
231
+ }) => {
232
+ const isTabPressed = e.key === "Tab" || e.keyCode === 9;
233
+
234
+ if (!isTabPressed) return;
235
+
236
+ if (e.shiftKey) {
237
+ if (document.activeElement === firstFocusableEl) {
238
+ (lastFocusableEl as HTMLButtonElement).focus?.();
239
+ e.preventDefault();
240
+ }
241
+ } else {
242
+ if (document.activeElement === lastFocusableEl) {
243
+ (firstFocusableEl as HTMLButtonElement).focus?.();
244
+ e.preventDefault();
245
+ }
246
+ }
247
+ };
248
+
249
+ window.addEventListener("keydown", handleKeyDown);
250
+
251
+ return () => {
252
+ document.body.style.overflowY = "unset";
253
+ window.removeEventListener("keydown", handleKeyDown);
254
+ };
255
+ }, [isOpen]);
256
+
257
+ const closeMenu = () => {
258
+ setIsOpen(false);
259
+ };
260
+
261
+ return (
262
+ <div>
263
+ <ContentfulButton
264
+ showButtonAs="unstyled"
265
+ buttonClassName="flex"
266
+ onClick={() => setIsOpen(true)}
267
+ >
268
+ <MaterialIcon name="menu" />
269
+ </ContentfulButton>
270
+ {isOpen ? (
271
+ <div className="fixed bottom-0 left-0 right-0 top-0 z-[90] h-full w-full bg-scrim-bg-modal"></div>
272
+ ) : null}
273
+
274
+ <div
275
+ className={cx(
276
+ "fixed bottom-0 right-0 top-0",
277
+ "z-[100] h-full bg-bg px-0 py-4",
278
+ "transition-all duration-300 ease-in-out",
279
+ "block",
280
+ isOpen ? "right-0" : "-right-96"
281
+ )}
282
+ id="mobile-menu-overlay"
283
+ >
284
+ <div id="drawer-items" className="flex h-full flex-col gap-3">
285
+ <div className="flex items-center justify-between px-4">
286
+ <div>
287
+ <CallButton
288
+ className="border-none"
289
+ href={props.invocaPhoneNumberLink}
290
+ onClick={props.onCallClickMobile}
291
+ >
292
+ {props.invocaPhoneNumberDisplayText}
293
+ </CallButton>
294
+ </div>
295
+ <div>
296
+ <ContentfulButton
297
+ showButtonAs="unstyled"
298
+ buttonClassName="focus-item flex"
299
+ onClick={closeMenu}
300
+ >
301
+ <MaterialIcon name="close" />
302
+ </ContentfulButton>
303
+ </div>
304
+ </div>
305
+ <MobileSearchInput
306
+ closeMenu={closeMenu}
307
+ isMenuOpen={isOpen}
308
+ searchBarIconURL={
309
+ typeof props.searchBarIcon === "string"
310
+ ? props.searchBarIcon
311
+ : props.searchBarIcon?.url || ""
312
+ }
313
+ onSearch={props.onSearch || (() => {})}
314
+ />
315
+ <div
316
+ className="flex-grow overflow-y-auto"
317
+ onClick={event => {
318
+ // Close the drawer when a link (anchor) is clicked.
319
+ // Group-toggle buttons render as <button> and won't match,
320
+ // so they continue to expand/collapse normally.
321
+ // Required for Pages Router where MobileMenu persists across
322
+ // client-side navigations and its local `isOpen` state would
323
+ // otherwise stay true on the new route.
324
+ const target = event.target as HTMLElement | null;
325
+ if (target?.closest("a")) {
326
+ closeMenu();
327
+ }
328
+ }}
329
+ >
330
+ <ul className="mt-2 flex flex-col gap-2">
331
+ {[
332
+ ...(primaryNavigationLinks || []),
333
+ ...(supportNavigationLinks || []),
334
+ ...(accountNavigationLinks || []),
335
+ ].map((links, index) => (
336
+ <li key={`main-menu-items-${index}`}>
337
+ <MobileLinkGroups link={links} />
338
+ </li>
339
+ ))}
340
+ </ul>
341
+
342
+ <ul className="mt-2 flex gap-5 bg-bg-fill-info px-4">
343
+ {utilityNavigationLinks?.map((link, index) => {
344
+ return (
345
+ <li key={`utility-menu-items-${index}`}>
346
+ <ContentfulButton
347
+ key={`utility-submenu-link-btn-${link.anchorId}`}
348
+ {...(Object.fromEntries(
349
+ Object.entries(link).filter(([_, v]) => v !== null)
350
+ ) as any)}
351
+ linkClassName={cx(
352
+ "footnote flex items-center w-full h-11 text-text-link",
353
+ typeof props.utilityNavActiveIndex !== "number"
354
+ ? index === 1 && "label4"
355
+ : props.utilityNavActiveIndex === index && "label4"
356
+ )}
357
+ linkVariant="unstyled"
358
+ />
359
+ </li>
360
+ );
361
+ })}
362
+ </ul>
363
+ </div>
364
+ </div>
365
+ </div>
366
+ </div>
367
+ );
368
+ };
369
+
370
+ const MobileSearchInput = (props: {
371
+ closeMenu: () => void;
372
+ onSearch: (query: string) => void;
373
+ isMenuOpen: boolean;
374
+ searchBarIconURL: string;
375
+ }) => {
376
+ const { closeMenu, onSearch, isMenuOpen, searchBarIconURL } = props;
377
+ const [searchValue, setSearchValue] = React.useState("");
378
+ const searchInputRef = React.useRef<HTMLInputElement>(null);
379
+
380
+ const redirectToSearchResults = (e: FormEvent<HTMLFormElement> | any) => {
381
+ closeMenu();
382
+ e.preventDefault();
383
+ onSearch(searchValue);
384
+ };
385
+
386
+ React.useEffect(() => {
387
+ if (!isMenuOpen) {
388
+ setSearchValue("");
389
+ }
390
+ }, [isMenuOpen]);
391
+
392
+ return (
393
+ <form
394
+ name="searchForm"
395
+ className="flex border-b border-t transition-colors focus-within:border-border-focus"
396
+ onSubmit={redirectToSearchResults}
397
+ >
398
+ <NextImage
399
+ src={searchBarIconURL}
400
+ width={32}
401
+ height={32}
402
+ alt="Search icon"
403
+ role="button"
404
+ className="ml-2"
405
+ onClick={redirectToSearchResults}
406
+ />
407
+ <div className="flex-grow">
408
+ <Input
409
+ ref={searchInputRef}
410
+ className={"body3 h-[34px] rounded-none px-3 text-text"}
411
+ name="search"
412
+ placeholder="Search..."
413
+ value={searchValue}
414
+ onChange={e => setSearchValue(e.target.value)}
415
+ autoComplete="off"
416
+ containerClassName="h-[46px] px-4 pl-0 rounded-none flex-grow border-none"
417
+ />
418
+ </div>
419
+ </form>
420
+ );
421
+ };
422
+
423
+ const DesktopSearchInput = (props: {
424
+ searchBarIconURL: string;
425
+ onSearch: (query: string) => void;
426
+ }) => {
427
+ const { searchBarIconURL, onSearch } = props;
428
+ const [searchValue, setSearchValue] = React.useState("");
429
+ const searchInputRef = React.useRef<HTMLInputElement>(null);
430
+
431
+ const redirectToSearchResults = (e: FormEvent<HTMLFormElement> | any) => {
432
+ e.preventDefault();
433
+ onSearch(searchValue);
434
+ };
435
+
436
+ return (
437
+ <form
438
+ name="searchForm"
439
+ className="flex h-9 w-60 rounded-input-xl border px-1 transition-colors focus-within:border-border-focus"
440
+ onSubmit={redirectToSearchResults}
441
+ >
442
+ <NextImage
443
+ src={searchBarIconURL}
444
+ width={32}
445
+ height={32}
446
+ alt="Search icon"
447
+ role="button"
448
+ onClick={redirectToSearchResults}
449
+ />
450
+ <Input
451
+ ref={searchInputRef}
452
+ className={"body3 rounded-full px-3 text-text"}
453
+ name="search"
454
+ placeholder="Search..."
455
+ value={searchValue}
456
+ onChange={e => setSearchValue(e.target.value)}
457
+ autoComplete="off"
458
+ containerClassName="px-0 h-full border-none rounded-full"
459
+ />
460
+ </form>
461
+ );
462
+ };