@retray-dev/ui-kit 12.2.0 → 13.0.0
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/COMPONENTS.md +85 -143
- package/CONSUMER.md +2 -2
- package/DESIGN.md +2 -2
- package/README.md +11 -6
- package/dist/Accordion.js +48 -210
- package/dist/Accordion.mjs +6 -5
- package/dist/AlertBanner.js +29 -153
- package/dist/AlertBanner.mjs +3 -3
- package/dist/AppHeader.js +37 -235
- package/dist/AppHeader.mjs +6 -7
- package/dist/Avatar.d.mts +17 -1
- package/dist/Avatar.d.ts +17 -1
- package/dist/Avatar.js +80 -115
- package/dist/Avatar.mjs +2 -2
- package/dist/Badge.js +24 -149
- package/dist/Badge.mjs +3 -3
- package/dist/Button.js +79 -267
- package/dist/Button.mjs +6 -6
- package/dist/Card.js +15 -200
- package/dist/Card.mjs +4 -5
- package/dist/CategoryStrip.d.mts +0 -5
- package/dist/CategoryStrip.d.ts +0 -5
- package/dist/CategoryStrip.js +47 -265
- package/dist/CategoryStrip.mjs +6 -6
- package/dist/Checkbox.js +15 -200
- package/dist/Checkbox.mjs +5 -5
- package/dist/Chip.js +44 -236
- package/dist/Chip.mjs +7 -6
- package/dist/ConfirmDialog.js +84 -286
- package/dist/ConfirmDialog.mjs +7 -7
- package/dist/CurrencyDisplay.js +1 -114
- package/dist/CurrencyDisplay.mjs +2 -2
- package/dist/CurrencyInput.js +35 -162
- package/dist/CurrencyInput.mjs +5 -5
- package/dist/DetailRow.js +25 -150
- package/dist/DetailRow.mjs +3 -3
- package/dist/EmptyState.js +80 -268
- package/dist/EmptyState.mjs +7 -7
- package/dist/ErrorBoundary.js +32 -199
- package/dist/ErrorBoundary.mjs +4 -4
- package/dist/Form.js +1 -114
- package/dist/Form.mjs +2 -2
- package/dist/HolographicCard.d.mts +0 -28
- package/dist/HolographicCard.d.ts +0 -28
- package/dist/HolographicCard.js +20 -130
- package/dist/HolographicCard.mjs +9 -32
- package/dist/IconButton.js +36 -234
- package/dist/IconButton.mjs +5 -6
- package/dist/IconPicker.js +222 -929
- package/dist/IconPicker.mjs +5 -5
- package/dist/ImageUpload.d.mts +3 -1
- package/dist/ImageUpload.d.ts +3 -1
- package/dist/ImageUpload.js +25 -215
- package/dist/ImageUpload.mjs +5 -6
- package/dist/ImageViewer.js +75 -266
- package/dist/ImageViewer.mjs +8 -8
- package/dist/Input.d.mts +1 -1
- package/dist/Input.d.ts +1 -1
- package/dist/Input.js +35 -162
- package/dist/Input.mjs +4 -4
- package/dist/LabelValue.js +24 -149
- package/dist/LabelValue.mjs +3 -3
- package/dist/ListGroup.js +1 -114
- package/dist/ListGroup.mjs +2 -2
- package/dist/ListItem.js +38 -235
- package/dist/ListItem.mjs +5 -6
- package/dist/MediaCard.d.mts +0 -14
- package/dist/MediaCard.d.ts +0 -14
- package/dist/MediaCard.js +69 -315
- package/dist/MediaCard.mjs +5 -6
- package/dist/MenuGroup.js +1 -114
- package/dist/MenuGroup.mjs +2 -2
- package/dist/MenuItem.js +36 -234
- package/dist/MenuItem.mjs +5 -6
- package/dist/MonthPicker.js +8 -163
- package/dist/MonthPicker.mjs +3 -3
- package/dist/NumberStepper.js +40 -238
- package/dist/NumberStepper.mjs +5 -6
- package/dist/PagerDots.d.mts +1 -1
- package/dist/PagerDots.d.ts +1 -1
- package/dist/PagerDots.js +69 -224
- package/dist/PagerDots.mjs +6 -5
- package/dist/Pressable.js +14 -85
- package/dist/Pressable.mjs +4 -4
- package/dist/PricingCard.js +87 -272
- package/dist/PricingCard.mjs +8 -8
- package/dist/Progress.js +3 -123
- package/dist/Progress.mjs +3 -3
- package/dist/RadioGroup.js +52 -265
- package/dist/RadioGroup.mjs +5 -5
- package/dist/RetrayProvider.js +3 -6
- package/dist/RetrayProvider.mjs +3 -3
- package/dist/Select.d.mts +2 -1
- package/dist/Select.d.ts +2 -1
- package/dist/Select.js +24 -232
- package/dist/Select.mjs +4 -5
- package/dist/SelectableCard.js +33 -209
- package/dist/SelectableCard.mjs +5 -5
- package/dist/SelectableGrid.d.mts +0 -21
- package/dist/SelectableGrid.d.ts +0 -21
- package/dist/SelectableGrid.js +49 -271
- package/dist/SelectableGrid.mjs +5 -6
- package/dist/Separator.js +1 -114
- package/dist/Separator.mjs +2 -2
- package/dist/Sheet.js +7 -162
- package/dist/Sheet.mjs +3 -3
- package/dist/SheetSelect.js +39 -236
- package/dist/SheetSelect.mjs +6 -6
- package/dist/Skeleton.js +4 -124
- package/dist/Skeleton.mjs +3 -3
- package/dist/Slider.js +6 -161
- package/dist/Slider.mjs +3 -3
- package/dist/Spinner.js +3 -116
- package/dist/Spinner.mjs +2 -2
- package/dist/Stats.js +36 -234
- package/dist/Stats.mjs +5 -6
- package/dist/Switch.js +24 -175
- package/dist/Switch.mjs +5 -4
- package/dist/TabBar.js +43 -200
- package/dist/TabBar.mjs +5 -4
- package/dist/Tabs.js +15 -199
- package/dist/Tabs.mjs +5 -5
- package/dist/Text.js +9 -130
- package/dist/Text.mjs +2 -2
- package/dist/Textarea.d.mts +2 -1
- package/dist/Textarea.d.ts +2 -1
- package/dist/Textarea.js +71 -219
- package/dist/Textarea.mjs +4 -4
- package/dist/Toast.js +1 -114
- package/dist/Toast.mjs +2 -2
- package/dist/Toggle.js +39 -236
- package/dist/Toggle.mjs +6 -6
- package/dist/{chunk-M3C7XM2M.mjs → chunk-2QOHHBJC.mjs} +3 -3
- package/dist/{chunk-LIS6I5UP.mjs → chunk-2VIDP72N.mjs} +3 -3
- package/dist/{chunk-DF7JA72E.mjs → chunk-4NQFTHN3.mjs} +13 -7
- package/dist/{chunk-UBUXUMER.mjs → chunk-4ZO5PTKF.mjs} +4 -4
- package/dist/{chunk-3XCFYSX4.mjs → chunk-5MYNAAFE.mjs} +13 -17
- package/dist/{chunk-E7NEHHXV.mjs → chunk-62BBSSUF.mjs} +3 -3
- package/dist/{chunk-MVMGPZN6.mjs → chunk-6CR4S6W2.mjs} +3 -3
- package/dist/{chunk-EDLCGYIO.mjs → chunk-6QLBHUEG.mjs} +8 -7
- package/dist/chunk-ARONDO7M.mjs +40 -0
- package/dist/{chunk-GH67YXG6.mjs → chunk-AZV7KNJI.mjs} +3 -3
- package/dist/{chunk-RMRS44MQ.mjs → chunk-BTUW5LSG.mjs} +11 -8
- package/dist/{chunk-2P2CB235.mjs → chunk-BULKGOIZ.mjs} +7 -8
- package/dist/{chunk-NHDI3VQB.mjs → chunk-CBIZLRYH.mjs} +15 -12
- package/dist/chunk-CM2DG4MR.mjs +142 -0
- package/dist/{chunk-TS7DGUIR.mjs → chunk-DBHSUUKU.mjs} +2 -2
- package/dist/{chunk-57V2LXCK.mjs → chunk-DE25XTVQ.mjs} +3 -3
- package/dist/{chunk-UQ4742ET.mjs → chunk-E4EQSCKR.mjs} +5 -5
- package/dist/{chunk-GUTDFUNF.mjs → chunk-EHGBHFMH.mjs} +9 -17
- package/dist/{chunk-CF27NBXO.mjs → chunk-EROPDCB5.mjs} +16 -24
- package/dist/{chunk-ZIMY2QUM.mjs → chunk-ERWJPVX7.mjs} +2 -2
- package/dist/{chunk-NLZY4TXU.mjs → chunk-ESQDPO5E.mjs} +7 -7
- package/dist/{chunk-VJBUCITV.mjs → chunk-EW2FIDSM.mjs} +1 -1
- package/dist/{chunk-HC4VVCWY.mjs → chunk-FTTI6T5Q.mjs} +4 -4
- package/dist/{chunk-2HFD4IHU.mjs → chunk-HUSSF6TF.mjs} +1 -1
- package/dist/chunk-IFYMBOEN.mjs +14 -0
- package/dist/{chunk-QOLWA2PW.mjs → chunk-IGU223UM.mjs} +80 -4
- package/dist/chunk-IJCMPVW5.mjs +121 -0
- package/dist/{chunk-AENAVIKT.mjs → chunk-ITG4JQM3.mjs} +4 -4
- package/dist/{chunk-E5UKLSJZ.mjs → chunk-K3QX2M26.mjs} +11 -8
- package/dist/{chunk-4OORJ2DY.mjs → chunk-K7TKID3V.mjs} +8 -7
- package/dist/{chunk-2LG326TT.mjs → chunk-KAGADD2O.mjs} +4 -4
- package/dist/{chunk-IVSRW4HS.mjs → chunk-KC5QDYGZ.mjs} +4 -4
- package/dist/{chunk-7AFZWSCI.mjs → chunk-KPTY7UYQ.mjs} +1 -1
- package/dist/{chunk-YTXRIXNZ.mjs → chunk-KSSVIFYR.mjs} +9 -12
- package/dist/chunk-L3YKPTJQ.mjs +119 -0
- package/dist/chunk-M53LC4Q7.mjs +35 -0
- package/dist/{chunk-ZR6HSEAB.mjs → chunk-MP7GLMIR.mjs} +17 -25
- package/dist/chunk-MZ6WRTD2.mjs +40 -0
- package/dist/chunk-NGEN2EES.mjs +581 -0
- package/dist/{chunk-C43HRKXH.mjs → chunk-OBV72JD4.mjs} +1 -1
- package/dist/{chunk-LPV4NJJK.mjs → chunk-PGQ6FMXS.mjs} +6 -5
- package/dist/{chunk-MEPSKGBO.mjs → chunk-PI6RULJX.mjs} +1 -1
- package/dist/{chunk-F3YTWO3T.mjs → chunk-RA6SAAFE.mjs} +9 -8
- package/dist/{chunk-UNNRUJTM.mjs → chunk-RRKM4MKB.mjs} +7 -7
- package/dist/{chunk-ULGNQPNE.mjs → chunk-S2VGME7X.mjs} +1 -1
- package/dist/{chunk-OLVJFKXS.mjs → chunk-S44XWTTC.mjs} +35 -25
- package/dist/{chunk-YMYIEVZP.mjs → chunk-SZEKQAOY.mjs} +1 -1
- package/dist/{chunk-ELGEOM7I.mjs → chunk-TETMEKZE.mjs} +9 -9
- package/dist/{chunk-BXF4AMHY.mjs → chunk-TMH263OK.mjs} +5 -4
- package/dist/{chunk-NJG7DHVF.mjs → chunk-U6DEBYU5.mjs} +10 -9
- package/dist/{chunk-RJNLAH76.mjs → chunk-UOKFSFNJ.mjs} +2 -2
- package/dist/{chunk-HEDQPK4I.mjs → chunk-URIH43IJ.mjs} +13 -21
- package/dist/{chunk-QXDGGOLC.mjs → chunk-V2ZB2XNS.mjs} +6 -6
- package/dist/{chunk-KSUWPU2F.mjs → chunk-WIPEDNSD.mjs} +7 -7
- package/dist/{chunk-QDAZGZUF.mjs → chunk-XCIG6HT2.mjs} +3 -3
- package/dist/{chunk-4J2PXL36.mjs → chunk-Y6YS33GM.mjs} +40 -38
- package/dist/{chunk-4XOB5TTD.mjs → chunk-ZKDKKQCE.mjs} +5 -5
- package/dist/{chunk-LOBLCFMN.mjs → chunk-ZTPYUU5C.mjs} +5 -5
- package/dist/index.d.mts +12 -72
- package/dist/index.d.ts +12 -72
- package/dist/index.js +1051 -1838
- package/dist/index.mjs +81 -85
- package/package.json +8 -10
- package/src/components/Accordion/Accordion.tsx +12 -9
- package/src/components/AlertBanner/AlertBanner.tsx +7 -6
- package/src/components/AppHeader/AppHeader.tsx +1 -1
- package/src/components/Avatar/Avatar.tsx +92 -1
- package/src/components/Avatar/index.ts +2 -2
- package/src/components/Badge/Badge.tsx +2 -2
- package/src/components/Button/Button.tsx +50 -46
- package/src/components/Card/Card.tsx +1 -0
- package/src/components/CategoryStrip/CategoryStrip.tsx +36 -49
- package/src/components/Chip/Chip.tsx +5 -4
- package/src/components/ConfirmDialog/ConfirmDialog.tsx +3 -3
- package/src/components/DetailRow/DetailRow.tsx +3 -3
- package/src/components/EmptyState/EmptyState.tsx +2 -2
- package/src/components/ErrorBoundary/ErrorBoundary.tsx +6 -6
- package/src/components/HolographicCard/HolographicCard.tsx +14 -95
- package/src/components/IconButton/IconButton.tsx +2 -2
- package/src/components/IconPicker/IconPicker.tsx +13 -12
- package/src/components/ImageUpload/ImageUpload.tsx +14 -25
- package/src/components/ImageViewer/ImageViewer.tsx +3 -3
- package/src/components/Input/Input.tsx +11 -5
- package/src/components/LabelValue/LabelValue.tsx +2 -2
- package/src/components/ListItem/ListItem.tsx +4 -4
- package/src/components/MediaCard/MediaCard.tsx +21 -59
- package/src/components/MenuItem/MenuItem.tsx +2 -2
- package/src/components/MonthPicker/MonthPicker.tsx +2 -2
- package/src/components/NumberStepper/NumberStepper.tsx +6 -6
- package/src/components/PagerDots/PagerDots.tsx +38 -28
- package/src/components/PricingCard/PricingCard.tsx +6 -6
- package/src/components/RadioGroup/RadioGroup.tsx +18 -31
- package/src/components/Select/Select.tsx +32 -39
- package/src/components/SelectableCard/SelectableCard.tsx +4 -6
- package/src/components/SelectableGrid/SelectableGrid.tsx +38 -72
- package/src/components/Sheet/Sheet.tsx +1 -1
- package/src/components/SheetSelect/SheetSelect.tsx +3 -3
- package/src/components/Skeleton/Skeleton.tsx +1 -1
- package/src/components/Spinner/Spinner.tsx +2 -2
- package/src/components/Stats/Stats.tsx +2 -2
- package/src/components/Switch/Switch.tsx +9 -6
- package/src/components/TabBar/TabBar.tsx +9 -8
- package/src/components/Text/Text.tsx +12 -1
- package/src/components/Textarea/Textarea.tsx +18 -32
- package/src/components/Toggle/Toggle.tsx +3 -3
- package/src/hooks/useConfirmDialog.ts +31 -42
- package/src/index.ts +3 -4
- package/src/theme/ThemeProvider.tsx +1 -4
- package/src/theme/colorUtils.ts +1 -72
- package/src/theme/colors.ts +40 -1
- package/src/theme/types.ts +2 -2
- package/src/utils/animations.ts +0 -47
- package/src/utils/curatedIcons.ts +93 -801
- package/src/utils/haptics.ts +13 -208
- package/src/utils/icons.ts +27 -91
- package/src/utils/pressable.ts +10 -61
- package/dist/VirtualList.d.mts +0 -19
- package/dist/VirtualList.d.ts +0 -19
- package/dist/VirtualList.js +0 -38
- package/dist/VirtualList.mjs +0 -2
- package/dist/chunk-2BA3JMKK.mjs +0 -136
- package/dist/chunk-3DKJ2GIC.mjs +0 -30
- package/dist/chunk-7ELGZ66G.mjs +0 -164
- package/dist/chunk-DVK4G2GT.mjs +0 -59
- package/dist/chunk-EJ7ZPXOH.mjs +0 -163
- package/dist/chunk-KA7LTET3.mjs +0 -71
- package/dist/chunk-LNPKGWBG.mjs +0 -134
- package/dist/chunk-NC5ZTR2Y.mjs +0 -32
- package/dist/chunk-SAWUXP3A.mjs +0 -1114
- package/dist/chunk-YNROWHQJ.mjs +0 -46
- package/src/components/VirtualList/VirtualList.tsx +0 -60
- package/src/components/VirtualList/index.ts +0 -1
- package/src/utils/fontGuard.ts +0 -35
- package/src/utils/hover.ts +0 -25
- package/src/utils/useColorTransition.ts +0 -40
- package/src/utils/usePressScale.ts +0 -75
|
@@ -1,67 +1,56 @@
|
|
|
1
|
-
import { useState, useCallback } from 'react'
|
|
1
|
+
import { useState, useCallback, useRef, useEffect } from 'react'
|
|
2
2
|
|
|
3
3
|
export interface UseConfirmDialogOptions {
|
|
4
4
|
onConfirm: () => void | Promise<void>
|
|
5
5
|
onCancel?: () => void
|
|
6
6
|
}
|
|
7
7
|
|
|
8
|
-
export interface UseConfirmDialogResult
|
|
9
|
-
/** Pass to ConfirmDialog `visible` prop. */
|
|
8
|
+
export interface UseConfirmDialogResult {
|
|
10
9
|
visible: boolean
|
|
11
|
-
/** The value passed to `open()` — available during the confirmation flow. */
|
|
12
|
-
target: T | null
|
|
13
|
-
/** Whether `onConfirm` is currently executing. Pass to ConfirmDialog `loading` prop. */
|
|
14
10
|
loading: boolean
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
dialogProps: {
|
|
19
|
-
visible: boolean
|
|
20
|
-
loading: boolean
|
|
21
|
-
onConfirm: () => void
|
|
22
|
-
onCancel: () => void
|
|
23
|
-
}
|
|
11
|
+
open: () => void
|
|
12
|
+
onConfirm: () => void
|
|
13
|
+
onCancel: () => void
|
|
24
14
|
}
|
|
25
15
|
|
|
26
|
-
export function useConfirmDialog
|
|
27
|
-
options: UseConfirmDialogOptions,
|
|
28
|
-
): UseConfirmDialogResult<T> {
|
|
16
|
+
export function useConfirmDialog(options: UseConfirmDialogOptions): UseConfirmDialogResult {
|
|
29
17
|
const [visible, setVisible] = useState(false)
|
|
30
|
-
const [target, setTarget] = useState<T | null>(null)
|
|
31
18
|
const [loading, setLoading] = useState(false)
|
|
19
|
+
const mountedRef = useRef(true)
|
|
20
|
+
const onConfirmRef = useRef(options.onConfirm)
|
|
21
|
+
const onCancelRef = useRef(options.onCancel)
|
|
32
22
|
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
23
|
+
useEffect(() => {
|
|
24
|
+
onConfirmRef.current = options.onConfirm
|
|
25
|
+
onCancelRef.current = options.onCancel
|
|
26
|
+
})
|
|
27
|
+
|
|
28
|
+
useEffect(() => {
|
|
29
|
+
return () => {
|
|
30
|
+
mountedRef.current = false
|
|
31
|
+
}
|
|
36
32
|
}, [])
|
|
37
33
|
|
|
34
|
+
const open = useCallback(() => setVisible(true), [])
|
|
35
|
+
|
|
38
36
|
const handleConfirm = useCallback(async () => {
|
|
39
37
|
setLoading(true)
|
|
40
38
|
try {
|
|
41
|
-
await
|
|
39
|
+
await onConfirmRef.current()
|
|
40
|
+
} catch {
|
|
41
|
+
/* consumer handles error in onConfirm */
|
|
42
42
|
} finally {
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
43
|
+
if (mountedRef.current) {
|
|
44
|
+
setLoading(false)
|
|
45
|
+
setVisible(false)
|
|
46
|
+
}
|
|
46
47
|
}
|
|
47
|
-
}, [
|
|
48
|
+
}, [])
|
|
48
49
|
|
|
49
50
|
const handleCancel = useCallback(() => {
|
|
50
51
|
setVisible(false)
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
}, [options])
|
|
52
|
+
onCancelRef.current?.()
|
|
53
|
+
}, [])
|
|
54
54
|
|
|
55
|
-
return {
|
|
56
|
-
visible,
|
|
57
|
-
target,
|
|
58
|
-
loading,
|
|
59
|
-
open,
|
|
60
|
-
dialogProps: {
|
|
61
|
-
visible,
|
|
62
|
-
loading,
|
|
63
|
-
onConfirm: handleConfirm,
|
|
64
|
-
onCancel: handleCancel,
|
|
65
|
-
},
|
|
66
|
-
}
|
|
55
|
+
return { visible, loading, open, onConfirm: handleConfirm, onCancel: handleCancel }
|
|
67
56
|
}
|
package/src/index.ts
CHANGED
|
@@ -44,7 +44,6 @@ export * from './components/CategoryStrip'
|
|
|
44
44
|
export * from './components/Pressable'
|
|
45
45
|
export * from './components/DetailRow'
|
|
46
46
|
export * from './components/Form'
|
|
47
|
-
export * from './components/VirtualList'
|
|
48
47
|
export * from './components/RetrayProvider'
|
|
49
48
|
export * from './components/ErrorBoundary'
|
|
50
49
|
export * from './components/PagerDots'
|
|
@@ -64,10 +63,10 @@ export * from './components/Stats'
|
|
|
64
63
|
// barrel's module graph. Deep-import it: '@retray-dev/ui-kit/HolographicCard'.
|
|
65
64
|
|
|
66
65
|
// Icon utility
|
|
67
|
-
export { Icon
|
|
66
|
+
export { Icon } from './utils/icons'
|
|
68
67
|
|
|
69
68
|
// Color utilities
|
|
70
|
-
export { withAlpha } from './theme/colorUtils'
|
|
69
|
+
export { withAlpha, hexToRgb } from './theme/colorUtils'
|
|
71
70
|
|
|
72
71
|
// Typography utilities
|
|
73
72
|
export { getResponsiveFontSize } from './utils/typography'
|
|
@@ -82,13 +81,13 @@ export {
|
|
|
82
81
|
notificationSuccess,
|
|
83
82
|
notificationError,
|
|
84
83
|
notificationWarning,
|
|
85
|
-
richHaptics,
|
|
86
84
|
} from './utils/haptics'
|
|
87
85
|
|
|
88
86
|
// Hooks
|
|
89
87
|
export { useConfirmDialog } from './hooks/useConfirmDialog'
|
|
90
88
|
export type { UseConfirmDialogOptions, UseConfirmDialogResult } from './hooks/useConfirmDialog'
|
|
91
89
|
|
|
90
|
+
|
|
92
91
|
// Design tokens
|
|
93
92
|
export {
|
|
94
93
|
SPACING,
|
|
@@ -3,10 +3,7 @@ import { useColorScheme } from 'react-native'
|
|
|
3
3
|
import { ThemeColors, Theme, ColorScheme, ThemeContextValue } from './types'
|
|
4
4
|
import { defaultLight, defaultDark, deriveColors } from './colors'
|
|
5
5
|
|
|
6
|
-
const ThemeContext = createContext<ThemeContextValue>(
|
|
7
|
-
colors: deriveColors(defaultLight, 'light'),
|
|
8
|
-
colorScheme: 'light',
|
|
9
|
-
})
|
|
6
|
+
const ThemeContext = createContext<ThemeContextValue | undefined>(undefined)
|
|
10
7
|
|
|
11
8
|
export interface ThemeProviderProps {
|
|
12
9
|
children: React.ReactNode
|
package/src/theme/colorUtils.ts
CHANGED
|
@@ -1,7 +1,4 @@
|
|
|
1
|
-
|
|
2
|
-
// All functions are pure — no side effects, no React dependencies.
|
|
3
|
-
|
|
4
|
-
function hexToRgb(hex: string): { r: number; g: number; b: number } | null {
|
|
1
|
+
export function hexToRgb(hex: string): { r: number; g: number; b: number } | null {
|
|
5
2
|
const clean = hex.replace('#', '')
|
|
6
3
|
const full = clean.length === 3
|
|
7
4
|
? clean.split('').map(c => c + c).join('')
|
|
@@ -14,74 +11,6 @@ function hexToRgb(hex: string): { r: number; g: number; b: number } | null {
|
|
|
14
11
|
}
|
|
15
12
|
}
|
|
16
13
|
|
|
17
|
-
function componentToHex(c: number): string {
|
|
18
|
-
return Math.round(Math.max(0, Math.min(255, c))).toString(16).padStart(2, '0')
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
function rgbToHex(r: number, g: number, b: number): string {
|
|
22
|
-
return `#${componentToHex(r)}${componentToHex(g)}${componentToHex(b)}`
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
// Returns hex color with alpha blended onto a white background (for tint derivation)
|
|
26
|
-
export function withAlphaOnWhite(hex: string, alpha: number): string {
|
|
27
|
-
const rgb = hexToRgb(hex)
|
|
28
|
-
if (!rgb) return hex
|
|
29
|
-
const r = rgb.r * alpha + 255 * (1 - alpha)
|
|
30
|
-
const g = rgb.g * alpha + 255 * (1 - alpha)
|
|
31
|
-
const b = rgb.b * alpha + 255 * (1 - alpha)
|
|
32
|
-
return rgbToHex(r, g, b)
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
// Returns hex color with alpha blended onto a dark background (for dark mode tints)
|
|
36
|
-
export function withAlphaOnDark(hex: string, alpha: number, bgHex = '#0f0f0f'): string {
|
|
37
|
-
const rgb = hexToRgb(hex)
|
|
38
|
-
const bg = hexToRgb(bgHex)
|
|
39
|
-
if (!rgb || !bg) return hex
|
|
40
|
-
const r = rgb.r * alpha + bg.r * (1 - alpha)
|
|
41
|
-
const g = rgb.g * alpha + bg.g * (1 - alpha)
|
|
42
|
-
const b = rgb.b * alpha + bg.b * (1 - alpha)
|
|
43
|
-
return rgbToHex(r, g, b)
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
// Mix foreground color with background at given opacity (for text hierarchy)
|
|
47
|
-
export function mixWithBackground(fgHex: string, bgHex: string, opacity: number): string {
|
|
48
|
-
const fg = hexToRgb(fgHex)
|
|
49
|
-
const bg = hexToRgb(bgHex)
|
|
50
|
-
if (!fg || !bg) return fgHex
|
|
51
|
-
const r = fg.r * opacity + bg.r * (1 - opacity)
|
|
52
|
-
const g = fg.g * opacity + bg.g * (1 - opacity)
|
|
53
|
-
const b = fg.b * opacity + bg.b * (1 - opacity)
|
|
54
|
-
return rgbToHex(r, g, b)
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
// Lighten a hex color by mixing with white
|
|
58
|
-
export function lighten(hex: string, amount: number): string {
|
|
59
|
-
return withAlphaOnWhite(hex, 1 - amount)
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
// Darken a hex color by mixing with black
|
|
63
|
-
export function darken(hex: string, amount: number): string {
|
|
64
|
-
const rgb = hexToRgb(hex)
|
|
65
|
-
if (!rgb) return hex
|
|
66
|
-
return rgbToHex(rgb.r * (1 - amount), rgb.g * (1 - amount), rgb.b * (1 - amount))
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
// Detect if a hex color is "dark" (luminance < 0.5)
|
|
70
|
-
export function isDark(hex: string): boolean {
|
|
71
|
-
const rgb = hexToRgb(hex)
|
|
72
|
-
if (!rgb) return false
|
|
73
|
-
// Relative luminance (WCAG formula)
|
|
74
|
-
const toLinear = (c: number) => {
|
|
75
|
-
const s = c / 255
|
|
76
|
-
return s <= 0.03928 ? s / 12.92 : Math.pow((s + 0.055) / 1.055, 2.4)
|
|
77
|
-
}
|
|
78
|
-
const L = 0.2126 * toLinear(rgb.r) + 0.7152 * toLinear(rgb.g) + 0.0722 * toLinear(rgb.b)
|
|
79
|
-
return L < 0.5
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
// Convert a hex color to rgba with the given alpha.
|
|
83
|
-
// Returns an rgba() string suitable for use with semi-transparent backgrounds,
|
|
84
|
-
// borders, and overlays.
|
|
85
14
|
export function withAlpha(hex: string, alpha: number): string {
|
|
86
15
|
const rgb = hexToRgb(hex)
|
|
87
16
|
if (!rgb) return hex
|
package/src/theme/colors.ts
CHANGED
|
@@ -1,5 +1,44 @@
|
|
|
1
1
|
import { ThemeColors, ResolvedColors } from './types'
|
|
2
|
-
import {
|
|
2
|
+
import { hexToRgb } from './colorUtils'
|
|
3
|
+
|
|
4
|
+
function componentToHex(c: number): string {
|
|
5
|
+
return Math.round(Math.max(0, Math.min(255, c))).toString(16).padStart(2, '0')
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
function rgbToHex(r: number, g: number, b: number): string {
|
|
9
|
+
return `#${componentToHex(r)}${componentToHex(g)}${componentToHex(b)}`
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
function withAlphaOnWhite(hex: string, alpha: number): string {
|
|
13
|
+
const rgb = hexToRgb(hex)
|
|
14
|
+
if (!rgb) return hex
|
|
15
|
+
const r = rgb.r * alpha + 255 * (1 - alpha); const g = rgb.g * alpha + 255 * (1 - alpha); const b = rgb.b * alpha + 255 * (1 - alpha)
|
|
16
|
+
return rgbToHex(r, g, b)
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
function withAlphaOnDark(hex: string, alpha: number, bgHex = '#0f0f0f'): string {
|
|
20
|
+
const rgb = hexToRgb(hex); const bg = hexToRgb(bgHex)
|
|
21
|
+
if (!rgb || !bg) return hex
|
|
22
|
+
const r = rgb.r * alpha + bg.r * (1 - alpha); const g = rgb.g * alpha + bg.g * (1 - alpha); const b = rgb.b * alpha + bg.b * (1 - alpha)
|
|
23
|
+
return rgbToHex(r, g, b)
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function mixWithBackground(fgHex: string, bgHex: string, opacity: number): string {
|
|
27
|
+
const fg = hexToRgb(fgHex); const bg = hexToRgb(bgHex)
|
|
28
|
+
if (!fg || !bg) return fgHex
|
|
29
|
+
const r = fg.r * opacity + bg.r * (1 - opacity); const g = fg.g * opacity + bg.g * (1 - opacity); const b = fg.b * opacity + bg.b * (1 - opacity)
|
|
30
|
+
return rgbToHex(r, g, b)
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function lighten(hex: string, amount: number): string {
|
|
34
|
+
return withAlphaOnWhite(hex, 1 - amount)
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function darken(hex: string, amount: number): string {
|
|
38
|
+
const rgb = hexToRgb(hex)
|
|
39
|
+
if (!rgb) return hex
|
|
40
|
+
return rgbToHex(rgb.r * (1 - amount), rgb.g * (1 - amount), rgb.b * (1 - amount))
|
|
41
|
+
}
|
|
3
42
|
|
|
4
43
|
// ─── Default palettes ─────────────────────────────────────────────────────────
|
|
5
44
|
// AUDIT FIXES applied:
|
package/src/theme/types.ts
CHANGED
|
@@ -26,8 +26,8 @@ export type ThemeColors = {
|
|
|
26
26
|
// Derived from ThemeColors. Never supplied by consumers directly.
|
|
27
27
|
export type ResolvedColors = ThemeColors & {
|
|
28
28
|
// Text hierarchy
|
|
29
|
-
foregroundSubtle: string // ~
|
|
30
|
-
foregroundMuted: string // ~
|
|
29
|
+
foregroundSubtle: string // ~70% — body text, subtitles
|
|
30
|
+
foregroundMuted: string // ~62% — captions, timestamps, placeholders
|
|
31
31
|
|
|
32
32
|
// Surface fills (chips unselected, input bg, tag bg)
|
|
33
33
|
surface: string // background slightly off-canvas
|
package/src/utils/animations.ts
CHANGED
|
@@ -1,74 +1,29 @@
|
|
|
1
|
-
import { Easing } from 'react-native-reanimated'
|
|
2
1
|
import type { SingleTransition } from 'react-native-ease'
|
|
3
2
|
|
|
4
|
-
// ─── Spring presets ──────────────────────────────────────────────────────────
|
|
5
|
-
// Tuned for the "Apple HIG / Airbnb" press-feel: snap inward fast, settle out elastically.
|
|
6
|
-
// `stiffness`/`damping`/`mass` model — Reanimated v4 default physics units.
|
|
7
|
-
//
|
|
8
|
-
// pressIn: high stiffness, heavy damping → fast, controlled compression
|
|
9
|
-
// pressOut: lower stiffness, less damping → soft, elastic rebound
|
|
10
|
-
// settle: pillows / drawers / large surfaces → calm, never twitchy
|
|
11
3
|
export const SPRINGS = {
|
|
12
|
-
/** Tight, premium press feel — Buttons, Toggle, Tabs triggers. */
|
|
13
|
-
pressIn: { stiffness: 600, damping: 35, mass: 0.8 },
|
|
14
|
-
pressOut: { stiffness: 280, damping: 22, mass: 0.8 },
|
|
15
|
-
|
|
16
|
-
/** Slightly softer for larger surfaces — Card, ListItem, MenuItem. */
|
|
17
|
-
surfacePressIn: { stiffness: 380, damping: 30, mass: 0.95 },
|
|
18
|
-
surfacePressOut: { stiffness: 220, damping: 20, mass: 0.95 },
|
|
19
|
-
|
|
20
|
-
/** Settled transitions for moving indicators — Tabs pill, Switch thumb. */
|
|
21
4
|
glide: { stiffness: 380, damping: 38, mass: 1 },
|
|
22
|
-
|
|
23
|
-
/** Elastic indicator — Switch thumb, RadioGroup dot. */
|
|
24
5
|
elastic: { stiffness: 320, damping: 22, mass: 0.7 },
|
|
25
6
|
} as const
|
|
26
7
|
|
|
27
|
-
// ─── Timing presets ──────────────────────────────────────────────────────────
|
|
28
|
-
// All timings target the UI thread via Reanimated `withTiming`.
|
|
29
8
|
export const TIMINGS = {
|
|
30
|
-
/** Color/opacity transitions on toggles, checkboxes, switches. */
|
|
31
9
|
state: { duration: 160 },
|
|
32
|
-
/** Focus ring on inputs. */
|
|
33
|
-
focusIn: { duration: 140 },
|
|
34
|
-
focusOut: { duration: 100 },
|
|
35
|
-
/** Accordion / collapsible content. */
|
|
36
10
|
expand: { duration: 240 },
|
|
37
11
|
collapse: { duration: 200 },
|
|
38
|
-
/** Skeleton shimmer cycle (full pass). */
|
|
39
12
|
shimmer: { duration: 1400 },
|
|
40
13
|
} as const
|
|
41
14
|
|
|
42
|
-
// ─── Easing presets ──────────────────────────────────────────────────────────
|
|
43
|
-
export const EASINGS = {
|
|
44
|
-
/** Material-style ease-out — natural deceleration for state changes. */
|
|
45
|
-
standard: Easing.bezier(0.2, 0, 0, 1),
|
|
46
|
-
/** Strong ease-out for expanding surfaces (Accordion open). */
|
|
47
|
-
expand: Easing.bezier(0.23, 1, 0.32, 1),
|
|
48
|
-
/** Quick ease-in for collapsing. */
|
|
49
|
-
collapse: Easing.in(Easing.ease),
|
|
50
|
-
} as const
|
|
51
|
-
|
|
52
|
-
// ─── EaseView transition presets ─────────────────────────────────────────────
|
|
53
|
-
// Equivalents of the reanimated presets above, in `react-native-ease` units.
|
|
54
|
-
// EaseView spring takes raw damping/stiffness/mass (same physical model). EaseView
|
|
55
|
-
// timing takes duration + an easing curve as a cubic-bezier tuple.
|
|
56
|
-
|
|
57
|
-
/** Color/border state transition for Toggle, Checkbox, Chip, CategoryStrip, Switch track. Mirrors TIMINGS.state + EASINGS.standard. */
|
|
58
15
|
export const COLOR_TRANSITION: SingleTransition = {
|
|
59
16
|
type: 'timing',
|
|
60
17
|
duration: TIMINGS.state.duration,
|
|
61
18
|
easing: [0.2, 0, 0, 1],
|
|
62
19
|
}
|
|
63
20
|
|
|
64
|
-
/** Icon/opacity crossfade. Mirrors TIMINGS.state + EASINGS.standard. */
|
|
65
21
|
export const OPACITY_TRANSITION: SingleTransition = {
|
|
66
22
|
type: 'timing',
|
|
67
23
|
duration: TIMINGS.state.duration,
|
|
68
24
|
easing: [0.2, 0, 0, 1],
|
|
69
25
|
}
|
|
70
26
|
|
|
71
|
-
/** Elastic indicator spring — Switch thumb, RadioGroup dot. Mirrors SPRINGS.elastic. */
|
|
72
27
|
export const SPRING_ELASTIC: SingleTransition = {
|
|
73
28
|
type: 'spring',
|
|
74
29
|
stiffness: 320,
|
|
@@ -76,8 +31,6 @@ export const SPRING_ELASTIC: SingleTransition = {
|
|
|
76
31
|
mass: 0.7,
|
|
77
32
|
}
|
|
78
33
|
|
|
79
|
-
// ─── Press scale tokens ──────────────────────────────────────────────────────
|
|
80
|
-
// Per-component press intensities — taken from DESIGN.md.
|
|
81
34
|
export const PRESS_SCALE = {
|
|
82
35
|
button: 0.95,
|
|
83
36
|
card: 0.98,
|