jfs-components 0.1.2 → 0.1.8

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 (107) hide show
  1. package/CHANGELOG.md +29 -0
  2. package/lib/commonjs/components/AmountInput/AmountInput.js +8 -5
  3. package/lib/commonjs/components/BenefitCard/BenefitCard.js +231 -0
  4. package/lib/commonjs/components/CcCard/CcCard.js +470 -0
  5. package/lib/commonjs/components/Checkbox/Checkbox.js +4 -3
  6. package/lib/commonjs/components/CheckboxItem/CheckboxItem.js +4 -3
  7. package/lib/commonjs/components/CompareTable/CompareTable.js +372 -0
  8. package/lib/commonjs/components/ComparisonBar/ComparisonBar.js +266 -0
  9. package/lib/commonjs/components/DropdownInput/DropdownInput.js +35 -3
  10. package/lib/commonjs/components/FormField/FormField.js +4 -3
  11. package/lib/commonjs/components/InputSearch/InputSearch.js +6 -4
  12. package/lib/commonjs/components/NoteInput/NoteInput.js +6 -5
  13. package/lib/commonjs/components/PdpCcCard/PdpCcCard.js +273 -0
  14. package/lib/commonjs/components/ProductMerchandisingCard/GlassFill.js +263 -0
  15. package/lib/commonjs/components/ProductMerchandisingCard/GlassFill.web.js +116 -0
  16. package/lib/commonjs/components/ProductMerchandisingCard/ProductMerchandisingCard.js +353 -0
  17. package/lib/commonjs/components/ProjectionMarker/ProjectionMarker.js +161 -0
  18. package/lib/commonjs/components/Radio/Radio.js +5 -5
  19. package/lib/commonjs/components/Slider/Slider.js +473 -0
  20. package/lib/commonjs/components/TextInput/TextInput.js +13 -8
  21. package/lib/commonjs/components/TextSegment/TextSegment.js +118 -0
  22. package/lib/commonjs/components/index.js +63 -0
  23. package/lib/commonjs/design-tokens/Coin Variables-variables-full.json +1 -1
  24. package/lib/commonjs/design-tokens/figma-modes.generated.js +38 -9
  25. package/lib/commonjs/icons/registry.js +1 -1
  26. package/lib/commonjs/utils/react-utils.js +22 -0
  27. package/lib/module/components/AmountInput/AmountInput.js +6 -4
  28. package/lib/module/components/BenefitCard/BenefitCard.js +225 -0
  29. package/lib/module/components/CcCard/CcCard.js +464 -0
  30. package/lib/module/components/Checkbox/Checkbox.js +5 -4
  31. package/lib/module/components/CheckboxItem/CheckboxItem.js +5 -4
  32. package/lib/module/components/CompareTable/CompareTable.js +367 -0
  33. package/lib/module/components/ComparisonBar/ComparisonBar.js +260 -0
  34. package/lib/module/components/DropdownInput/DropdownInput.js +36 -4
  35. package/lib/module/components/FormField/FormField.js +5 -4
  36. package/lib/module/components/InputSearch/InputSearch.js +6 -4
  37. package/lib/module/components/NoteInput/NoteInput.js +7 -6
  38. package/lib/module/components/PdpCcCard/PdpCcCard.js +267 -0
  39. package/lib/module/components/ProductMerchandisingCard/GlassFill.js +257 -0
  40. package/lib/module/components/ProductMerchandisingCard/GlassFill.web.js +111 -0
  41. package/lib/module/components/ProductMerchandisingCard/ProductMerchandisingCard.js +347 -0
  42. package/lib/module/components/ProjectionMarker/ProjectionMarker.js +156 -0
  43. package/lib/module/components/Radio/Radio.js +5 -4
  44. package/lib/module/components/Slider/Slider.js +468 -0
  45. package/lib/module/components/TextInput/TextInput.js +15 -10
  46. package/lib/module/components/TextSegment/TextSegment.js +113 -0
  47. package/lib/module/components/index.js +9 -0
  48. package/lib/module/design-tokens/Coin Variables-variables-full.json +1 -1
  49. package/lib/module/design-tokens/figma-modes.generated.js +38 -9
  50. package/lib/module/icons/registry.js +1 -1
  51. package/lib/module/utils/react-utils.js +21 -0
  52. package/lib/typescript/src/components/AmountInput/AmountInput.d.ts +3 -2
  53. package/lib/typescript/src/components/BenefitCard/BenefitCard.d.ts +93 -0
  54. package/lib/typescript/src/components/CcCard/CcCard.d.ts +137 -0
  55. package/lib/typescript/src/components/Checkbox/Checkbox.d.ts +3 -2
  56. package/lib/typescript/src/components/CheckboxItem/CheckboxItem.d.ts +2 -2
  57. package/lib/typescript/src/components/CompareTable/CompareTable.d.ts +88 -0
  58. package/lib/typescript/src/components/ComparisonBar/ComparisonBar.d.ts +118 -0
  59. package/lib/typescript/src/components/DropdownInput/DropdownInput.d.ts +20 -1
  60. package/lib/typescript/src/components/FormField/FormField.d.ts +2 -2
  61. package/lib/typescript/src/components/InputSearch/InputSearch.d.ts +23 -2
  62. package/lib/typescript/src/components/NoteInput/NoteInput.d.ts +19 -2
  63. package/lib/typescript/src/components/PdpCcCard/PdpCcCard.d.ts +84 -0
  64. package/lib/typescript/src/components/ProductMerchandisingCard/GlassFill.d.ts +56 -0
  65. package/lib/typescript/src/components/ProductMerchandisingCard/GlassFill.web.d.ts +27 -0
  66. package/lib/typescript/src/components/ProductMerchandisingCard/ProductMerchandisingCard.d.ts +81 -0
  67. package/lib/typescript/src/components/ProjectionMarker/ProjectionMarker.d.ts +82 -0
  68. package/lib/typescript/src/components/Radio/Radio.d.ts +3 -2
  69. package/lib/typescript/src/components/RadioButton/RadioButton.d.ts +2 -2
  70. package/lib/typescript/src/components/Slider/Slider.d.ts +99 -0
  71. package/lib/typescript/src/components/TextInput/TextInput.d.ts +9 -29
  72. package/lib/typescript/src/components/TextSegment/TextSegment.d.ts +100 -0
  73. package/lib/typescript/src/components/index.d.ts +10 -1
  74. package/lib/typescript/src/design-tokens/figma-modes.generated.d.ts +22 -2
  75. package/lib/typescript/src/icons/registry.d.ts +1 -1
  76. package/lib/typescript/src/utils/react-utils.d.ts +10 -0
  77. package/package.json +2 -1
  78. package/src/components/AmountInput/AmountInput.tsx +7 -5
  79. package/src/components/BenefitCard/BenefitCard.tsx +309 -0
  80. package/src/components/CcCard/CcCard.tsx +598 -0
  81. package/src/components/Checkbox/Checkbox.tsx +5 -4
  82. package/src/components/CheckboxItem/CheckboxItem.tsx +5 -4
  83. package/src/components/CompareTable/CompareTable.tsx +477 -0
  84. package/src/components/ComparisonBar/ComparisonBar.tsx +356 -0
  85. package/src/components/DropdownInput/DropdownInput.tsx +55 -3
  86. package/src/components/FormField/FormField.tsx +5 -4
  87. package/src/components/InputSearch/InputSearch.tsx +8 -5
  88. package/src/components/NoteInput/NoteInput.tsx +8 -6
  89. package/src/components/PdpCcCard/PdpCcCard.tsx +356 -0
  90. package/src/components/ProductMerchandisingCard/GlassFill.tsx +276 -0
  91. package/src/components/ProductMerchandisingCard/GlassFill.web.tsx +127 -0
  92. package/src/components/ProductMerchandisingCard/ProductMerchandisingCard.tsx +423 -0
  93. package/src/components/ProjectionMarker/ProjectionMarker.tsx +277 -0
  94. package/src/components/Radio/Radio.tsx +5 -4
  95. package/src/components/Slider/Slider.tsx +628 -0
  96. package/src/components/TextInput/TextInput.tsx +15 -11
  97. package/src/components/TextSegment/TextSegment.tsx +166 -0
  98. package/src/components/index.ts +10 -1
  99. package/src/design-tokens/Coin Variables-variables-full.json +1 -1
  100. package/src/design-tokens/figma-modes.generated.ts +38 -9
  101. package/src/icons/registry.ts +1 -1
  102. package/src/utils/react-utils.ts +23 -0
  103. package/lib/typescript/scripts/extract-component-tokens.d.ts +0 -9
  104. package/lib/typescript/scripts/generate-component-docs.d.ts +0 -9
  105. package/lib/typescript/scripts/generate-icon-registry.d.ts +0 -3
  106. package/lib/typescript/scripts/generate-mode-types.d.ts +0 -2
  107. package/lib/typescript/scripts/retype-modes.d.cts +0 -2
@@ -0,0 +1,470 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _react = _interopRequireWildcard(require("react"));
8
+ var _reactNative = require("react-native");
9
+ var _figmaVariablesResolver = require("../../design-tokens/figma-variables-resolver");
10
+ var _reactUtils = require("../../utils/react-utils");
11
+ var _Badge = _interopRequireDefault(require("../Badge/Badge"));
12
+ var _Avatar = _interopRequireDefault(require("../Avatar/Avatar"));
13
+ var _Image = _interopRequireDefault(require("../Image/Image"));
14
+ var _Icon = _interopRequireDefault(require("../Icon/Icon"));
15
+ var _Title = _interopRequireDefault(require("../Title/Title"));
16
+ var _ListItem = _interopRequireDefault(require("../ListItem/ListItem"));
17
+ var _TextSegment = _interopRequireDefault(require("../TextSegment/TextSegment"));
18
+ var _Button = _interopRequireDefault(require("../Button/Button"));
19
+ var _jsxRuntime = require("react/jsx-runtime");
20
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
21
+ function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function (e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != typeof e && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (const t in e) "default" !== t && {}.hasOwnProperty.call(e, t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, t)) && (i.get || i.set) ? o(f, t, i) : f[t] = e[t]); return f; })(e, t); }
22
+ /**
23
+ * A single coloured chip rendered in the card header. Each badge resolves its
24
+ * own design tokens through `modes`, so different appearances (brand, tonal,
25
+ * etc.) can sit side-by-side in the same group.
26
+ */
27
+
28
+ /**
29
+ * A single benefit row in the card's list, rendered through the shared
30
+ * {@link ListItem} primitive (leading icon + title).
31
+ */
32
+
33
+ /**
34
+ * CcCard — Figma node 5434:1992 ("Cc Card").
35
+ *
36
+ * A white, rounded credit-card product card composed entirely from the shared
37
+ * design-system primitives so it stays in sync with the rest of the library:
38
+ *
39
+ * - **Header** — two {@link Badge} groups (a leading group + a right-aligned
40
+ * trailing group), e.g. `Pre-qualified` / `Lifetime free`.
41
+ * - **Media** — a product `Image` + a {@link Title} (title + subtitle resolved
42
+ * through the `context7: Card` mode → 14px/12px).
43
+ * - **List** — a column of benefit {@link ListItem}s (`List Item Style:
44
+ * Minimal`) with leading icons.
45
+ * - **Nudge** — an inline upsell row: an {@link Avatar} + a {@link TextSegment}.
46
+ * - **Footer** — a headline + description + subtitle text block alongside a
47
+ * small {@link Button} (`Button / Size: S`, `AppearanceBrand: Secondary`).
48
+ *
49
+ * The {@link CcCardProps.compact} variant collapses the card to just the
50
+ * leading avatar slot + the footer text block (no CTA button). All defaults can
51
+ * be overridden via `modes`.
52
+ */
53
+ function CcCard({
54
+ compact = false,
55
+ showHeader = true,
56
+ badges = DEFAULT_BADGES,
57
+ trailingBadges,
58
+ header,
59
+ imageSource,
60
+ imageWidth = 88,
61
+ imageHeight = 54,
62
+ title = 'Title',
63
+ subtitle = 'Subtitle',
64
+ media,
65
+ items = DEFAULT_ITEMS,
66
+ showNudge = true,
67
+ nudgeSegments = DEFAULT_NUDGE_SEGMENTS,
68
+ nudgeAvatarSource,
69
+ nudge,
70
+ headline = 'Headline',
71
+ description = 'Description',
72
+ footerSubtitle = 'Subtitle',
73
+ footer,
74
+ buttonLabel = 'Button',
75
+ onButtonPress,
76
+ button,
77
+ showButton = true,
78
+ onPress,
79
+ width = 343,
80
+ modes = _reactUtils.EMPTY_MODES,
81
+ style,
82
+ accessibilityLabel
83
+ }) {
84
+ const tokens = (0, _react.useMemo)(() => resolveTokens(modes), [modes]);
85
+
86
+ // The product title uses the card-sized type ramp (14px bold + 12px
87
+ // subtitle) via the `context7: Card` mode. Consumer modes still win.
88
+ const titleModes = (0, _react.useMemo)(() => ({
89
+ context7: 'Card',
90
+ ...modes
91
+ }), [modes]);
92
+ // List rows use the minimal style (12px / regular, no padding).
93
+ const listItemModes = (0, _react.useMemo)(() => ({
94
+ 'List Item Style': 'Minimal',
95
+ ...modes
96
+ }), [modes]);
97
+ // The nudge avatar is the smallest size token.
98
+ const nudgeAvatarModes = (0, _react.useMemo)(() => ({
99
+ 'Avatar Size': 'XS',
100
+ ...modes
101
+ }), [modes]);
102
+ // The CTA is the small, brand-secondary (lilac→purple) button.
103
+ const ctaModes = (0, _react.useMemo)(() => ({
104
+ AppearanceBrand: 'Secondary',
105
+ 'Button / Size': 'S',
106
+ ...modes
107
+ }), [modes]);
108
+
109
+ // ---- Header -----------------------------------------------------------
110
+ const headerNode = header ? (0, _reactUtils.cloneChildrenWithModes)(header, modes) : showHeader && (badges?.length || trailingBadges?.length) ? /*#__PURE__*/(0, _jsxRuntime.jsxs)(_reactNative.View, {
111
+ style: styles.header,
112
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(BadgeGroup, {
113
+ badges: badges,
114
+ modes: modes
115
+ }), trailingBadges?.length ? /*#__PURE__*/(0, _jsxRuntime.jsx)(BadgeGroup, {
116
+ badges: trailingBadges,
117
+ modes: modes
118
+ }) : null]
119
+ }) : null;
120
+
121
+ // ---- Media slot (shared by both compact + non-compact) ----------------
122
+ // Whatever lives in the slot is rendered identically in both variants; only
123
+ // the surrounding sections differ. The default is an Image driven by
124
+ // `imageSource`, but a consumer can pass an Avatar (or any node) via `media`.
125
+ const mediaSlotContent = media ? (0, _reactUtils.cloneChildrenWithModes)(media, modes) : /*#__PURE__*/(0, _jsxRuntime.jsx)(_Image.default, {
126
+ imageSource: imageSource,
127
+ width: imageWidth,
128
+ height: imageHeight,
129
+ borderRadius: tokens.imageRadius,
130
+ resizeMode: "cover",
131
+ accessibilityElementsHidden: true,
132
+ importantForAccessibility: "no"
133
+ });
134
+
135
+ // Non-compact lays the slot out in a row next to the product Title.
136
+ const mediaRow = /*#__PURE__*/(0, _jsxRuntime.jsxs)(_reactNative.View, {
137
+ style: styles.mediaSlot,
138
+ children: [mediaSlotContent, /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.View, {
139
+ style: styles.titleWrap,
140
+ children: /*#__PURE__*/(0, _jsxRuntime.jsx)(_Title.default, {
141
+ title: title,
142
+ subtitle: subtitle,
143
+ textAlign: "Left",
144
+ modes: titleModes,
145
+ style: styles.titleInner
146
+ })
147
+ })]
148
+ });
149
+
150
+ // ---- List -------------------------------------------------------------
151
+ const listNode = items.length > 0 ? /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.View, {
152
+ style: tokens.listGroup,
153
+ children: items.map((item, index) => {
154
+ const leading = item.leading ?? (item.icon !== null ? /*#__PURE__*/(0, _jsxRuntime.jsx)(_Icon.default, {
155
+ iconName: item.icon ?? 'ic_card',
156
+ color: item.iconColor,
157
+ modes: listItemModes
158
+ }) : null);
159
+ return /*#__PURE__*/(0, _jsxRuntime.jsx)(_ListItem.default, {
160
+ layout: "Horizontal",
161
+ navArrow: false,
162
+ showSupportText: false,
163
+ title: item.title,
164
+ leading: leading,
165
+ onPress: item.onPress,
166
+ modes: listItemModes
167
+ }, index);
168
+ })
169
+ }) : null;
170
+
171
+ // ---- Nudge ------------------------------------------------------------
172
+ const nudgeNode = nudge ? (0, _reactUtils.cloneChildrenWithModes)(nudge, modes) : showNudge ? /*#__PURE__*/(0, _jsxRuntime.jsxs)(_reactNative.View, {
173
+ style: tokens.nudge,
174
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(_Avatar.default, {
175
+ imageSource: nudgeAvatarSource,
176
+ modes: nudgeAvatarModes
177
+ }), /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.View, {
178
+ style: styles.nudgeContent,
179
+ children: /*#__PURE__*/(0, _jsxRuntime.jsx)(_TextSegment.default, {
180
+ segments: nudgeSegments,
181
+ modes: modes
182
+ })
183
+ })]
184
+ }) : null;
185
+
186
+ // ---- Footer -----------------------------------------------------------
187
+ const ctaNode = button ? (0, _reactUtils.cloneChildrenWithModes)(button, ctaModes) : /*#__PURE__*/(0, _jsxRuntime.jsx)(_Button.default, {
188
+ label: buttonLabel,
189
+ modes: ctaModes,
190
+ onPress: onButtonPress
191
+ });
192
+ const footerTextBlock = footer ? (0, _reactUtils.cloneChildrenWithModes)(footer, modes) : /*#__PURE__*/(0, _jsxRuntime.jsxs)(_reactNative.View, {
193
+ style: tokens.footerTextWrap,
194
+ children: [headline != null ? /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.Text, {
195
+ style: tokens.headline,
196
+ children: headline
197
+ }) : null, /*#__PURE__*/(0, _jsxRuntime.jsxs)(_reactNative.View, {
198
+ style: compact ? tokens.titleWrapCompact : tokens.titleWrapDefault,
199
+ children: [description != null ? /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.Text, {
200
+ style: tokens.footerTitle,
201
+ children: description
202
+ }) : null, footerSubtitle != null ? /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.Text, {
203
+ style: tokens.footerSubtitle,
204
+ children: footerSubtitle
205
+ }) : null]
206
+ })]
207
+ });
208
+ const footerNode = /*#__PURE__*/(0, _jsxRuntime.jsxs)(_reactNative.View, {
209
+ style: compact ? tokens.footerRowCompact : tokens.footerRowDefault,
210
+ children: [footerTextBlock, !compact && showButton ? ctaNode : null]
211
+ });
212
+
213
+ // ---- Compose ----------------------------------------------------------
214
+ const content = compact ? /*#__PURE__*/(0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
215
+ children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.View, {
216
+ style: styles.compactSlot,
217
+ children: mediaSlotContent
218
+ }), footerNode]
219
+ }) : /*#__PURE__*/(0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
220
+ children: [headerNode, mediaRow, listNode, nudgeNode, footerNode]
221
+ });
222
+ const containerStyle = (0, _react.useMemo)(() => ({
223
+ ...tokens.container,
224
+ width
225
+ }), [tokens.container, width]);
226
+ if (onPress) {
227
+ return /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.Pressable, {
228
+ style: ({
229
+ pressed
230
+ }) => [containerStyle, pressed ? styles.pressed : null, style],
231
+ accessibilityRole: "button",
232
+ accessibilityLabel: accessibilityLabel ?? title,
233
+ onPress: onPress,
234
+ children: content
235
+ });
236
+ }
237
+ return /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.View, {
238
+ style: [containerStyle, style],
239
+ accessibilityLabel: accessibilityLabel,
240
+ children: content
241
+ });
242
+ }
243
+
244
+ // ---------------------------------------------------------------------------
245
+ // BadgeGroup — internal wrapping row of badges
246
+ // ---------------------------------------------------------------------------
247
+
248
+ function BadgeGroup({
249
+ badges,
250
+ modes
251
+ }) {
252
+ return /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.View, {
253
+ style: styles.badgeGroup,
254
+ children: badges.map((badge, index) => {
255
+ const badgeModes = badge.modes ? {
256
+ ...modes,
257
+ ...badge.modes
258
+ } : modes;
259
+ return /*#__PURE__*/(0, _jsxRuntime.jsx)(_Badge.default, {
260
+ label: badge.label,
261
+ modes: badgeModes,
262
+ style: badge.color ? {
263
+ backgroundColor: badge.color
264
+ } : undefined,
265
+ labelStyle: badge.labelColor ? {
266
+ color: badge.labelColor
267
+ } : undefined
268
+ }, index);
269
+ })
270
+ });
271
+ }
272
+
273
+ // ---------------------------------------------------------------------------
274
+ // Tokens / static styles
275
+ // ---------------------------------------------------------------------------
276
+
277
+ function asNum(raw, fallback) {
278
+ const n = typeof raw === 'number' ? raw : parseFloat(raw);
279
+ return Number.isFinite(n) ? n : fallback;
280
+ }
281
+ function asStr(raw, fallback) {
282
+ return raw != null ? String(raw) : fallback;
283
+ }
284
+ function resolveTokens(modes) {
285
+ // NOTE: token names are passed as string literals DIRECTLY to
286
+ // getVariableByName so the `extract-component-tokens` script can statically
287
+ // collect them for the generated docs. Do not refactor these into a helper
288
+ // that receives the name as a variable.
289
+ const background = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/bg', modes), '#ffffff');
290
+ const borderColor = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/border/color', modes), '#e3e4e4');
291
+ const borderWidth = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/border/size', modes), 1);
292
+ const radius = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/radius', modes), 12);
293
+ const paddingHorizontal = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/padding/horizontal', modes), 12);
294
+ const paddingVertical = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/padding/vertical', modes), 12);
295
+ const gap = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/padding/gap', modes), 8);
296
+ const imageRadius = asNum((0, _figmaVariablesResolver.getVariableByName)('image/radius', modes), 0);
297
+ const listGroupGap = asNum((0, _figmaVariablesResolver.getVariableByName)('listGroup/gap', modes), 4);
298
+ const nudgeGap = asNum((0, _figmaVariablesResolver.getVariableByName)('nudge/gap', modes), 6);
299
+ const footerGap = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/footer/gap', modes), 8);
300
+ const textWrapGap = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/footer/textWrap/gap', modes), 4);
301
+ const titleWrapGap = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/footer/titleWrap/gap', modes), 8);
302
+ const headlineColor = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/headline/color', modes), '#707275');
303
+ const headlineSize = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/headline/fontSize', modes), 10);
304
+ const headlineFamily = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/headline/fontFamily', modes), 'JioType Var');
305
+ const headlineLineHeight = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/headline/lineHeight', modes), 13);
306
+ const headlineWeight = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/headline/fontWeight', modes), '500');
307
+ const titleColor = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/title/color', modes), '#081007');
308
+ const titleSize = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/title/fontSize', modes), 14);
309
+ const titleFamily = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/title/fontFamily', modes), 'JioType Var');
310
+ const titleLineHeight = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/title/lineHeight', modes), 15);
311
+ const titleWeight = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/title/fontWeight', modes), '700');
312
+ const subtitleColor = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/subtitle/color', modes), '#707275');
313
+ const subtitleSize = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/subtitle/fontSize', modes), 10);
314
+ const subtitleFamily = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/subtitle/fontFamily', modes), 'JioType Var');
315
+ const subtitleLineHeight = asNum((0, _figmaVariablesResolver.getVariableByName)('ccCard/subtitle/lineHeight', modes), 13);
316
+ const subtitleWeight = asStr((0, _figmaVariablesResolver.getVariableByName)('ccCard/subtitle/fontWeight', modes), '500');
317
+ return {
318
+ container: {
319
+ backgroundColor: background,
320
+ borderColor,
321
+ borderWidth,
322
+ borderRadius: radius,
323
+ paddingHorizontal,
324
+ paddingVertical,
325
+ gap,
326
+ flexDirection: 'column',
327
+ alignItems: 'flex-start',
328
+ overflow: 'hidden'
329
+ },
330
+ imageRadius,
331
+ listGroup: {
332
+ alignSelf: 'stretch',
333
+ gap: listGroupGap
334
+ },
335
+ nudge: {
336
+ alignSelf: 'stretch',
337
+ flexDirection: 'row',
338
+ alignItems: 'center',
339
+ gap: nudgeGap
340
+ },
341
+ footerRowDefault: {
342
+ alignSelf: 'stretch',
343
+ flexDirection: 'row',
344
+ alignItems: 'flex-end',
345
+ gap: footerGap
346
+ },
347
+ footerRowCompact: {
348
+ alignSelf: 'stretch',
349
+ flexDirection: 'row',
350
+ alignItems: 'center'
351
+ },
352
+ footerTextWrap: {
353
+ flex: 1,
354
+ minWidth: 1,
355
+ alignItems: 'flex-start',
356
+ gap: textWrapGap
357
+ },
358
+ titleWrapDefault: {
359
+ flexDirection: 'row',
360
+ alignItems: 'flex-end',
361
+ gap: textWrapGap
362
+ },
363
+ titleWrapCompact: {
364
+ flexDirection: 'row',
365
+ alignItems: 'flex-end',
366
+ gap: titleWrapGap
367
+ },
368
+ headline: {
369
+ color: headlineColor,
370
+ fontSize: headlineSize,
371
+ fontFamily: headlineFamily,
372
+ fontWeight: headlineWeight,
373
+ lineHeight: Math.max(headlineLineHeight, Math.ceil(headlineSize * 1.2)),
374
+ includeFontPadding: false
375
+ },
376
+ footerTitle: {
377
+ color: titleColor,
378
+ fontSize: titleSize,
379
+ fontFamily: titleFamily,
380
+ fontWeight: titleWeight,
381
+ lineHeight: Math.max(titleLineHeight, Math.ceil(titleSize * 1.2)),
382
+ includeFontPadding: false
383
+ },
384
+ footerSubtitle: {
385
+ color: subtitleColor,
386
+ fontSize: subtitleSize,
387
+ fontFamily: subtitleFamily,
388
+ fontWeight: subtitleWeight,
389
+ lineHeight: Math.max(subtitleLineHeight, Math.ceil(subtitleSize * 1.2)),
390
+ includeFontPadding: false
391
+ }
392
+ };
393
+ }
394
+ const DEFAULT_BADGES = [{
395
+ label: 'Pre-qualified',
396
+ modes: {
397
+ AppearanceBrand: 'Tertiary'
398
+ },
399
+ labelColor: '#ffffff'
400
+ }, {
401
+ label: 'Lifetime free',
402
+ modes: {
403
+ AppearanceBrand: 'Tertiary'
404
+ },
405
+ labelColor: '#ffffff'
406
+ }];
407
+ const DEFAULT_ITEMS = [{
408
+ icon: 'ic_card',
409
+ title: '4 domestic + 2 intl. lounge access yearly'
410
+ }, {
411
+ icon: 'ic_card',
412
+ title: '5% cashback on dining & travel'
413
+ }, {
414
+ icon: 'ic_card',
415
+ title: 'No annual fee for the first year'
416
+ }];
417
+ const DEFAULT_NUDGE_SEGMENTS = [{
418
+ text: 'Upsell message '
419
+ }, {
420
+ text: 'JioFinance+',
421
+ modes: {
422
+ 'Text Appearance': 'Primary'
423
+ }
424
+ }];
425
+ const styles = _reactNative.StyleSheet.create({
426
+ header: {
427
+ alignSelf: 'stretch',
428
+ flexDirection: 'row',
429
+ alignItems: 'flex-start',
430
+ justifyContent: 'space-between',
431
+ gap: 8
432
+ },
433
+ badgeGroup: {
434
+ flexDirection: 'row',
435
+ flexWrap: 'wrap',
436
+ alignItems: 'center',
437
+ gap: 4,
438
+ flexShrink: 1
439
+ },
440
+ mediaSlot: {
441
+ alignSelf: 'stretch',
442
+ flexDirection: 'row',
443
+ alignItems: 'center',
444
+ gap: 8
445
+ },
446
+ titleWrap: {
447
+ flex: 1,
448
+ minWidth: 1
449
+ },
450
+ // The shared Title falls back to 16/8 padding when its tokens resolve to 0
451
+ // (0 is falsy). Zero it out so the only horizontal gap to the media slot is
452
+ // the mediaSlot's hard-coded 8px.
453
+ titleInner: {
454
+ paddingHorizontal: 0,
455
+ paddingVertical: 0
456
+ },
457
+ // Compact media wrapper hugs its child so the slot sizes to the avatar /
458
+ // image's intrinsic dimensions instead of stretching to the card width.
459
+ compactSlot: {
460
+ alignSelf: 'flex-start'
461
+ },
462
+ nudgeContent: {
463
+ flex: 1,
464
+ minWidth: 1
465
+ },
466
+ pressed: {
467
+ opacity: 0.92
468
+ }
469
+ });
470
+ var _default = exports.default = CcCard;
@@ -83,7 +83,7 @@ function invisibleTouchHitSlop(checkboxSize) {
83
83
  * @component
84
84
  * @param {CheckboxProps} props
85
85
  */
86
- function Checkbox({
86
+ const Checkbox = /*#__PURE__*/(0, _react.forwardRef)(function Checkbox({
87
87
  checked: controlledChecked,
88
88
  defaultChecked = false,
89
89
  onValueChange,
@@ -91,7 +91,7 @@ function Checkbox({
91
91
  modes = _reactUtils.EMPTY_MODES,
92
92
  style,
93
93
  accessibilityLabel
94
- }) {
94
+ }, ref) {
95
95
  const isControlled = controlledChecked !== undefined;
96
96
  const [internalChecked, setInternalChecked] = (0, _react.useState)(defaultChecked);
97
97
  const isChecked = isControlled ? controlledChecked : internalChecked;
@@ -199,6 +199,7 @@ function Checkbox({
199
199
  const markColor = disabled && isChecked ? disabledActiveMark : selectedMarkColor;
200
200
  const hitSlop = invisibleTouchHitSlop(size);
201
201
  return /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.Pressable, {
202
+ ref: ref,
202
203
  style: [touchTargetStyle, style],
203
204
  hitSlop: hitSlop,
204
205
  onPress: handlePress,
@@ -226,7 +227,7 @@ function Checkbox({
226
227
  })
227
228
  })
228
229
  });
229
- }
230
+ });
230
231
  function boxShadow(value) {
231
232
  if (_reactNative.Platform.OS === 'web') {
232
233
  return {
@@ -39,7 +39,7 @@ function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r
39
39
  * />
40
40
  * ```
41
41
  */
42
- function CheckboxItem({
42
+ const CheckboxItem = /*#__PURE__*/(0, _react.forwardRef)(function CheckboxItem({
43
43
  checked: controlledChecked,
44
44
  defaultChecked = false,
45
45
  onValueChange,
@@ -52,7 +52,7 @@ function CheckboxItem({
52
52
  style,
53
53
  labelStyle,
54
54
  accessibilityLabel
55
- }) {
55
+ }, ref) {
56
56
  const isTrailing = control === 'trailing';
57
57
  const isControlled = controlledChecked !== undefined;
58
58
  const [internalChecked, setInternalChecked] = (0, _react.useState)(defaultChecked);
@@ -121,6 +121,7 @@ function CheckboxItem({
121
121
  children: (0, _reactUtils.cloneChildrenWithModes)(endSlot, modes)
122
122
  }) : null;
123
123
  return /*#__PURE__*/(0, _jsxRuntime.jsx)(_reactNative.Pressable, {
124
+ ref: ref,
124
125
  style: [containerStyle, style],
125
126
  onPress: handleToggle,
126
127
  disabled: disabled,
@@ -136,5 +137,5 @@ function CheckboxItem({
136
137
  children: [checkboxNode, labelNode, endSlotNode]
137
138
  })
138
139
  });
139
- }
140
+ });
140
141
  var _default = exports.default = CheckboxItem;