@windstream/react-shared-components 0.1.31 → 0.1.32

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 (166) hide show
  1. package/README.md +635 -635
  2. package/dist/contentful/index.esm.js +2 -2
  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/core.d.ts +2 -2
  7. package/dist/index.d.ts +4 -4
  8. package/dist/index.esm.js +1 -1
  9. package/dist/index.esm.js.map +1 -1
  10. package/dist/index.js +1 -1
  11. package/dist/index.js.map +1 -1
  12. package/dist/styles.css +1 -1
  13. package/package.json +182 -182
  14. package/src/components/accordion/Accordion.stories.tsx +230 -230
  15. package/src/components/accordion/index.tsx +70 -70
  16. package/src/components/accordion/types.ts +12 -12
  17. package/src/components/alert-card/AlertCard.stories.tsx +171 -171
  18. package/src/components/alert-card/index.tsx +41 -41
  19. package/src/components/alert-card/types.ts +13 -13
  20. package/src/components/brand-button/BrandButton.stories.tsx +223 -223
  21. package/src/components/brand-button/helpers.ts +35 -35
  22. package/src/components/brand-button/index.tsx +115 -115
  23. package/src/components/brand-button/types.ts +37 -37
  24. package/src/components/button/Button.stories.tsx +108 -108
  25. package/src/components/button/index.tsx +27 -27
  26. package/src/components/button/types.ts +14 -14
  27. package/src/components/call-button/CallButton.stories.tsx +324 -324
  28. package/src/components/call-button/index.tsx +86 -86
  29. package/src/components/call-button/types.ts +11 -11
  30. package/src/components/checkbox/Checkbox.stories.tsx +247 -247
  31. package/src/components/checkbox/index.tsx +197 -197
  32. package/src/components/checkbox/types.ts +27 -27
  33. package/src/components/checklist/Checklist.stories.tsx +150 -150
  34. package/src/components/checklist/index.tsx +61 -61
  35. package/src/components/checklist/types.ts +17 -17
  36. package/src/components/collapse/Collapse.stories.tsx +255 -255
  37. package/src/components/collapse/index.tsx +46 -46
  38. package/src/components/collapse/types.ts +6 -6
  39. package/src/components/divider/Divider.stories.tsx +205 -205
  40. package/src/components/divider/index.tsx +22 -22
  41. package/src/components/divider/type.ts +3 -3
  42. package/src/components/image/Image.stories.tsx +113 -113
  43. package/src/components/image/index.tsx +25 -25
  44. package/src/components/image/types.ts +40 -40
  45. package/src/components/input/Input.stories.tsx +325 -325
  46. package/src/components/input/index.tsx +177 -177
  47. package/src/components/input/types.ts +37 -37
  48. package/src/components/link/Link.stories.tsx +163 -163
  49. package/src/components/link/types.ts +25 -25
  50. package/src/components/list/List.stories.tsx +272 -272
  51. package/src/components/list/index.tsx +88 -88
  52. package/src/components/list/list-item/index.tsx +38 -38
  53. package/src/components/list/list-item/types.ts +13 -13
  54. package/src/components/list/types.ts +29 -29
  55. package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
  56. package/src/components/material-icon/constants.ts +98 -98
  57. package/src/components/material-icon/index.tsx +47 -47
  58. package/src/components/material-icon/types.ts +31 -31
  59. package/src/components/modal/Modal.stories.tsx +171 -171
  60. package/src/components/modal/index.tsx +164 -164
  61. package/src/components/modal/types.ts +24 -24
  62. package/src/components/next-image/index.tsx +54 -54
  63. package/src/components/next-image/types.ts +1 -1
  64. package/src/components/pagination/index.tsx +100 -100
  65. package/src/components/pagination/types.ts +6 -6
  66. package/src/components/radio-button/RadioButton.stories.tsx +307 -307
  67. package/src/components/radio-button/index.tsx +75 -75
  68. package/src/components/radio-button/types.ts +21 -21
  69. package/src/components/see-more/SeeMore.stories.tsx +181 -181
  70. package/src/components/see-more/index.tsx +44 -44
  71. package/src/components/see-more/types.ts +4 -4
  72. package/src/components/select/Select.stories.tsx +411 -411
  73. package/src/components/select/index.tsx +155 -155
  74. package/src/components/select/types.ts +36 -36
  75. package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
  76. package/src/components/select-plan-button/index.tsx +59 -59
  77. package/src/components/select-plan-button/types.ts +17 -17
  78. package/src/components/skeleton/Skeleton.stories.tsx +179 -179
  79. package/src/components/skeleton/index.tsx +61 -61
  80. package/src/components/skeleton/types.ts +4 -4
  81. package/src/components/spinner/Spinner.stories.tsx +335 -335
  82. package/src/components/spinner/index.tsx +44 -44
  83. package/src/components/spinner/types.ts +5 -5
  84. package/src/components/text/Text.stories.tsx +321 -321
  85. package/src/components/text/index.tsx +25 -25
  86. package/src/components/text/types.ts +45 -45
  87. package/src/components/tooltip/Tooltip.stories.tsx +219 -219
  88. package/src/components/tooltip/index.tsx +74 -74
  89. package/src/components/tooltip/types.ts +7 -7
  90. package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
  91. package/src/components/view-cart-button/index.tsx +42 -42
  92. package/src/components/view-cart-button/types.ts +5 -5
  93. package/src/contentful/blocks/accordion/index.tsx +62 -62
  94. package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
  95. package/src/contentful/blocks/address-input-banner/types.ts +14 -14
  96. package/src/contentful/blocks/anchored-bottom-banner/index.tsx +65 -65
  97. package/src/contentful/blocks/anchored-bottom-banner/types.ts +9 -9
  98. package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
  99. package/src/contentful/blocks/blogs-grid/types.ts +26 -26
  100. package/src/contentful/blocks/button/Button.stories.tsx +40 -40
  101. package/src/contentful/blocks/button/index.tsx +129 -129
  102. package/src/contentful/blocks/button/types.ts +39 -39
  103. package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
  104. package/src/contentful/blocks/callout/index.tsx +88 -88
  105. package/src/contentful/blocks/callout/types.ts +15 -15
  106. package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
  107. package/src/contentful/blocks/cards/blog-card/index.tsx +110 -110
  108. package/src/contentful/blocks/cards/blog-card/types.ts +18 -18
  109. package/src/contentful/blocks/cards/index.tsx +13 -13
  110. package/src/contentful/blocks/cards/product-card/index.tsx +252 -252
  111. package/src/contentful/blocks/cards/product-card/types.ts +28 -28
  112. package/src/contentful/blocks/cards/simple-card/index.tsx +89 -89
  113. package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
  114. package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
  115. package/src/contentful/blocks/cards/types.ts +1 -1
  116. package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
  117. package/src/contentful/blocks/carousel/helper.tsx +440 -440
  118. package/src/contentful/blocks/carousel/index.tsx +85 -85
  119. package/src/contentful/blocks/carousel/types.ts +144 -144
  120. package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
  121. package/src/contentful/blocks/cta-callout/index.tsx +60 -60
  122. package/src/contentful/blocks/cta-callout/types.ts +26 -26
  123. package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
  124. package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
  125. package/src/contentful/blocks/find-kinetic/index.tsx +130 -130
  126. package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
  127. package/src/contentful/blocks/floating-banner/index.tsx +97 -97
  128. package/src/contentful/blocks/floating-banner/types.ts +22 -22
  129. package/src/contentful/blocks/footer/Footer.stories.tsx +30 -30
  130. package/src/contentful/blocks/footer/index.tsx +90 -90
  131. package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
  132. package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
  133. package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
  134. package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
  135. package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
  136. package/src/contentful/blocks/modal/constants.ts +53 -53
  137. package/src/contentful/blocks/modal/types.ts +12 -12
  138. package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +113 -113
  139. package/src/contentful/blocks/navigation/index.tsx +394 -394
  140. package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
  141. package/src/contentful/blocks/navigation/types.ts +41 -41
  142. package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
  143. package/src/contentful/blocks/primary-hero/index.tsx +234 -234
  144. package/src/contentful/blocks/primary-hero/types.ts +35 -35
  145. package/src/contentful/blocks/search-block/index.tsx +90 -90
  146. package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
  147. package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
  148. package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
  149. package/src/contentful/blocks/text/Text.stories.tsx +23 -23
  150. package/src/contentful/blocks/text/index.tsx +12 -12
  151. package/src/contentful/blocks/text/types.ts +1 -1
  152. package/src/contentful/index.ts +78 -78
  153. package/src/hooks/use-body-scroll-lock.ts +34 -34
  154. package/src/hooks/use-carousel-swipe.ts +264 -264
  155. package/src/hooks/use-outside-click.ts +17 -17
  156. package/src/index.ts +101 -101
  157. package/src/next/index.ts +5 -5
  158. package/src/setupTests.ts +46 -46
  159. package/src/stories/DocsTemplate.tsx +24 -24
  160. package/src/styles/globals.css +343 -343
  161. package/src/types/global.d.ts +9 -9
  162. package/src/types/micro-components.ts +99 -99
  163. package/src/types/utm.ts +49 -49
  164. package/src/utils/cookie.ts +58 -58
  165. package/src/utils/index.ts +65 -65
  166. package/src/utils/utm.ts +221 -221
@@ -1,252 +1,252 @@
1
- "use client";
2
-
3
- import React, { useState } from "react";
4
- import { ProductCardProps } from "./types";
5
-
6
- import { Button } from "@shared/components/button";
7
- import { Checklist } from "@shared/components/checklist";
8
- import { Image } from "@shared/components/image";
9
- import { MaterialIcon } from "@shared/components/material-icon";
10
- import { SelectPlanButton } from "@shared/components/select-plan-button";
11
- import { Text } from "@shared/components/text";
12
- import { cx } from "@shared/utils";
13
-
14
- export const ProductCard: React.FC<ProductCardProps> = ({
15
- planName,
16
- planSubtext,
17
- speed,
18
- price,
19
- description,
20
- bestValue = false,
21
- bestValueText = "best value",
22
- giftBadge,
23
- innerBadge,
24
- theme = "light",
25
- featuresTitle = "Business Ready Internet features",
26
- features = [],
27
- isExpanded: controlledExpanded,
28
- onToggleExpand,
29
- onCtaClick,
30
- renderCheckPlans,
31
- cta,
32
- techType,
33
- ismaxSpeed,
34
- }) => {
35
- const [internalExpanded, setInternalExpanded] = useState(false);
36
- // Use parent state if provided, otherwise use internal state
37
- const isExpanded =
38
- controlledExpanded !== undefined ? controlledExpanded : internalExpanded;
39
-
40
- const handleToggle = () => {
41
- if (onToggleExpand) {
42
- onToggleExpand();
43
- } else {
44
- setInternalExpanded(!internalExpanded);
45
- }
46
- };
47
-
48
- const isDark = theme === "dark";
49
-
50
- return (
51
- <article
52
- className={cx(
53
- "relative flex h-full w-full max-w-[392px] flex-col",
54
- !bestValue && "md:pt-[38px]"
55
- )}
56
- >
57
- {/* Best Value Banner */}
58
- {bestValue && (
59
- <div className="label3 rounded-t-[20px] bg-bg-fill-brand-accent px-6 py-2 text-center text-text">
60
- {bestValueText}
61
- </div>
62
- )}
63
-
64
- {/* Main Card */}
65
- <div
66
- className={cx(
67
- "flex flex-grow flex-col gap-5 rounded-card px-5 py-7 shadow-cardDrop",
68
- isDark
69
- ? "text-inverse border-2 border-border-accent bg-bg-fill-inverse"
70
- : "bg-bg text-text",
71
- bestValue ? "rounded-t-none border-t-0" : ""
72
- )}
73
- >
74
- {/* Plan Name & Price */}
75
- <header className="overflow-none flex items-start justify-between gap-4">
76
- <div className="flex flex-col">
77
- <Text
78
- as="h3"
79
- className={cx(
80
- "subheading2",
81
- isDark ? "text-text-accent-on-bg-inverse" : "text-text-brand"
82
- )}
83
- >
84
- {planName}
85
- </Text>
86
- <Text
87
- as="p"
88
- className={`label3 text-wrap ${isDark ? "text-text-accent-on-bg-inverse" : "text-text-brand"}`}
89
- >
90
- {planSubtext}
91
- </Text>
92
- </div>
93
- <div className="flex items-start">
94
- <Text
95
- as="span"
96
- className={cx(
97
- "label2 mr-1 mt-[2px]",
98
- isDark ? "text-text-inverse" : "text-text"
99
- )}
100
- >
101
- $
102
- </Text>
103
- <Text
104
- as="span"
105
- className={cx(
106
- "subheading6",
107
- isDark ? "text-text-inverse" : "text-text"
108
- )}
109
- >
110
- {price.split(".")[0]}
111
- </Text>
112
- <Text
113
- as="span"
114
- className={cx(
115
- "label2",
116
- isDark ? "text-text-inverse" : "text-text"
117
- )}
118
- >
119
- {price.split(".")[1]}/mo
120
- </Text>
121
- </div>
122
- </header>
123
-
124
- {/* Description */}
125
- <section>
126
- <Text
127
- className={cx("body2", isDark ? "text-text-inverse" : "text-text")}
128
- >
129
- {description}
130
- </Text>
131
- </section>
132
-
133
- {/* Gift Badge */}
134
- {giftBadge?.length ? (
135
- <div
136
- className={cx(
137
- "flex-col items-center rounded-surface-sm p-2",
138
- isDark ? "bg-bg-surface-inverse" : "bg-bg-surface-info"
139
- )}
140
- >
141
- {giftBadge?.map((badge: any) => {
142
- return (
143
- <div key={badge.title} className="flex items-center gap-2">
144
- <Image
145
- src={badge.icon}
146
- alt={badge.title}
147
- width={24}
148
- height={24}
149
- />
150
- <Text
151
- as="div"
152
- className={cx(
153
- "body3",
154
- isDark ? "text-text-inverse" : "text-text"
155
- )}
156
- >
157
- {badge.title}
158
- </Text>
159
- </div>
160
- );
161
- })}
162
- </div>
163
- ) : null}
164
-
165
- {/* CTA */}
166
- <SelectPlanButton
167
- onSelect={onCtaClick}
168
- speed={speed}
169
- isSelected={isDark}
170
- renderCheckPlans={renderCheckPlans}
171
- cta={cta}
172
- iconSize={32}
173
- iconClassName="!h-8 !w-8"
174
- techType={techType}
175
- isMax={ismaxSpeed}
176
- />
177
- {/* Features */}
178
- {features.length > 0 && (
179
- <section className="flex flex-col gap-3">
180
- <Button
181
- onClick={handleToggle}
182
- className="group flex w-full items-center gap-2 text-left"
183
- >
184
- <Text
185
- as="h4"
186
- className={cx(
187
- "label3",
188
- isDark ? "text-text-inverse" : "text-text"
189
- )}
190
- >
191
- {featuresTitle}
192
- </Text>
193
- <MaterialIcon
194
- name="keyboard_arrow_down"
195
- fill={1}
196
- size={24}
197
- className={cx(
198
- "transition-transform duration-300",
199
- isExpanded && "rotate-180",
200
- isDark ? "text-text-inverse" : "text-text"
201
- )}
202
- />
203
- </Button>
204
- <div
205
- className={cx(
206
- "overflow-hidden transition-all duration-200 ease-in-out",
207
- isExpanded ? "opacity-100" : "max-h-0 opacity-0"
208
- )}
209
- >
210
- <div className="flex flex-col">
211
- <Checklist
212
- items={features}
213
- iconSize={24}
214
- iconPosition="top"
215
- iconClassName="mt-0"
216
- iconColor={isDark ? "yellow" : "green"}
217
- listContainerClassName="mt-0 space-y-0 flex flex-col gap-3"
218
- listItemClassName={`${isDark ? "text-text-inverse" : "text-text"}`}
219
- />
220
- </div>
221
- {innerBadge?.badgeIcon ? (
222
- <div className="flex items-center gap-2 text-center">
223
- <Image
224
- src={innerBadge.badgeIcon}
225
- alt="Inner Badge"
226
- width={12}
227
- height={12}
228
- />
229
- <span className="footnote text-text">
230
- {innerBadge.badgeText.includes("|") ? (
231
- <>
232
- <span className="font-bold">
233
- {innerBadge.badgeText.split("|")[0]}
234
- </span>
235
- <span className="font-normal">
236
- {innerBadge.badgeText.split("|")[1]}
237
- </span>
238
- </>
239
- ) : (
240
- <Text className="font-bold">{innerBadge.badgeText}</Text>
241
- )}
242
- </span>
243
- </div>
244
- ) : null}
245
- </div>
246
- </section>
247
- )}
248
- </div>
249
- </article>
250
- );
251
- };
252
- export default ProductCard;
1
+ "use client";
2
+
3
+ import React, { useState } from "react";
4
+ import { ProductCardProps } from "./types";
5
+
6
+ import { Button } from "@shared/components/button";
7
+ import { Checklist } from "@shared/components/checklist";
8
+ import { Image } from "@shared/components/image";
9
+ import { MaterialIcon } from "@shared/components/material-icon";
10
+ import { SelectPlanButton } from "@shared/components/select-plan-button";
11
+ import { Text } from "@shared/components/text";
12
+ import { cx } from "@shared/utils";
13
+
14
+ export const ProductCard: React.FC<ProductCardProps> = ({
15
+ planName,
16
+ planSubtext,
17
+ speed,
18
+ price,
19
+ description,
20
+ bestValue = false,
21
+ bestValueText = "best value",
22
+ giftBadge,
23
+ innerBadge,
24
+ theme = "light",
25
+ featuresTitle = "Business Ready Internet features",
26
+ features = [],
27
+ isExpanded: controlledExpanded,
28
+ onToggleExpand,
29
+ onCtaClick,
30
+ renderCheckPlans,
31
+ cta,
32
+ techType,
33
+ ismaxSpeed,
34
+ }) => {
35
+ const [internalExpanded, setInternalExpanded] = useState(false);
36
+ // Use parent state if provided, otherwise use internal state
37
+ const isExpanded =
38
+ controlledExpanded !== undefined ? controlledExpanded : internalExpanded;
39
+
40
+ const handleToggle = () => {
41
+ if (onToggleExpand) {
42
+ onToggleExpand();
43
+ } else {
44
+ setInternalExpanded(!internalExpanded);
45
+ }
46
+ };
47
+
48
+ const isDark = theme === "dark";
49
+
50
+ return (
51
+ <article
52
+ className={cx(
53
+ "relative flex h-full w-full max-w-[392px] flex-col",
54
+ !bestValue && "md:pt-[38px]"
55
+ )}
56
+ >
57
+ {/* Best Value Banner */}
58
+ {bestValue && (
59
+ <div className="label3 rounded-t-[20px] bg-bg-fill-brand-accent px-6 py-2 text-center text-text">
60
+ {bestValueText}
61
+ </div>
62
+ )}
63
+
64
+ {/* Main Card */}
65
+ <div
66
+ className={cx(
67
+ "flex flex-grow flex-col gap-5 rounded-card px-5 py-7 shadow-cardDrop",
68
+ isDark
69
+ ? "text-inverse border-2 border-border-accent bg-bg-fill-inverse"
70
+ : "bg-bg text-text",
71
+ bestValue ? "rounded-t-none border-t-0" : ""
72
+ )}
73
+ >
74
+ {/* Plan Name & Price */}
75
+ <header className="overflow-none flex items-start justify-between gap-4">
76
+ <div className="flex flex-col">
77
+ <Text
78
+ as="h3"
79
+ className={cx(
80
+ "subheading2",
81
+ isDark ? "text-text-accent-on-bg-inverse" : "text-text-brand"
82
+ )}
83
+ >
84
+ {planName}
85
+ </Text>
86
+ <Text
87
+ as="p"
88
+ className={`label3 text-wrap ${isDark ? "text-text-accent-on-bg-inverse" : "text-text-brand"}`}
89
+ >
90
+ {planSubtext}
91
+ </Text>
92
+ </div>
93
+ <div className="flex items-start">
94
+ <Text
95
+ as="span"
96
+ className={cx(
97
+ "label2 mr-1 mt-[2px]",
98
+ isDark ? "text-text-inverse" : "text-text"
99
+ )}
100
+ >
101
+ $
102
+ </Text>
103
+ <Text
104
+ as="span"
105
+ className={cx(
106
+ "subheading6",
107
+ isDark ? "text-text-inverse" : "text-text"
108
+ )}
109
+ >
110
+ {price.split(".")[0]}
111
+ </Text>
112
+ <Text
113
+ as="span"
114
+ className={cx(
115
+ "label2",
116
+ isDark ? "text-text-inverse" : "text-text"
117
+ )}
118
+ >
119
+ {price.split(".")[1]}/mo
120
+ </Text>
121
+ </div>
122
+ </header>
123
+
124
+ {/* Description */}
125
+ <section>
126
+ <Text
127
+ className={cx("body2", isDark ? "text-text-inverse" : "text-text")}
128
+ >
129
+ {description}
130
+ </Text>
131
+ </section>
132
+
133
+ {/* Gift Badge */}
134
+ {giftBadge?.length ? (
135
+ <div
136
+ className={cx(
137
+ "flex-col items-center rounded-surface-sm p-2",
138
+ isDark ? "bg-bg-surface-inverse" : "bg-bg-surface-info"
139
+ )}
140
+ >
141
+ {giftBadge?.map((badge: any) => {
142
+ return (
143
+ <div key={badge.title} className="flex items-center gap-2">
144
+ <Image
145
+ src={badge.icon}
146
+ alt={badge.title}
147
+ width={24}
148
+ height={24}
149
+ />
150
+ <Text
151
+ as="div"
152
+ className={cx(
153
+ "body3",
154
+ isDark ? "text-text-inverse" : "text-text"
155
+ )}
156
+ >
157
+ {badge.title}
158
+ </Text>
159
+ </div>
160
+ );
161
+ })}
162
+ </div>
163
+ ) : null}
164
+
165
+ {/* CTA */}
166
+ <SelectPlanButton
167
+ onSelect={onCtaClick}
168
+ speed={speed}
169
+ isSelected={isDark}
170
+ renderCheckPlans={renderCheckPlans}
171
+ cta={cta}
172
+ iconSize={32}
173
+ iconClassName="!h-8 !w-8"
174
+ techType={techType}
175
+ isMax={ismaxSpeed}
176
+ />
177
+ {/* Features */}
178
+ {features.length > 0 && (
179
+ <section className="flex flex-col gap-3">
180
+ <Button
181
+ onClick={handleToggle}
182
+ className="group flex w-full items-center gap-2 text-left"
183
+ >
184
+ <Text
185
+ as="h4"
186
+ className={cx(
187
+ "label3",
188
+ isDark ? "text-text-inverse" : "text-text"
189
+ )}
190
+ >
191
+ {featuresTitle}
192
+ </Text>
193
+ <MaterialIcon
194
+ name="keyboard_arrow_down"
195
+ fill={1}
196
+ size={24}
197
+ className={cx(
198
+ "transition-transform duration-300",
199
+ isExpanded && "rotate-180",
200
+ isDark ? "text-text-inverse" : "text-text"
201
+ )}
202
+ />
203
+ </Button>
204
+ <div
205
+ className={cx(
206
+ "overflow-hidden transition-all duration-200 ease-in-out",
207
+ isExpanded ? "opacity-100" : "max-h-0 opacity-0"
208
+ )}
209
+ >
210
+ <div className="flex flex-col">
211
+ <Checklist
212
+ items={features}
213
+ iconSize={24}
214
+ iconPosition="top"
215
+ iconClassName="mt-0"
216
+ iconColor={isDark ? "yellow" : "green"}
217
+ listContainerClassName="mt-0 space-y-0 flex flex-col gap-3"
218
+ listItemClassName={`${isDark ? "text-text-inverse" : "text-text"}`}
219
+ />
220
+ </div>
221
+ {innerBadge?.badgeIcon ? (
222
+ <div className="flex items-center gap-2 text-center">
223
+ <Image
224
+ src={innerBadge.badgeIcon}
225
+ alt="Inner Badge"
226
+ width={12}
227
+ height={12}
228
+ />
229
+ <span className="footnote text-text">
230
+ {innerBadge.badgeText.includes("|") ? (
231
+ <>
232
+ <span className="font-bold">
233
+ {innerBadge.badgeText.split("|")[0]}
234
+ </span>
235
+ <span className="font-normal">
236
+ {innerBadge.badgeText.split("|")[1]}
237
+ </span>
238
+ </>
239
+ ) : (
240
+ <Text className="font-bold">{innerBadge.badgeText}</Text>
241
+ )}
242
+ </span>
243
+ </div>
244
+ ) : null}
245
+ </div>
246
+ </section>
247
+ )}
248
+ </div>
249
+ </article>
250
+ );
251
+ };
252
+ export default ProductCard;
@@ -1,28 +1,28 @@
1
- import React, { ReactNode } from "react";
2
-
3
- import { ButtonProps } from "@shared/contentful/blocks/button/types";
4
- import { CheckPlansProps } from "@shared/types/micro-components";
5
-
6
- export interface ProductCardProps {
7
- planName: string;
8
- planSubtext?: string;
9
- speed: string;
10
- techType?: string;
11
- ismaxSpeed?: boolean;
12
- price: string;
13
- description: string;
14
- bestValue?: boolean;
15
- bestValueText?: string;
16
- giftBadge?: [{ title: ReactNode; icon: string }];
17
- innerBadge?: { badgeText: string; badgeIcon: string };
18
- theme?: "light" | "dark";
19
- featuresTitle?: string;
20
- features: ReactNode[];
21
- isExpanded?: boolean;
22
- onToggleExpand?: () => void;
23
- ctaText?: string;
24
- cta?: ButtonProps;
25
- onCtaClick: () => void;
26
- hostType: "residential" | "smb";
27
- renderCheckPlans?: (overrides?: CheckPlansProps) => React.ReactNode;
28
- }
1
+ import React, { ReactNode } from "react";
2
+
3
+ import { ButtonProps } from "@shared/contentful/blocks/button/types";
4
+ import { CheckPlansProps } from "@shared/types/micro-components";
5
+
6
+ export interface ProductCardProps {
7
+ planName: string;
8
+ planSubtext?: string;
9
+ speed: string;
10
+ techType?: string;
11
+ ismaxSpeed?: boolean;
12
+ price: string;
13
+ description: string;
14
+ bestValue?: boolean;
15
+ bestValueText?: string;
16
+ giftBadge?: [{ title: ReactNode; icon: string }];
17
+ innerBadge?: { badgeText: string; badgeIcon: string };
18
+ theme?: "light" | "dark";
19
+ featuresTitle?: string;
20
+ features: ReactNode[];
21
+ isExpanded?: boolean;
22
+ onToggleExpand?: () => void;
23
+ ctaText?: string;
24
+ cta?: ButtonProps;
25
+ onCtaClick: () => void;
26
+ hostType: "residential" | "smb";
27
+ renderCheckPlans?: (overrides?: CheckPlansProps) => React.ReactNode;
28
+ }