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.
- package/CHANGELOG.md +29 -0
- package/lib/commonjs/components/AmountInput/AmountInput.js +8 -5
- package/lib/commonjs/components/BenefitCard/BenefitCard.js +231 -0
- package/lib/commonjs/components/CcCard/CcCard.js +470 -0
- package/lib/commonjs/components/Checkbox/Checkbox.js +4 -3
- package/lib/commonjs/components/CheckboxItem/CheckboxItem.js +4 -3
- package/lib/commonjs/components/CompareTable/CompareTable.js +372 -0
- package/lib/commonjs/components/ComparisonBar/ComparisonBar.js +266 -0
- package/lib/commonjs/components/DropdownInput/DropdownInput.js +35 -3
- package/lib/commonjs/components/FormField/FormField.js +4 -3
- package/lib/commonjs/components/InputSearch/InputSearch.js +6 -4
- package/lib/commonjs/components/NoteInput/NoteInput.js +6 -5
- package/lib/commonjs/components/PdpCcCard/PdpCcCard.js +273 -0
- package/lib/commonjs/components/ProductMerchandisingCard/GlassFill.js +263 -0
- package/lib/commonjs/components/ProductMerchandisingCard/GlassFill.web.js +116 -0
- package/lib/commonjs/components/ProductMerchandisingCard/ProductMerchandisingCard.js +353 -0
- package/lib/commonjs/components/ProjectionMarker/ProjectionMarker.js +161 -0
- package/lib/commonjs/components/Radio/Radio.js +5 -5
- package/lib/commonjs/components/Slider/Slider.js +473 -0
- package/lib/commonjs/components/TextInput/TextInput.js +13 -8
- package/lib/commonjs/components/TextSegment/TextSegment.js +118 -0
- package/lib/commonjs/components/index.js +63 -0
- package/lib/commonjs/design-tokens/Coin Variables-variables-full.json +1 -1
- package/lib/commonjs/design-tokens/figma-modes.generated.js +38 -9
- package/lib/commonjs/icons/registry.js +1 -1
- package/lib/commonjs/utils/react-utils.js +22 -0
- package/lib/module/components/AmountInput/AmountInput.js +6 -4
- package/lib/module/components/BenefitCard/BenefitCard.js +225 -0
- package/lib/module/components/CcCard/CcCard.js +464 -0
- package/lib/module/components/Checkbox/Checkbox.js +5 -4
- package/lib/module/components/CheckboxItem/CheckboxItem.js +5 -4
- package/lib/module/components/CompareTable/CompareTable.js +367 -0
- package/lib/module/components/ComparisonBar/ComparisonBar.js +260 -0
- package/lib/module/components/DropdownInput/DropdownInput.js +36 -4
- package/lib/module/components/FormField/FormField.js +5 -4
- package/lib/module/components/InputSearch/InputSearch.js +6 -4
- package/lib/module/components/NoteInput/NoteInput.js +7 -6
- package/lib/module/components/PdpCcCard/PdpCcCard.js +267 -0
- package/lib/module/components/ProductMerchandisingCard/GlassFill.js +257 -0
- package/lib/module/components/ProductMerchandisingCard/GlassFill.web.js +111 -0
- package/lib/module/components/ProductMerchandisingCard/ProductMerchandisingCard.js +347 -0
- package/lib/module/components/ProjectionMarker/ProjectionMarker.js +156 -0
- package/lib/module/components/Radio/Radio.js +5 -4
- package/lib/module/components/Slider/Slider.js +468 -0
- package/lib/module/components/TextInput/TextInput.js +15 -10
- package/lib/module/components/TextSegment/TextSegment.js +113 -0
- package/lib/module/components/index.js +9 -0
- package/lib/module/design-tokens/Coin Variables-variables-full.json +1 -1
- package/lib/module/design-tokens/figma-modes.generated.js +38 -9
- package/lib/module/icons/registry.js +1 -1
- package/lib/module/utils/react-utils.js +21 -0
- package/lib/typescript/src/components/AmountInput/AmountInput.d.ts +3 -2
- package/lib/typescript/src/components/BenefitCard/BenefitCard.d.ts +93 -0
- package/lib/typescript/src/components/CcCard/CcCard.d.ts +137 -0
- package/lib/typescript/src/components/Checkbox/Checkbox.d.ts +3 -2
- package/lib/typescript/src/components/CheckboxItem/CheckboxItem.d.ts +2 -2
- package/lib/typescript/src/components/CompareTable/CompareTable.d.ts +88 -0
- package/lib/typescript/src/components/ComparisonBar/ComparisonBar.d.ts +118 -0
- package/lib/typescript/src/components/DropdownInput/DropdownInput.d.ts +20 -1
- package/lib/typescript/src/components/FormField/FormField.d.ts +2 -2
- package/lib/typescript/src/components/InputSearch/InputSearch.d.ts +23 -2
- package/lib/typescript/src/components/NoteInput/NoteInput.d.ts +19 -2
- package/lib/typescript/src/components/PdpCcCard/PdpCcCard.d.ts +84 -0
- package/lib/typescript/src/components/ProductMerchandisingCard/GlassFill.d.ts +56 -0
- package/lib/typescript/src/components/ProductMerchandisingCard/GlassFill.web.d.ts +27 -0
- package/lib/typescript/src/components/ProductMerchandisingCard/ProductMerchandisingCard.d.ts +81 -0
- package/lib/typescript/src/components/ProjectionMarker/ProjectionMarker.d.ts +82 -0
- package/lib/typescript/src/components/Radio/Radio.d.ts +3 -2
- package/lib/typescript/src/components/RadioButton/RadioButton.d.ts +2 -2
- package/lib/typescript/src/components/Slider/Slider.d.ts +99 -0
- package/lib/typescript/src/components/TextInput/TextInput.d.ts +9 -29
- package/lib/typescript/src/components/TextSegment/TextSegment.d.ts +100 -0
- package/lib/typescript/src/components/index.d.ts +10 -1
- package/lib/typescript/src/design-tokens/figma-modes.generated.d.ts +22 -2
- package/lib/typescript/src/icons/registry.d.ts +1 -1
- package/lib/typescript/src/utils/react-utils.d.ts +10 -0
- package/package.json +2 -1
- package/src/components/AmountInput/AmountInput.tsx +7 -5
- package/src/components/BenefitCard/BenefitCard.tsx +309 -0
- package/src/components/CcCard/CcCard.tsx +598 -0
- package/src/components/Checkbox/Checkbox.tsx +5 -4
- package/src/components/CheckboxItem/CheckboxItem.tsx +5 -4
- package/src/components/CompareTable/CompareTable.tsx +477 -0
- package/src/components/ComparisonBar/ComparisonBar.tsx +356 -0
- package/src/components/DropdownInput/DropdownInput.tsx +55 -3
- package/src/components/FormField/FormField.tsx +5 -4
- package/src/components/InputSearch/InputSearch.tsx +8 -5
- package/src/components/NoteInput/NoteInput.tsx +8 -6
- package/src/components/PdpCcCard/PdpCcCard.tsx +356 -0
- package/src/components/ProductMerchandisingCard/GlassFill.tsx +276 -0
- package/src/components/ProductMerchandisingCard/GlassFill.web.tsx +127 -0
- package/src/components/ProductMerchandisingCard/ProductMerchandisingCard.tsx +423 -0
- package/src/components/ProjectionMarker/ProjectionMarker.tsx +277 -0
- package/src/components/Radio/Radio.tsx +5 -4
- package/src/components/Slider/Slider.tsx +628 -0
- package/src/components/TextInput/TextInput.tsx +15 -11
- package/src/components/TextSegment/TextSegment.tsx +166 -0
- package/src/components/index.ts +10 -1
- package/src/design-tokens/Coin Variables-variables-full.json +1 -1
- package/src/design-tokens/figma-modes.generated.ts +38 -9
- package/src/icons/registry.ts +1 -1
- package/src/utils/react-utils.ts +23 -0
- package/lib/typescript/scripts/extract-component-tokens.d.ts +0 -9
- package/lib/typescript/scripts/generate-component-docs.d.ts +0 -9
- package/lib/typescript/scripts/generate-icon-registry.d.ts +0 -3
- package/lib/typescript/scripts/generate-mode-types.d.ts +0 -2
- package/lib/typescript/scripts/retype-modes.d.cts +0 -2
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import React from 'react'
|
|
2
|
+
import { View, StyleSheet, type ViewStyle, type StyleProp } from 'react-native'
|
|
3
|
+
|
|
4
|
+
export type GlassTint = 'dark' | 'light'
|
|
5
|
+
|
|
6
|
+
export interface GlassFillProps {
|
|
7
|
+
tint?: GlassTint
|
|
8
|
+
intensity?: number
|
|
9
|
+
overlayColor?: string
|
|
10
|
+
/**
|
|
11
|
+
* Render a *progressive* (variable) blur: fully clear at the top, easing
|
|
12
|
+
* into a soft blur toward the bottom. On web this mirrors the native
|
|
13
|
+
* technique — two `backdrop-filter` layers, each clipped with an eased
|
|
14
|
+
* multi-stop `mask-image` linear-gradient so the effective blur swells
|
|
15
|
+
* smoothly toward the bottom.
|
|
16
|
+
*/
|
|
17
|
+
progressive?: boolean
|
|
18
|
+
style?: StyleProp<ViewStyle>
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
const DEFAULT_FALLBACK_DARK = '#1414174a'
|
|
22
|
+
const DEFAULT_FALLBACK_LIGHT = '#ffffff66'
|
|
23
|
+
|
|
24
|
+
// Mirror of the native ramp (see GlassFill.tsx). Each layer is a `mask-image`
|
|
25
|
+
// gradient whose alpha stops (0 = top, 1 = bottom) ease the blur in; `amount`
|
|
26
|
+
// is the layer's share of the max blur. Two gently-overlapping layers (faint
|
|
27
|
+
// base + bottom accent) keep the surface subtle and glass-like rather than a
|
|
28
|
+
// heavy frosted block.
|
|
29
|
+
interface WebLayer {
|
|
30
|
+
amount: number
|
|
31
|
+
stops: { offset: number; opacity: number }[]
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const PROGRESSIVE_LAYERS: WebLayer[] = [
|
|
35
|
+
{
|
|
36
|
+
amount: 0.65,
|
|
37
|
+
stops: [
|
|
38
|
+
{ offset: 0.0, opacity: 0 },
|
|
39
|
+
{ offset: 0.08, opacity: 0.12 },
|
|
40
|
+
{ offset: 0.35, opacity: 0.4 },
|
|
41
|
+
{ offset: 0.65, opacity: 0.8 },
|
|
42
|
+
{ offset: 1.0, opacity: 1 },
|
|
43
|
+
],
|
|
44
|
+
},
|
|
45
|
+
{
|
|
46
|
+
amount: 1.0,
|
|
47
|
+
stops: [
|
|
48
|
+
{ offset: 0.0, opacity: 0 },
|
|
49
|
+
{ offset: 0.3, opacity: 0.15 },
|
|
50
|
+
{ offset: 0.65, opacity: 0.65 },
|
|
51
|
+
{ offset: 1.0, opacity: 1 },
|
|
52
|
+
],
|
|
53
|
+
},
|
|
54
|
+
]
|
|
55
|
+
|
|
56
|
+
function toGradient(stops: WebLayer['stops']): string {
|
|
57
|
+
const parts = stops
|
|
58
|
+
.map((s) => `rgba(0,0,0,${s.opacity}) ${Math.round(s.offset * 100)}%`)
|
|
59
|
+
.join(', ')
|
|
60
|
+
return `linear-gradient(to bottom, ${parts})`
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Web counterpart of `GlassFill`.
|
|
65
|
+
*
|
|
66
|
+
* `@react-native-community/blur` ships no web implementation, so on web we
|
|
67
|
+
* render a translucent `View` with `backdrop-filter: blur()`. Native bundles
|
|
68
|
+
* pick up `GlassFill.tsx` instead via Metro's platform resolver; the web
|
|
69
|
+
* bundle picks up this file, so the native-only module is never imported here.
|
|
70
|
+
*/
|
|
71
|
+
function GlassFill({
|
|
72
|
+
tint = 'dark',
|
|
73
|
+
intensity = 50,
|
|
74
|
+
overlayColor,
|
|
75
|
+
progressive = false,
|
|
76
|
+
style,
|
|
77
|
+
}: GlassFillProps) {
|
|
78
|
+
// intensity 0-100 -> ~0-32px CSS blur, kept in parity with the native scale.
|
|
79
|
+
const blurPx = Math.max(0, Math.min(32, Math.round(intensity * 0.32)))
|
|
80
|
+
const tintColor = overlayColor ?? (tint === 'light' ? DEFAULT_FALLBACK_LIGHT : DEFAULT_FALLBACK_DARK)
|
|
81
|
+
|
|
82
|
+
if (progressive) {
|
|
83
|
+
// Full peak blur (parity with native) so the frosted glass is clearly
|
|
84
|
+
// engaged at the bottom; the eased masks keep the top fully clear.
|
|
85
|
+
const peakPx = Math.max(1, Math.min(32, Math.round(intensity * 0.32)))
|
|
86
|
+
return (
|
|
87
|
+
<View style={[StyleSheet.absoluteFill, style]} pointerEvents="none">
|
|
88
|
+
{PROGRESSIVE_LAYERS.map((layer, i) => {
|
|
89
|
+
const px = Math.max(1, Math.round(peakPx * layer.amount))
|
|
90
|
+
const gradient = toGradient(layer.stops)
|
|
91
|
+
// web-only CSS (backdrop-filter / mask-image); cast because
|
|
92
|
+
// these keys aren't in RN's ViewStyle.
|
|
93
|
+
const webStyle = {
|
|
94
|
+
backdropFilter: `blur(${px}px)`,
|
|
95
|
+
WebkitBackdropFilter: `blur(${px}px)`,
|
|
96
|
+
maskImage: gradient,
|
|
97
|
+
WebkitMaskImage: gradient,
|
|
98
|
+
} as unknown as ViewStyle
|
|
99
|
+
return (
|
|
100
|
+
<View
|
|
101
|
+
key={i}
|
|
102
|
+
style={[StyleSheet.absoluteFill, webStyle]}
|
|
103
|
+
pointerEvents="none"
|
|
104
|
+
/>
|
|
105
|
+
)
|
|
106
|
+
})}
|
|
107
|
+
</View>
|
|
108
|
+
)
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
return (
|
|
112
|
+
<View
|
|
113
|
+
style={[
|
|
114
|
+
StyleSheet.absoluteFill,
|
|
115
|
+
{ backgroundColor: tintColor },
|
|
116
|
+
// backdrop-filter is web-only CSS; ignored by RN on native
|
|
117
|
+
// (we never bundle this file there anyway).
|
|
118
|
+
// @ts-ignore web-only style
|
|
119
|
+
{ backdropFilter: `blur(${blurPx}px)`, WebkitBackdropFilter: `blur(${blurPx}px)` },
|
|
120
|
+
style,
|
|
121
|
+
]}
|
|
122
|
+
pointerEvents="none"
|
|
123
|
+
/>
|
|
124
|
+
)
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
export default GlassFill
|
|
@@ -0,0 +1,423 @@
|
|
|
1
|
+
import React, { useId, useMemo } from 'react'
|
|
2
|
+
import {
|
|
3
|
+
View,
|
|
4
|
+
Text,
|
|
5
|
+
Pressable,
|
|
6
|
+
StyleSheet,
|
|
7
|
+
type ViewStyle,
|
|
8
|
+
type TextStyle,
|
|
9
|
+
type StyleProp,
|
|
10
|
+
type ImageSourcePropType,
|
|
11
|
+
} from 'react-native'
|
|
12
|
+
import Svg, { Defs, LinearGradient, Stop, Rect } from 'react-native-svg'
|
|
13
|
+
import { getVariableByName } from '../../design-tokens/figma-variables-resolver'
|
|
14
|
+
import { EMPTY_MODES } from '../../utils/react-utils'
|
|
15
|
+
import Avatar from '../Avatar/Avatar'
|
|
16
|
+
import Badge from '../Badge/Badge'
|
|
17
|
+
import Button from '../Button/Button'
|
|
18
|
+
import Image from '../Image/Image'
|
|
19
|
+
import GlassFill from './GlassFill'
|
|
20
|
+
import type { Modes } from '../../design-tokens'
|
|
21
|
+
|
|
22
|
+
export interface ProductMerchandisingCardProps {
|
|
23
|
+
/**
|
|
24
|
+
* Background media for the card. Accepts a URL string or any RN
|
|
25
|
+
* `ImageSourcePropType` (same shape as `Avatar`, `MediaCard`, etc.). The
|
|
26
|
+
* image is rendered full-bleed behind the header and glass footer.
|
|
27
|
+
*/
|
|
28
|
+
imageSource?: ImageSourcePropType | string
|
|
29
|
+
/** Footer title (bold, white). */
|
|
30
|
+
title?: string
|
|
31
|
+
/** Footer subtitle (muted). */
|
|
32
|
+
subtitle?: string
|
|
33
|
+
/**
|
|
34
|
+
* Header badge label (top-right). The badge is hidden when this is
|
|
35
|
+
* `undefined` and no `badge` override is provided.
|
|
36
|
+
*/
|
|
37
|
+
badgeLabel?: string
|
|
38
|
+
/**
|
|
39
|
+
* Full override for the header badge. Takes precedence over `badgeLabel`.
|
|
40
|
+
*/
|
|
41
|
+
badge?: React.ReactNode
|
|
42
|
+
/** Show the header avatar (top-left). Defaults to `true`. */
|
|
43
|
+
showAvatar?: boolean
|
|
44
|
+
/** Avatar image source. */
|
|
45
|
+
avatarSource?: ImageSourcePropType | string
|
|
46
|
+
/** Avatar monogram fallback when no image is supplied. */
|
|
47
|
+
avatarMonogram?: string
|
|
48
|
+
/** Full override for the header avatar. Takes precedence over avatar props. */
|
|
49
|
+
avatar?: React.ReactNode
|
|
50
|
+
/**
|
|
51
|
+
* Footer "special badge" label (small frosted pill above the title). The
|
|
52
|
+
* badge is hidden when this is `undefined` and no `specialBadge` override
|
|
53
|
+
* is provided.
|
|
54
|
+
*/
|
|
55
|
+
specialBadgeLabel?: string
|
|
56
|
+
/** Optional leading node (icon/image) rendered inside the special badge. */
|
|
57
|
+
specialBadgeIcon?: React.ReactNode
|
|
58
|
+
/** Full override for the special badge. Takes precedence over the props above. */
|
|
59
|
+
specialBadge?: React.ReactNode
|
|
60
|
+
/** CTA button label. Defaults to `"CTA"`. */
|
|
61
|
+
ctaLabel?: string
|
|
62
|
+
/** CTA button press handler. */
|
|
63
|
+
onCtaPress?: () => void
|
|
64
|
+
/** Full override for the CTA. Takes precedence over `ctaLabel`/`onCtaPress`. */
|
|
65
|
+
cta?: React.ReactNode
|
|
66
|
+
/** Press handler for the whole card. When set, the card becomes pressable. */
|
|
67
|
+
onPress?: () => void
|
|
68
|
+
/** Card height in px. Defaults to `223` (Figma spec). */
|
|
69
|
+
height?: number
|
|
70
|
+
/** Modes object for design-token resolution. Cascaded to all children. */
|
|
71
|
+
modes?: Modes
|
|
72
|
+
/** Style overrides for the card container. */
|
|
73
|
+
style?: StyleProp<ViewStyle>
|
|
74
|
+
/** Accessibility label for the card. */
|
|
75
|
+
accessibilityLabel?: string
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* ProductMerchandisingCard — a full-bleed image card (Figma node 5277:317)
|
|
80
|
+
* with a top header (avatar + badge) and a bottom **glass footer** that
|
|
81
|
+
* combines a real native background blur with a transparent→black gradient
|
|
82
|
+
* scrim for legible text over any image.
|
|
83
|
+
*
|
|
84
|
+
* Glass footer implementation (works on iOS, Android and Web):
|
|
85
|
+
* - **iOS / Android:** `<GlassFill>` wraps `@react-native-community/blur`'s
|
|
86
|
+
* `BlurView` — iOS gets a real `UIVisualEffectView` (live OS blur),
|
|
87
|
+
* Android gets the community `RealtimeBlurView` with a tinted scrim
|
|
88
|
+
* fallback. The native-only module is isolated in `GlassFill.tsx`.
|
|
89
|
+
* - **Web:** the platform-extension `GlassFill.web.tsx` renders a translucent
|
|
90
|
+
* `View` with `backdrop-filter: blur()` — Metro picks the right file so the
|
|
91
|
+
* web bundle never imports the native blur module.
|
|
92
|
+
* - The gradient scrim is drawn with `react-native-svg` (one renderer for all
|
|
93
|
+
* platforms) so the fade is identical everywhere.
|
|
94
|
+
*
|
|
95
|
+
* The blur strength is driven by the Figma `blur/minimal` token, mapped to the
|
|
96
|
+
* `GlassFill` 0–100 intensity scale the same way `MediaCard` does.
|
|
97
|
+
*/
|
|
98
|
+
function ProductMerchandisingCard({
|
|
99
|
+
imageSource,
|
|
100
|
+
title = 'Title',
|
|
101
|
+
subtitle = 'Subtitle',
|
|
102
|
+
badgeLabel,
|
|
103
|
+
badge,
|
|
104
|
+
showAvatar = true,
|
|
105
|
+
avatarSource,
|
|
106
|
+
avatarMonogram,
|
|
107
|
+
avatar,
|
|
108
|
+
specialBadgeLabel,
|
|
109
|
+
specialBadgeIcon,
|
|
110
|
+
specialBadge,
|
|
111
|
+
ctaLabel = 'CTA',
|
|
112
|
+
onCtaPress,
|
|
113
|
+
cta,
|
|
114
|
+
onPress,
|
|
115
|
+
height = 223,
|
|
116
|
+
modes = EMPTY_MODES,
|
|
117
|
+
style,
|
|
118
|
+
accessibilityLabel,
|
|
119
|
+
}: ProductMerchandisingCardProps) {
|
|
120
|
+
// Unique gradient id so multiple cards on one (web) document never collide.
|
|
121
|
+
const rawId = useId()
|
|
122
|
+
const scrimId = `pmc-footer-scrim-${rawId.replace(/[^a-zA-Z0-9_-]/g, '')}`
|
|
123
|
+
|
|
124
|
+
const tokens = useMemo(() => resolveTokens(modes), [modes])
|
|
125
|
+
|
|
126
|
+
const containerStyle: ViewStyle = {
|
|
127
|
+
height,
|
|
128
|
+
borderRadius: tokens.radius,
|
|
129
|
+
overflow: 'hidden',
|
|
130
|
+
position: 'relative',
|
|
131
|
+
flexDirection: 'column',
|
|
132
|
+
justifyContent: 'space-between',
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
// Header avatar defaults to the design's 36px (Avatar Size = M). A
|
|
136
|
+
// consumer-supplied `modes['Avatar Size']` still wins via spread order.
|
|
137
|
+
const avatarModes = useMemo<Modes>(() => ({ 'Avatar Size': 'M', ...modes }), [modes])
|
|
138
|
+
|
|
139
|
+
// The Figma node renders the header badge with the brand "Secondary"
|
|
140
|
+
// appearance at "Medium" emphasis (lilac fill + dark label). These are set
|
|
141
|
+
// as defaults so the card matches the design out of the box; a
|
|
142
|
+
// consumer-supplied `modes` value still wins via spread order.
|
|
143
|
+
const badgeModes = useMemo<Modes>(
|
|
144
|
+
() => ({ AppearanceBrand: 'Secondary', Emphasis: 'Medium', ...modes }),
|
|
145
|
+
[modes]
|
|
146
|
+
)
|
|
147
|
+
|
|
148
|
+
const headerBadge =
|
|
149
|
+
badge ?? (badgeLabel != null ? <Badge label={badgeLabel} modes={badgeModes} /> : null)
|
|
150
|
+
|
|
151
|
+
const headerAvatar = showAvatar
|
|
152
|
+
? avatar ?? (
|
|
153
|
+
<Avatar
|
|
154
|
+
style={avatarSource ? 'Image' : avatarMonogram ? 'Monogram' : 'Image'}
|
|
155
|
+
imageSource={avatarSource}
|
|
156
|
+
monogram={avatarMonogram}
|
|
157
|
+
modes={avatarModes}
|
|
158
|
+
/>
|
|
159
|
+
)
|
|
160
|
+
: null
|
|
161
|
+
|
|
162
|
+
const footerSpecialBadge =
|
|
163
|
+
specialBadge ??
|
|
164
|
+
(specialBadgeLabel != null ? (
|
|
165
|
+
<View style={styles.specialBadge}>
|
|
166
|
+
<GlassFill
|
|
167
|
+
tint="light"
|
|
168
|
+
intensity={SPECIAL_BADGE_INTENSITY}
|
|
169
|
+
overlayColor={SPECIAL_BADGE_BG}
|
|
170
|
+
style={styles.specialBadgeFill}
|
|
171
|
+
/>
|
|
172
|
+
{specialBadgeIcon != null ? (
|
|
173
|
+
<View style={styles.specialBadgeIcon}>{specialBadgeIcon}</View>
|
|
174
|
+
) : null}
|
|
175
|
+
<Text style={tokens.specialBadgeText} numberOfLines={1}>
|
|
176
|
+
{specialBadgeLabel}
|
|
177
|
+
</Text>
|
|
178
|
+
</View>
|
|
179
|
+
) : null)
|
|
180
|
+
|
|
181
|
+
// The Figma CTA uses the brand "Secondary" appearance (purple fill, white
|
|
182
|
+
// label) at the small size. Defaults match the design; consumer `modes`
|
|
183
|
+
// win via spread order.
|
|
184
|
+
const ctaModes = useMemo<Modes>(
|
|
185
|
+
() => ({ 'Button / Size': 'S', AppearanceBrand: 'Secondary', ...modes }),
|
|
186
|
+
[modes]
|
|
187
|
+
)
|
|
188
|
+
|
|
189
|
+
const footerCta =
|
|
190
|
+
cta ?? <Button label={ctaLabel} modes={ctaModes} onPress={onCtaPress} />
|
|
191
|
+
|
|
192
|
+
const content = (
|
|
193
|
+
<>
|
|
194
|
+
{imageSource != null ? (
|
|
195
|
+
<Image
|
|
196
|
+
imageSource={imageSource}
|
|
197
|
+
style={StyleSheet.absoluteFillObject as any}
|
|
198
|
+
width="100%"
|
|
199
|
+
height={height}
|
|
200
|
+
resizeMode="cover"
|
|
201
|
+
accessibilityElementsHidden
|
|
202
|
+
importantForAccessibility="no"
|
|
203
|
+
/>
|
|
204
|
+
) : null}
|
|
205
|
+
|
|
206
|
+
<View style={tokens.header} pointerEvents="box-none">
|
|
207
|
+
<View style={styles.avatarContainer}>{headerAvatar}</View>
|
|
208
|
+
{headerBadge}
|
|
209
|
+
</View>
|
|
210
|
+
|
|
211
|
+
<View style={tokens.footer} pointerEvents="box-none">
|
|
212
|
+
{/* Progressive background blur: clear at the top, ramping to a
|
|
213
|
+
soft blur at the bottom (native MaskedView+BlurView / web
|
|
214
|
+
backdrop-filter + mask-image). */}
|
|
215
|
+
<GlassFill tint="dark" intensity={tokens.blurIntensity} progressive />
|
|
216
|
+
{/* Transparent -> black gradient scrim, identical on all platforms. */}
|
|
217
|
+
<View style={[StyleSheet.absoluteFill, styles.scrim]} pointerEvents="none">
|
|
218
|
+
<Svg width="100%" height="100%">
|
|
219
|
+
<Defs>
|
|
220
|
+
<LinearGradient id={scrimId} x1="0" y1="0" x2="0" y2="1">
|
|
221
|
+
<Stop offset="0.018" stopColor="#000000" stopOpacity={0} />
|
|
222
|
+
<Stop offset="0.974" stopColor="#000000" stopOpacity={1} />
|
|
223
|
+
</LinearGradient>
|
|
224
|
+
</Defs>
|
|
225
|
+
<Rect x="0" y="0" width="100%" height="100%" fill={`url(#${scrimId})`} />
|
|
226
|
+
</Svg>
|
|
227
|
+
</View>
|
|
228
|
+
|
|
229
|
+
{footerSpecialBadge}
|
|
230
|
+
|
|
231
|
+
<View style={styles.footerRow}>
|
|
232
|
+
<View style={tokens.textWrap}>
|
|
233
|
+
<Text style={tokens.title} numberOfLines={1}>
|
|
234
|
+
{title}
|
|
235
|
+
</Text>
|
|
236
|
+
{subtitle != null ? (
|
|
237
|
+
<Text style={tokens.subtitle} numberOfLines={1}>
|
|
238
|
+
{subtitle}
|
|
239
|
+
</Text>
|
|
240
|
+
) : null}
|
|
241
|
+
</View>
|
|
242
|
+
{footerCta}
|
|
243
|
+
</View>
|
|
244
|
+
</View>
|
|
245
|
+
</>
|
|
246
|
+
)
|
|
247
|
+
|
|
248
|
+
if (onPress) {
|
|
249
|
+
return (
|
|
250
|
+
<Pressable
|
|
251
|
+
style={({ pressed }) => [containerStyle, pressed ? styles.pressed : null, style]}
|
|
252
|
+
accessibilityRole="button"
|
|
253
|
+
accessibilityLabel={accessibilityLabel ?? title}
|
|
254
|
+
onPress={onPress}
|
|
255
|
+
>
|
|
256
|
+
{content}
|
|
257
|
+
</Pressable>
|
|
258
|
+
)
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
return (
|
|
262
|
+
<View
|
|
263
|
+
style={[containerStyle, style]}
|
|
264
|
+
accessibilityLabel={accessibilityLabel}
|
|
265
|
+
>
|
|
266
|
+
{content}
|
|
267
|
+
</View>
|
|
268
|
+
)
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
// ---------------------------------------------------------------------------
|
|
272
|
+
// Tokens / static styles
|
|
273
|
+
// ---------------------------------------------------------------------------
|
|
274
|
+
|
|
275
|
+
// Figma footer backdrop-blur token (`blur/minimal`). The footer uses a
|
|
276
|
+
// *progressive* blur, so this maps the token to the intensity at the BOTTOM of
|
|
277
|
+
// the ramp (the strongest point); the upper part fades to fully clear. Kept
|
|
278
|
+
// intentionally small to match the design's subtle "minimal" blur.
|
|
279
|
+
const BLUR_INTENSITY_FACTOR = 1.0
|
|
280
|
+
|
|
281
|
+
// Special badge ("frosted" pill) — values are literals in the Figma node
|
|
282
|
+
// (no dedicated tokens): bg rgba(255,245,229,0.15), 5px backdrop blur.
|
|
283
|
+
const SPECIAL_BADGE_BG = 'rgba(255,245,229,0.15)'
|
|
284
|
+
const SPECIAL_BADGE_INTENSITY = 17 // ~5px CSS blur / ~2 native blurAmount
|
|
285
|
+
|
|
286
|
+
interface ResolvedTokens {
|
|
287
|
+
radius: number
|
|
288
|
+
blurIntensity: number
|
|
289
|
+
header: ViewStyle
|
|
290
|
+
footer: ViewStyle
|
|
291
|
+
textWrap: ViewStyle
|
|
292
|
+
title: TextStyle
|
|
293
|
+
subtitle: TextStyle
|
|
294
|
+
specialBadgeText: TextStyle
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
function asNum(raw: unknown, fallback: number): number {
|
|
298
|
+
const n = typeof raw === 'number' ? raw : parseFloat(raw as string)
|
|
299
|
+
return Number.isFinite(n) ? n : fallback
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
function asStr(raw: unknown, fallback: string): string {
|
|
303
|
+
return raw != null ? String(raw) : fallback
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
function resolveTokens(modes: Modes): ResolvedTokens {
|
|
307
|
+
// NOTE: token names are passed as string literals DIRECTLY to
|
|
308
|
+
// getVariableByName so the `extract-component-tokens` script can statically
|
|
309
|
+
// collect them for the generated docs. Do not refactor these into a helper
|
|
310
|
+
// that receives the name as a variable.
|
|
311
|
+
const radius = asNum(getVariableByName('productMerchandisingcard/radius', modes), 12)
|
|
312
|
+
const headerPadH = asNum(getVariableByName('productMerchandisingcard/header/padding/horizontal', modes), 12)
|
|
313
|
+
const headerPadV = asNum(getVariableByName('productMerchandisingcard/header/padding/vertical', modes), 12)
|
|
314
|
+
const footerGap = asNum(getVariableByName('productMerchandisingcard/footer/gap', modes), 8)
|
|
315
|
+
const footerPadH = asNum(getVariableByName('productMerchandisingcard/footer/padding/horizontal', modes), 16)
|
|
316
|
+
const footerPadV = asNum(getVariableByName('productMerchandisingcard/footer/padding/vertical', modes), 16)
|
|
317
|
+
const textGap = asNum(getVariableByName('productMerchandisingcard/footer/text/gap', modes), 2)
|
|
318
|
+
const blurRadius = asNum(getVariableByName('blur/minimal', modes), 29)
|
|
319
|
+
|
|
320
|
+
const titleColor = asStr(getVariableByName('productMerchandisingcard/footer/title/font/color', modes), '#ffffff')
|
|
321
|
+
const titleSize = asNum(getVariableByName('productMerchandisingcard/footer/title/fontsize', modes), 14)
|
|
322
|
+
const titleFamily = asStr(getVariableByName('productMerchandisingcard/footer/title/fontfamily', modes), 'JioType Var')
|
|
323
|
+
const titleWeight = asStr(getVariableByName('productMerchandisingcard/footer/title/fontweight', modes), '700')
|
|
324
|
+
|
|
325
|
+
const subtitleColor = asStr(getVariableByName('productMerchandisingcard/footer/subtitle/font/color', modes), '#c5bcb5')
|
|
326
|
+
const subtitleSize = asNum(getVariableByName('productMerchandisingcard/footer/subtitle/fontsize', modes), 12)
|
|
327
|
+
const subtitleFamily = asStr(getVariableByName('productMerchandisingcard/footer/subtitle/fontfamily', modes), 'JioType Var')
|
|
328
|
+
const subtitleWeight = asStr(getVariableByName('productMerchandisingcard/footer/subtitle/fontweight', modes), '400')
|
|
329
|
+
|
|
330
|
+
const sbSize = asNum(getVariableByName('productMerchandisingcard/specialbadge/text/fontsize', modes), 12)
|
|
331
|
+
const sbFamily = asStr(getVariableByName('productMerchandisingcard/specialbadge/text/fontfamily', modes), 'JioType Var')
|
|
332
|
+
const sbWeight = asStr(getVariableByName('productMerchandisingcard/specialbadge/text/fontweight', modes), '500')
|
|
333
|
+
|
|
334
|
+
return {
|
|
335
|
+
radius,
|
|
336
|
+
blurIntensity: Math.max(0, Math.min(100, Math.round(blurRadius * BLUR_INTENSITY_FACTOR))),
|
|
337
|
+
header: {
|
|
338
|
+
flexDirection: 'row',
|
|
339
|
+
alignItems: 'flex-start',
|
|
340
|
+
justifyContent: 'space-between',
|
|
341
|
+
paddingHorizontal: headerPadH,
|
|
342
|
+
paddingVertical: headerPadV,
|
|
343
|
+
zIndex: 1,
|
|
344
|
+
},
|
|
345
|
+
footer: {
|
|
346
|
+
paddingHorizontal: footerPadH,
|
|
347
|
+
paddingVertical: footerPadV,
|
|
348
|
+
gap: footerGap,
|
|
349
|
+
overflow: 'hidden',
|
|
350
|
+
zIndex: 2,
|
|
351
|
+
},
|
|
352
|
+
textWrap: {
|
|
353
|
+
flex: 1,
|
|
354
|
+
gap: textGap,
|
|
355
|
+
justifyContent: 'center',
|
|
356
|
+
},
|
|
357
|
+
title: {
|
|
358
|
+
color: titleColor,
|
|
359
|
+
fontSize: titleSize,
|
|
360
|
+
fontFamily: titleFamily,
|
|
361
|
+
fontWeight: titleWeight as TextStyle['fontWeight'],
|
|
362
|
+
lineHeight: Math.round(titleSize * 1.2),
|
|
363
|
+
includeFontPadding: false as any,
|
|
364
|
+
},
|
|
365
|
+
subtitle: {
|
|
366
|
+
color: subtitleColor,
|
|
367
|
+
fontSize: subtitleSize,
|
|
368
|
+
fontFamily: subtitleFamily,
|
|
369
|
+
fontWeight: subtitleWeight as TextStyle['fontWeight'],
|
|
370
|
+
lineHeight: Math.round(subtitleSize * 1.2),
|
|
371
|
+
includeFontPadding: false as any,
|
|
372
|
+
},
|
|
373
|
+
specialBadgeText: {
|
|
374
|
+
color: '#ffffff',
|
|
375
|
+
fontSize: sbSize,
|
|
376
|
+
fontFamily: sbFamily,
|
|
377
|
+
fontWeight: sbWeight as TextStyle['fontWeight'],
|
|
378
|
+
lineHeight: Math.round(sbSize * 1.2),
|
|
379
|
+
includeFontPadding: false as any,
|
|
380
|
+
},
|
|
381
|
+
}
|
|
382
|
+
}
|
|
383
|
+
|
|
384
|
+
const styles = StyleSheet.create({
|
|
385
|
+
avatarContainer: {
|
|
386
|
+
flexShrink: 1,
|
|
387
|
+
justifyContent: 'center',
|
|
388
|
+
},
|
|
389
|
+
footerRow: {
|
|
390
|
+
flexDirection: 'row',
|
|
391
|
+
alignItems: 'center',
|
|
392
|
+
justifyContent: 'space-between',
|
|
393
|
+
gap: 8,
|
|
394
|
+
},
|
|
395
|
+
scrim: {
|
|
396
|
+
// The footer node sits at 90% opacity in Figma; applying it to the
|
|
397
|
+
// scrim (not the content) keeps the title/CTA crisp while preserving
|
|
398
|
+
// the intended fade strength.
|
|
399
|
+
opacity: 0.9,
|
|
400
|
+
},
|
|
401
|
+
specialBadge: {
|
|
402
|
+
flexDirection: 'row',
|
|
403
|
+
alignItems: 'center',
|
|
404
|
+
gap: 4,
|
|
405
|
+
paddingHorizontal: 6,
|
|
406
|
+
paddingVertical: 2,
|
|
407
|
+
borderRadius: 4,
|
|
408
|
+
overflow: 'hidden',
|
|
409
|
+
alignSelf: 'flex-start',
|
|
410
|
+
},
|
|
411
|
+
specialBadgeFill: {
|
|
412
|
+
borderRadius: 4,
|
|
413
|
+
},
|
|
414
|
+
specialBadgeIcon: {
|
|
415
|
+
justifyContent: 'center',
|
|
416
|
+
alignItems: 'center',
|
|
417
|
+
},
|
|
418
|
+
pressed: {
|
|
419
|
+
opacity: 0.92,
|
|
420
|
+
},
|
|
421
|
+
})
|
|
422
|
+
|
|
423
|
+
export default ProductMerchandisingCard
|