@windstream/react-shared-components 0.1.26 → 0.1.27

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 (158) 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 +2 -2
  5. package/dist/contentful/index.js.map +1 -1
  6. package/dist/index.d.ts +4 -4
  7. package/dist/index.js.map +1 -1
  8. package/dist/styles.css +1 -1
  9. package/package.json +182 -182
  10. package/src/components/accordion/Accordion.stories.tsx +230 -230
  11. package/src/components/accordion/types.ts +11 -11
  12. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  13. package/src/components/alert-card/index.tsx +41 -41
  14. package/src/components/alert-card/types.ts +13 -13
  15. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  16. package/src/components/brand-button/helpers.ts +35 -35
  17. package/src/components/brand-button/index.tsx +115 -115
  18. package/src/components/brand-button/types.ts +37 -37
  19. package/src/components/button/Button.stories.tsx +108 -108
  20. package/src/components/button/index.tsx +27 -27
  21. package/src/components/button/types.ts +14 -14
  22. package/src/components/call-button/CallButton.stories.tsx +324 -324
  23. package/src/components/call-button/index.tsx +86 -86
  24. package/src/components/call-button/types.ts +11 -11
  25. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  26. package/src/components/checkbox/index.tsx +197 -197
  27. package/src/components/checkbox/types.ts +27 -27
  28. package/src/components/checklist/Checklist.stories.tsx +150 -150
  29. package/src/components/checklist/index.tsx +59 -59
  30. package/src/components/checklist/types.ts +16 -16
  31. package/src/components/collapse/Collapse.stories.tsx +255 -255
  32. package/src/components/collapse/index.tsx +46 -46
  33. package/src/components/collapse/types.ts +6 -6
  34. package/src/components/divider/Divider.stories.tsx +205 -205
  35. package/src/components/divider/index.tsx +22 -22
  36. package/src/components/divider/type.ts +3 -3
  37. package/src/components/image/Image.stories.tsx +113 -113
  38. package/src/components/image/index.tsx +25 -25
  39. package/src/components/image/types.ts +40 -40
  40. package/src/components/input/Input.stories.tsx +325 -325
  41. package/src/components/input/index.tsx +177 -177
  42. package/src/components/input/types.ts +37 -37
  43. package/src/components/link/Link.stories.tsx +163 -163
  44. package/src/components/link/types.ts +25 -25
  45. package/src/components/list/List.stories.tsx +272 -272
  46. package/src/components/list/index.tsx +88 -88
  47. package/src/components/list/list-item/index.tsx +38 -38
  48. package/src/components/list/list-item/types.ts +13 -13
  49. package/src/components/list/types.ts +29 -29
  50. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  51. package/src/components/material-icon/constants.ts +98 -98
  52. package/src/components/material-icon/index.tsx +47 -47
  53. package/src/components/material-icon/types.ts +31 -31
  54. package/src/components/modal/Modal.stories.tsx +171 -171
  55. package/src/components/modal/index.tsx +164 -164
  56. package/src/components/modal/types.ts +24 -24
  57. package/src/components/next-image/index.tsx +54 -54
  58. package/src/components/next-image/types.ts +1 -1
  59. package/src/components/pagination/index.tsx +100 -100
  60. package/src/components/pagination/types.ts +6 -6
  61. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  62. package/src/components/radio-button/index.tsx +75 -75
  63. package/src/components/radio-button/types.ts +21 -21
  64. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  65. package/src/components/see-more/index.tsx +44 -44
  66. package/src/components/see-more/types.ts +4 -4
  67. package/src/components/select/Select.stories.tsx +411 -411
  68. package/src/components/select/index.tsx +155 -155
  69. package/src/components/select/types.ts +36 -36
  70. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  71. package/src/components/select-plan-button/index.tsx +57 -57
  72. package/src/components/select-plan-button/types.ts +14 -14
  73. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  74. package/src/components/skeleton/index.tsx +61 -61
  75. package/src/components/skeleton/types.ts +4 -4
  76. package/src/components/spinner/Spinner.stories.tsx +335 -335
  77. package/src/components/spinner/index.tsx +44 -44
  78. package/src/components/spinner/types.ts +5 -5
  79. package/src/components/text/Text.stories.tsx +321 -321
  80. package/src/components/text/index.tsx +25 -25
  81. package/src/components/text/types.ts +45 -45
  82. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  83. package/src/components/tooltip/index.tsx +74 -74
  84. package/src/components/tooltip/types.ts +7 -7
  85. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  86. package/src/components/view-cart-button/index.tsx +42 -42
  87. package/src/components/view-cart-button/types.ts +5 -5
  88. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  89. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  90. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  91. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  92. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  93. package/src/contentful/blocks/button/index.tsx +121 -121
  94. package/src/contentful/blocks/button/types.ts +36 -36
  95. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  96. package/src/contentful/blocks/callout/index.tsx +86 -86
  97. package/src/contentful/blocks/callout/types.ts +15 -15
  98. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  99. package/src/contentful/blocks/cards/blog-card/index.tsx +109 -109
  100. package/src/contentful/blocks/cards/blog-card/types.ts +16 -16
  101. package/src/contentful/blocks/cards/index.tsx +13 -13
  102. package/src/contentful/blocks/cards/product-card/index.tsx +208 -208
  103. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  104. package/src/contentful/blocks/cards/simple-card/index.tsx +1 -1
  105. package/src/contentful/blocks/cards/testimonial-card/index.tsx +88 -88
  106. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  107. package/src/contentful/blocks/cards/types.ts +1 -1
  108. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  109. package/src/contentful/blocks/carousel/helper.tsx +356 -356
  110. package/src/contentful/blocks/carousel/index.tsx +73 -73
  111. package/src/contentful/blocks/carousel/types.ts +143 -143
  112. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  113. package/src/contentful/blocks/cta-callout/index.tsx +60 -60
  114. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  115. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  116. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  117. package/src/contentful/blocks/find-kinetic/index.tsx +130 -130
  118. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  119. package/src/contentful/blocks/floating-banner/index.tsx +93 -97
  120. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  121. package/src/contentful/blocks/footer/Footer.stories.tsx +30 -30
  122. package/src/contentful/blocks/footer/index.tsx +6 -12
  123. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  124. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  125. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  126. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  127. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  128. package/src/contentful/blocks/modal/constants.ts +53 -53
  129. package/src/contentful/blocks/modal/types.ts +12 -12
  130. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +113 -113
  131. package/src/contentful/blocks/navigation/index.tsx +394 -394
  132. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  133. package/src/contentful/blocks/navigation/types.ts +41 -41
  134. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  135. package/src/contentful/blocks/primary-hero/index.tsx +234 -234
  136. package/src/contentful/blocks/primary-hero/types.ts +35 -35
  137. package/src/contentful/blocks/search-block/index.tsx +90 -90
  138. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  139. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  140. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  141. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  142. package/src/contentful/blocks/text/index.tsx +12 -12
  143. package/src/contentful/blocks/text/types.ts +1 -1
  144. package/src/contentful/index.ts +75 -75
  145. package/src/hooks/use-body-scroll-lock.ts +34 -34
  146. package/src/hooks/use-outside-click.ts +17 -17
  147. package/src/index.ts +96 -96
  148. package/src/next/index.ts +5 -5
  149. package/src/setupTests.ts +46 -46
  150. package/src/stories/DocsTemplate.tsx +24 -24
  151. package/src/styles/globals.css +343 -343
  152. package/src/types/global.d.ts +9 -9
  153. package/src/types/micro-components.ts +99 -99
  154. package/src/types/utm.ts +49 -49
  155. package/src/utils/cookie.ts +58 -58
  156. package/src/utils/index.ts +65 -65
  157. package/src/utils/utm.ts +221 -221
  158. package/tailwind.config.cjs +1 -0
@@ -1,24 +1,24 @@
1
- import { Props as RModalProps } from "react-modal";
2
-
3
- export type Size = "xl" | "lg" | "md" | "sm" | "xs";
4
- export type Shape = "rounded" | "default";
5
- export type Animation = "popper" | "bottomSheet";
6
-
7
- type OwnProps = {
8
- size?: Size;
9
- shape?: Shape;
10
- title?: string;
11
- bodyClassName?: string | null | undefined;
12
- closeWrapperClassName?: string | null | undefined;
13
- closeButtonClassName?: string | null | undefined;
14
- bodyStyle?: React.CSSProperties;
15
- centered?: boolean;
16
- hideScrollOnIsOpenFalse?: boolean;
17
- parentSelector?: string;
18
- animation?: Animation;
19
- hideCloseButton?: boolean;
20
- ["data-testid"]?: string;
21
- ["data-cy"]?: string;
22
- };
23
-
24
- export type ModalProps = OwnProps & RModalProps;
1
+ import { Props as RModalProps } from "react-modal";
2
+
3
+ export type Size = "xl" | "lg" | "md" | "sm" | "xs";
4
+ export type Shape = "rounded" | "default";
5
+ export type Animation = "popper" | "bottomSheet";
6
+
7
+ type OwnProps = {
8
+ size?: Size;
9
+ shape?: Shape;
10
+ title?: string;
11
+ bodyClassName?: string | null | undefined;
12
+ closeWrapperClassName?: string | null | undefined;
13
+ closeButtonClassName?: string | null | undefined;
14
+ bodyStyle?: React.CSSProperties;
15
+ centered?: boolean;
16
+ hideScrollOnIsOpenFalse?: boolean;
17
+ parentSelector?: string;
18
+ animation?: Animation;
19
+ hideCloseButton?: boolean;
20
+ ["data-testid"]?: string;
21
+ ["data-cy"]?: string;
22
+ };
23
+
24
+ export type ModalProps = OwnProps & RModalProps;
@@ -1,54 +1,54 @@
1
- "use client";
2
-
3
- import { forwardRef } from "react";
4
- import NextJsImage, {
5
- type ImageLoaderProps,
6
- type ImageProps as NextImageProps,
7
- } from "next/image";
8
-
9
- import { cx } from "@shared/utils";
10
-
11
- export interface NextImageComponentProps extends NextImageProps {
12
- className?: string;
13
- }
14
-
15
- /**
16
- * Image loader that uses Contentful's Image API to serve optimized WebP images
17
- * at the requested width and quality, avoiding an extra round-trip through
18
- * the Next.js image optimization server.
19
- */
20
- const contentfulImageLoader = ({ src, width, quality }: ImageLoaderProps) => {
21
- const url = new URL(src);
22
- url.searchParams.set("w", String(width));
23
- url.searchParams.set("q", String(quality || 90));
24
- url.searchParams.set("fm", "webp");
25
- return url.toString();
26
- };
27
-
28
- export const NextImage = forwardRef<HTMLImageElement, NextImageComponentProps>(
29
- ({ className, ...props }, ref) => {
30
- const srcString = typeof props.src === "string" ? props.src : "";
31
- const urlWithoutParams = srcString.toLowerCase().split("?")[0] || "";
32
- const isContentfulImage = srcString.includes("images.ctfassets.net");
33
- const isSvgFromContentful =
34
- isContentfulImage && urlWithoutParams.endsWith(".svg");
35
-
36
- // Use Contentful's Image API for non-SVG Contentful images;
37
- // skip optimization entirely for SVGs.
38
- const loaderProps =
39
- isContentfulImage && !isSvgFromContentful
40
- ? { loader: contentfulImageLoader, unoptimized: false }
41
- : { unoptimized: isSvgFromContentful };
42
-
43
- return (
44
- <NextJsImage
45
- ref={ref}
46
- className={cx(className)}
47
- {...props}
48
- {...loaderProps}
49
- />
50
- );
51
- }
52
- );
53
-
54
- NextImage.displayName = "NextImage";
1
+ "use client";
2
+
3
+ import { forwardRef } from "react";
4
+ import NextJsImage, {
5
+ type ImageLoaderProps,
6
+ type ImageProps as NextImageProps,
7
+ } from "next/image";
8
+
9
+ import { cx } from "@shared/utils";
10
+
11
+ export interface NextImageComponentProps extends NextImageProps {
12
+ className?: string;
13
+ }
14
+
15
+ /**
16
+ * Image loader that uses Contentful's Image API to serve optimized WebP images
17
+ * at the requested width and quality, avoiding an extra round-trip through
18
+ * the Next.js image optimization server.
19
+ */
20
+ const contentfulImageLoader = ({ src, width, quality }: ImageLoaderProps) => {
21
+ const url = new URL(src);
22
+ url.searchParams.set("w", String(width));
23
+ url.searchParams.set("q", String(quality || 90));
24
+ url.searchParams.set("fm", "webp");
25
+ return url.toString();
26
+ };
27
+
28
+ export const NextImage = forwardRef<HTMLImageElement, NextImageComponentProps>(
29
+ ({ className, ...props }, ref) => {
30
+ const srcString = typeof props.src === "string" ? props.src : "";
31
+ const urlWithoutParams = srcString.toLowerCase().split("?")[0] || "";
32
+ const isContentfulImage = srcString.includes("images.ctfassets.net");
33
+ const isSvgFromContentful =
34
+ isContentfulImage && urlWithoutParams.endsWith(".svg");
35
+
36
+ // Use Contentful's Image API for non-SVG Contentful images;
37
+ // skip optimization entirely for SVGs.
38
+ const loaderProps =
39
+ isContentfulImage && !isSvgFromContentful
40
+ ? { loader: contentfulImageLoader, unoptimized: false }
41
+ : { unoptimized: isSvgFromContentful };
42
+
43
+ return (
44
+ <NextJsImage
45
+ ref={ref}
46
+ className={cx(className)}
47
+ {...props}
48
+ {...loaderProps}
49
+ />
50
+ );
51
+ }
52
+ );
53
+
54
+ NextImage.displayName = "NextImage";
@@ -1 +1 @@
1
- export type { NextImageComponentProps } from "./index";
1
+ export type { NextImageComponentProps } from "./index";
@@ -1,100 +1,100 @@
1
- "use client";
2
-
3
- import React from "react";
4
- import { PaginationProps } from "./types";
5
-
6
- import { Button } from "@shared/components/button";
7
- import { MaterialIcon } from "@shared/components/material-icon";
8
-
9
- export const Pagination: React.FC<PaginationProps> = ({
10
- currentPage,
11
- totalPages,
12
- onPageChange,
13
- ariaLabel = "Pagination",
14
- }: PaginationProps) => {
15
- const pageItems = buildPageItems(currentPage, totalPages);
16
-
17
- const navBtnBase =
18
- "inline-flex items-center justify-center w-9 h-9 rounded-full bg-transparent cursor-pointer transition-colors duration-150 hover:border-text-brand hover:text-text-brand disabled:opacity-40 disabled:cursor-not-allowed";
19
-
20
- const pageBtnBase =
21
- "inline-flex items-center justify-center w-9 h-9 rounded-lg text-body2 text-gray-700 bg-transparent border-none cursor-pointer transition-colors duration-150 hover:bg-gray-100 hover:text-gray-900";
22
-
23
- const pageBtnActive = "bg-bg-surface-active font-bold hover:bg-bg-gray-300";
24
-
25
- return (
26
- <nav
27
- className="flex items-center justify-center gap-1 px-6 pb-12"
28
- aria-label={ariaLabel}
29
- >
30
- {/* Previous */}
31
- <Button
32
- className={navBtnBase}
33
- onClick={() => onPageChange(currentPage - 1)}
34
- disabled={currentPage === 1}
35
- aria-label="Go to previous page"
36
- >
37
- <MaterialIcon name="chevron_left" size={32} aria-hidden="true" />
38
- </Button>
39
-
40
- {/* Page numbers */}
41
- {pageItems.map((item, idx) =>
42
- item === "..." ? (
43
- <span
44
- key={`ellipsis-${idx}`}
45
- className="inline-flex h-9 w-9 cursor-default items-center justify-center text-body2 text-gray-500"
46
- aria-hidden="true"
47
- >
48
-
49
- </span>
50
- ) : (
51
- <Button
52
- key={item}
53
- className={`${pageBtnBase} ${item === currentPage ? pageBtnActive : ""}`}
54
- onClick={() => onPageChange(item)}
55
- aria-label={`Go to page ${item}`}
56
- aria-current={item === currentPage ? "page" : undefined}
57
- >
58
- {item}
59
- </Button>
60
- )
61
- )}
62
-
63
- {/* Next */}
64
- <Button
65
- className={navBtnBase}
66
- onClick={() => onPageChange(currentPage + 1)}
67
- disabled={currentPage === totalPages}
68
- aria-label="Go to next page"
69
- >
70
- <MaterialIcon name="chevron_right" size={32} aria-hidden="true" />
71
- </Button>
72
- </nav>
73
- );
74
- };
75
-
76
- /** Returns an array of page numbers and ellipsis to render. */
77
- function buildPageItems(current: number, total: number): (number | "...")[] {
78
- if (total <= 7) {
79
- return Array.from({ length: total }, (_, i) => i + 1);
80
- }
81
- const items: (number | "...")[] = [];
82
- items.push(1);
83
-
84
- if (current > 3) {
85
- items.push("...");
86
- }
87
- const rangeStart = Math.max(2, current - 1);
88
- const rangeEnd = Math.min(total - 1, current + 1);
89
-
90
- for (let i = rangeStart; i <= rangeEnd; i++) {
91
- items.push(i);
92
- }
93
-
94
- if (current < total - 2) {
95
- items.push("...");
96
- }
97
-
98
- items.push(total);
99
- return items;
100
- }
1
+ "use client";
2
+
3
+ import React from "react";
4
+ import { PaginationProps } from "./types";
5
+
6
+ import { Button } from "@shared/components/button";
7
+ import { MaterialIcon } from "@shared/components/material-icon";
8
+
9
+ export const Pagination: React.FC<PaginationProps> = ({
10
+ currentPage,
11
+ totalPages,
12
+ onPageChange,
13
+ ariaLabel = "Pagination",
14
+ }: PaginationProps) => {
15
+ const pageItems = buildPageItems(currentPage, totalPages);
16
+
17
+ const navBtnBase =
18
+ "inline-flex items-center justify-center w-9 h-9 rounded-full bg-transparent cursor-pointer transition-colors duration-150 hover:border-text-brand hover:text-text-brand disabled:opacity-40 disabled:cursor-not-allowed";
19
+
20
+ const pageBtnBase =
21
+ "inline-flex items-center justify-center w-9 h-9 rounded-lg text-body2 text-gray-700 bg-transparent border-none cursor-pointer transition-colors duration-150 hover:bg-gray-100 hover:text-gray-900";
22
+
23
+ const pageBtnActive = "bg-bg-surface-active font-bold hover:bg-bg-gray-300";
24
+
25
+ return (
26
+ <nav
27
+ className="flex items-center justify-center gap-1 px-6 pb-12"
28
+ aria-label={ariaLabel}
29
+ >
30
+ {/* Previous */}
31
+ <Button
32
+ className={navBtnBase}
33
+ onClick={() => onPageChange(currentPage - 1)}
34
+ disabled={currentPage === 1}
35
+ aria-label="Go to previous page"
36
+ >
37
+ <MaterialIcon name="chevron_left" size={32} aria-hidden="true" />
38
+ </Button>
39
+
40
+ {/* Page numbers */}
41
+ {pageItems.map((item, idx) =>
42
+ item === "..." ? (
43
+ <span
44
+ key={`ellipsis-${idx}`}
45
+ className="inline-flex h-9 w-9 cursor-default items-center justify-center text-body2 text-gray-500"
46
+ aria-hidden="true"
47
+ >
48
+
49
+ </span>
50
+ ) : (
51
+ <Button
52
+ key={item}
53
+ className={`${pageBtnBase} ${item === currentPage ? pageBtnActive : ""}`}
54
+ onClick={() => onPageChange(item)}
55
+ aria-label={`Go to page ${item}`}
56
+ aria-current={item === currentPage ? "page" : undefined}
57
+ >
58
+ {item}
59
+ </Button>
60
+ )
61
+ )}
62
+
63
+ {/* Next */}
64
+ <Button
65
+ className={navBtnBase}
66
+ onClick={() => onPageChange(currentPage + 1)}
67
+ disabled={currentPage === totalPages}
68
+ aria-label="Go to next page"
69
+ >
70
+ <MaterialIcon name="chevron_right" size={32} aria-hidden="true" />
71
+ </Button>
72
+ </nav>
73
+ );
74
+ };
75
+
76
+ /** Returns an array of page numbers and ellipsis to render. */
77
+ function buildPageItems(current: number, total: number): (number | "...")[] {
78
+ if (total <= 7) {
79
+ return Array.from({ length: total }, (_, i) => i + 1);
80
+ }
81
+ const items: (number | "...")[] = [];
82
+ items.push(1);
83
+
84
+ if (current > 3) {
85
+ items.push("...");
86
+ }
87
+ const rangeStart = Math.max(2, current - 1);
88
+ const rangeEnd = Math.min(total - 1, current + 1);
89
+
90
+ for (let i = rangeStart; i <= rangeEnd; i++) {
91
+ items.push(i);
92
+ }
93
+
94
+ if (current < total - 2) {
95
+ items.push("...");
96
+ }
97
+
98
+ items.push(total);
99
+ return items;
100
+ }
@@ -1,6 +1,6 @@
1
- export interface PaginationProps {
2
- currentPage: number;
3
- totalPages: number;
4
- onPageChange: (page: number) => void;
5
- ariaLabel?: string;
6
- }
1
+ export interface PaginationProps {
2
+ currentPage: number;
3
+ totalPages: number;
4
+ onPageChange: (page: number) => void;
5
+ ariaLabel?: string;
6
+ }