@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.
Files changed (268) hide show
  1. package/COMPONENTS.md +85 -143
  2. package/CONSUMER.md +2 -2
  3. package/DESIGN.md +2 -2
  4. package/README.md +11 -6
  5. package/dist/Accordion.js +48 -210
  6. package/dist/Accordion.mjs +6 -5
  7. package/dist/AlertBanner.js +29 -153
  8. package/dist/AlertBanner.mjs +3 -3
  9. package/dist/AppHeader.js +37 -235
  10. package/dist/AppHeader.mjs +6 -7
  11. package/dist/Avatar.d.mts +17 -1
  12. package/dist/Avatar.d.ts +17 -1
  13. package/dist/Avatar.js +80 -115
  14. package/dist/Avatar.mjs +2 -2
  15. package/dist/Badge.js +24 -149
  16. package/dist/Badge.mjs +3 -3
  17. package/dist/Button.js +79 -267
  18. package/dist/Button.mjs +6 -6
  19. package/dist/Card.js +15 -200
  20. package/dist/Card.mjs +4 -5
  21. package/dist/CategoryStrip.d.mts +0 -5
  22. package/dist/CategoryStrip.d.ts +0 -5
  23. package/dist/CategoryStrip.js +47 -265
  24. package/dist/CategoryStrip.mjs +6 -6
  25. package/dist/Checkbox.js +15 -200
  26. package/dist/Checkbox.mjs +5 -5
  27. package/dist/Chip.js +44 -236
  28. package/dist/Chip.mjs +7 -6
  29. package/dist/ConfirmDialog.js +84 -286
  30. package/dist/ConfirmDialog.mjs +7 -7
  31. package/dist/CurrencyDisplay.js +1 -114
  32. package/dist/CurrencyDisplay.mjs +2 -2
  33. package/dist/CurrencyInput.js +35 -162
  34. package/dist/CurrencyInput.mjs +5 -5
  35. package/dist/DetailRow.js +25 -150
  36. package/dist/DetailRow.mjs +3 -3
  37. package/dist/EmptyState.js +80 -268
  38. package/dist/EmptyState.mjs +7 -7
  39. package/dist/ErrorBoundary.js +32 -199
  40. package/dist/ErrorBoundary.mjs +4 -4
  41. package/dist/Form.js +1 -114
  42. package/dist/Form.mjs +2 -2
  43. package/dist/HolographicCard.d.mts +0 -28
  44. package/dist/HolographicCard.d.ts +0 -28
  45. package/dist/HolographicCard.js +20 -130
  46. package/dist/HolographicCard.mjs +9 -32
  47. package/dist/IconButton.js +36 -234
  48. package/dist/IconButton.mjs +5 -6
  49. package/dist/IconPicker.js +222 -929
  50. package/dist/IconPicker.mjs +5 -5
  51. package/dist/ImageUpload.d.mts +3 -1
  52. package/dist/ImageUpload.d.ts +3 -1
  53. package/dist/ImageUpload.js +25 -215
  54. package/dist/ImageUpload.mjs +5 -6
  55. package/dist/ImageViewer.js +75 -266
  56. package/dist/ImageViewer.mjs +8 -8
  57. package/dist/Input.d.mts +1 -1
  58. package/dist/Input.d.ts +1 -1
  59. package/dist/Input.js +35 -162
  60. package/dist/Input.mjs +4 -4
  61. package/dist/LabelValue.js +24 -149
  62. package/dist/LabelValue.mjs +3 -3
  63. package/dist/ListGroup.js +1 -114
  64. package/dist/ListGroup.mjs +2 -2
  65. package/dist/ListItem.js +38 -235
  66. package/dist/ListItem.mjs +5 -6
  67. package/dist/MediaCard.d.mts +0 -14
  68. package/dist/MediaCard.d.ts +0 -14
  69. package/dist/MediaCard.js +69 -315
  70. package/dist/MediaCard.mjs +5 -6
  71. package/dist/MenuGroup.js +1 -114
  72. package/dist/MenuGroup.mjs +2 -2
  73. package/dist/MenuItem.js +36 -234
  74. package/dist/MenuItem.mjs +5 -6
  75. package/dist/MonthPicker.js +8 -163
  76. package/dist/MonthPicker.mjs +3 -3
  77. package/dist/NumberStepper.js +40 -238
  78. package/dist/NumberStepper.mjs +5 -6
  79. package/dist/PagerDots.d.mts +1 -1
  80. package/dist/PagerDots.d.ts +1 -1
  81. package/dist/PagerDots.js +69 -224
  82. package/dist/PagerDots.mjs +6 -5
  83. package/dist/Pressable.js +14 -85
  84. package/dist/Pressable.mjs +4 -4
  85. package/dist/PricingCard.js +87 -272
  86. package/dist/PricingCard.mjs +8 -8
  87. package/dist/Progress.js +3 -123
  88. package/dist/Progress.mjs +3 -3
  89. package/dist/RadioGroup.js +52 -265
  90. package/dist/RadioGroup.mjs +5 -5
  91. package/dist/RetrayProvider.js +3 -6
  92. package/dist/RetrayProvider.mjs +3 -3
  93. package/dist/Select.d.mts +2 -1
  94. package/dist/Select.d.ts +2 -1
  95. package/dist/Select.js +24 -232
  96. package/dist/Select.mjs +4 -5
  97. package/dist/SelectableCard.js +33 -209
  98. package/dist/SelectableCard.mjs +5 -5
  99. package/dist/SelectableGrid.d.mts +0 -21
  100. package/dist/SelectableGrid.d.ts +0 -21
  101. package/dist/SelectableGrid.js +49 -271
  102. package/dist/SelectableGrid.mjs +5 -6
  103. package/dist/Separator.js +1 -114
  104. package/dist/Separator.mjs +2 -2
  105. package/dist/Sheet.js +7 -162
  106. package/dist/Sheet.mjs +3 -3
  107. package/dist/SheetSelect.js +39 -236
  108. package/dist/SheetSelect.mjs +6 -6
  109. package/dist/Skeleton.js +4 -124
  110. package/dist/Skeleton.mjs +3 -3
  111. package/dist/Slider.js +6 -161
  112. package/dist/Slider.mjs +3 -3
  113. package/dist/Spinner.js +3 -116
  114. package/dist/Spinner.mjs +2 -2
  115. package/dist/Stats.js +36 -234
  116. package/dist/Stats.mjs +5 -6
  117. package/dist/Switch.js +24 -175
  118. package/dist/Switch.mjs +5 -4
  119. package/dist/TabBar.js +43 -200
  120. package/dist/TabBar.mjs +5 -4
  121. package/dist/Tabs.js +15 -199
  122. package/dist/Tabs.mjs +5 -5
  123. package/dist/Text.js +9 -130
  124. package/dist/Text.mjs +2 -2
  125. package/dist/Textarea.d.mts +2 -1
  126. package/dist/Textarea.d.ts +2 -1
  127. package/dist/Textarea.js +71 -219
  128. package/dist/Textarea.mjs +4 -4
  129. package/dist/Toast.js +1 -114
  130. package/dist/Toast.mjs +2 -2
  131. package/dist/Toggle.js +39 -236
  132. package/dist/Toggle.mjs +6 -6
  133. package/dist/{chunk-M3C7XM2M.mjs → chunk-2QOHHBJC.mjs} +3 -3
  134. package/dist/{chunk-LIS6I5UP.mjs → chunk-2VIDP72N.mjs} +3 -3
  135. package/dist/{chunk-DF7JA72E.mjs → chunk-4NQFTHN3.mjs} +13 -7
  136. package/dist/{chunk-UBUXUMER.mjs → chunk-4ZO5PTKF.mjs} +4 -4
  137. package/dist/{chunk-3XCFYSX4.mjs → chunk-5MYNAAFE.mjs} +13 -17
  138. package/dist/{chunk-E7NEHHXV.mjs → chunk-62BBSSUF.mjs} +3 -3
  139. package/dist/{chunk-MVMGPZN6.mjs → chunk-6CR4S6W2.mjs} +3 -3
  140. package/dist/{chunk-EDLCGYIO.mjs → chunk-6QLBHUEG.mjs} +8 -7
  141. package/dist/chunk-ARONDO7M.mjs +40 -0
  142. package/dist/{chunk-GH67YXG6.mjs → chunk-AZV7KNJI.mjs} +3 -3
  143. package/dist/{chunk-RMRS44MQ.mjs → chunk-BTUW5LSG.mjs} +11 -8
  144. package/dist/{chunk-2P2CB235.mjs → chunk-BULKGOIZ.mjs} +7 -8
  145. package/dist/{chunk-NHDI3VQB.mjs → chunk-CBIZLRYH.mjs} +15 -12
  146. package/dist/chunk-CM2DG4MR.mjs +142 -0
  147. package/dist/{chunk-TS7DGUIR.mjs → chunk-DBHSUUKU.mjs} +2 -2
  148. package/dist/{chunk-57V2LXCK.mjs → chunk-DE25XTVQ.mjs} +3 -3
  149. package/dist/{chunk-UQ4742ET.mjs → chunk-E4EQSCKR.mjs} +5 -5
  150. package/dist/{chunk-GUTDFUNF.mjs → chunk-EHGBHFMH.mjs} +9 -17
  151. package/dist/{chunk-CF27NBXO.mjs → chunk-EROPDCB5.mjs} +16 -24
  152. package/dist/{chunk-ZIMY2QUM.mjs → chunk-ERWJPVX7.mjs} +2 -2
  153. package/dist/{chunk-NLZY4TXU.mjs → chunk-ESQDPO5E.mjs} +7 -7
  154. package/dist/{chunk-VJBUCITV.mjs → chunk-EW2FIDSM.mjs} +1 -1
  155. package/dist/{chunk-HC4VVCWY.mjs → chunk-FTTI6T5Q.mjs} +4 -4
  156. package/dist/{chunk-2HFD4IHU.mjs → chunk-HUSSF6TF.mjs} +1 -1
  157. package/dist/chunk-IFYMBOEN.mjs +14 -0
  158. package/dist/{chunk-QOLWA2PW.mjs → chunk-IGU223UM.mjs} +80 -4
  159. package/dist/chunk-IJCMPVW5.mjs +121 -0
  160. package/dist/{chunk-AENAVIKT.mjs → chunk-ITG4JQM3.mjs} +4 -4
  161. package/dist/{chunk-E5UKLSJZ.mjs → chunk-K3QX2M26.mjs} +11 -8
  162. package/dist/{chunk-4OORJ2DY.mjs → chunk-K7TKID3V.mjs} +8 -7
  163. package/dist/{chunk-2LG326TT.mjs → chunk-KAGADD2O.mjs} +4 -4
  164. package/dist/{chunk-IVSRW4HS.mjs → chunk-KC5QDYGZ.mjs} +4 -4
  165. package/dist/{chunk-7AFZWSCI.mjs → chunk-KPTY7UYQ.mjs} +1 -1
  166. package/dist/{chunk-YTXRIXNZ.mjs → chunk-KSSVIFYR.mjs} +9 -12
  167. package/dist/chunk-L3YKPTJQ.mjs +119 -0
  168. package/dist/chunk-M53LC4Q7.mjs +35 -0
  169. package/dist/{chunk-ZR6HSEAB.mjs → chunk-MP7GLMIR.mjs} +17 -25
  170. package/dist/chunk-MZ6WRTD2.mjs +40 -0
  171. package/dist/chunk-NGEN2EES.mjs +581 -0
  172. package/dist/{chunk-C43HRKXH.mjs → chunk-OBV72JD4.mjs} +1 -1
  173. package/dist/{chunk-LPV4NJJK.mjs → chunk-PGQ6FMXS.mjs} +6 -5
  174. package/dist/{chunk-MEPSKGBO.mjs → chunk-PI6RULJX.mjs} +1 -1
  175. package/dist/{chunk-F3YTWO3T.mjs → chunk-RA6SAAFE.mjs} +9 -8
  176. package/dist/{chunk-UNNRUJTM.mjs → chunk-RRKM4MKB.mjs} +7 -7
  177. package/dist/{chunk-ULGNQPNE.mjs → chunk-S2VGME7X.mjs} +1 -1
  178. package/dist/{chunk-OLVJFKXS.mjs → chunk-S44XWTTC.mjs} +35 -25
  179. package/dist/{chunk-YMYIEVZP.mjs → chunk-SZEKQAOY.mjs} +1 -1
  180. package/dist/{chunk-ELGEOM7I.mjs → chunk-TETMEKZE.mjs} +9 -9
  181. package/dist/{chunk-BXF4AMHY.mjs → chunk-TMH263OK.mjs} +5 -4
  182. package/dist/{chunk-NJG7DHVF.mjs → chunk-U6DEBYU5.mjs} +10 -9
  183. package/dist/{chunk-RJNLAH76.mjs → chunk-UOKFSFNJ.mjs} +2 -2
  184. package/dist/{chunk-HEDQPK4I.mjs → chunk-URIH43IJ.mjs} +13 -21
  185. package/dist/{chunk-QXDGGOLC.mjs → chunk-V2ZB2XNS.mjs} +6 -6
  186. package/dist/{chunk-KSUWPU2F.mjs → chunk-WIPEDNSD.mjs} +7 -7
  187. package/dist/{chunk-QDAZGZUF.mjs → chunk-XCIG6HT2.mjs} +3 -3
  188. package/dist/{chunk-4J2PXL36.mjs → chunk-Y6YS33GM.mjs} +40 -38
  189. package/dist/{chunk-4XOB5TTD.mjs → chunk-ZKDKKQCE.mjs} +5 -5
  190. package/dist/{chunk-LOBLCFMN.mjs → chunk-ZTPYUU5C.mjs} +5 -5
  191. package/dist/index.d.mts +12 -72
  192. package/dist/index.d.ts +12 -72
  193. package/dist/index.js +1051 -1838
  194. package/dist/index.mjs +81 -85
  195. package/package.json +8 -10
  196. package/src/components/Accordion/Accordion.tsx +12 -9
  197. package/src/components/AlertBanner/AlertBanner.tsx +7 -6
  198. package/src/components/AppHeader/AppHeader.tsx +1 -1
  199. package/src/components/Avatar/Avatar.tsx +92 -1
  200. package/src/components/Avatar/index.ts +2 -2
  201. package/src/components/Badge/Badge.tsx +2 -2
  202. package/src/components/Button/Button.tsx +50 -46
  203. package/src/components/Card/Card.tsx +1 -0
  204. package/src/components/CategoryStrip/CategoryStrip.tsx +36 -49
  205. package/src/components/Chip/Chip.tsx +5 -4
  206. package/src/components/ConfirmDialog/ConfirmDialog.tsx +3 -3
  207. package/src/components/DetailRow/DetailRow.tsx +3 -3
  208. package/src/components/EmptyState/EmptyState.tsx +2 -2
  209. package/src/components/ErrorBoundary/ErrorBoundary.tsx +6 -6
  210. package/src/components/HolographicCard/HolographicCard.tsx +14 -95
  211. package/src/components/IconButton/IconButton.tsx +2 -2
  212. package/src/components/IconPicker/IconPicker.tsx +13 -12
  213. package/src/components/ImageUpload/ImageUpload.tsx +14 -25
  214. package/src/components/ImageViewer/ImageViewer.tsx +3 -3
  215. package/src/components/Input/Input.tsx +11 -5
  216. package/src/components/LabelValue/LabelValue.tsx +2 -2
  217. package/src/components/ListItem/ListItem.tsx +4 -4
  218. package/src/components/MediaCard/MediaCard.tsx +21 -59
  219. package/src/components/MenuItem/MenuItem.tsx +2 -2
  220. package/src/components/MonthPicker/MonthPicker.tsx +2 -2
  221. package/src/components/NumberStepper/NumberStepper.tsx +6 -6
  222. package/src/components/PagerDots/PagerDots.tsx +38 -28
  223. package/src/components/PricingCard/PricingCard.tsx +6 -6
  224. package/src/components/RadioGroup/RadioGroup.tsx +18 -31
  225. package/src/components/Select/Select.tsx +32 -39
  226. package/src/components/SelectableCard/SelectableCard.tsx +4 -6
  227. package/src/components/SelectableGrid/SelectableGrid.tsx +38 -72
  228. package/src/components/Sheet/Sheet.tsx +1 -1
  229. package/src/components/SheetSelect/SheetSelect.tsx +3 -3
  230. package/src/components/Skeleton/Skeleton.tsx +1 -1
  231. package/src/components/Spinner/Spinner.tsx +2 -2
  232. package/src/components/Stats/Stats.tsx +2 -2
  233. package/src/components/Switch/Switch.tsx +9 -6
  234. package/src/components/TabBar/TabBar.tsx +9 -8
  235. package/src/components/Text/Text.tsx +12 -1
  236. package/src/components/Textarea/Textarea.tsx +18 -32
  237. package/src/components/Toggle/Toggle.tsx +3 -3
  238. package/src/hooks/useConfirmDialog.ts +31 -42
  239. package/src/index.ts +3 -4
  240. package/src/theme/ThemeProvider.tsx +1 -4
  241. package/src/theme/colorUtils.ts +1 -72
  242. package/src/theme/colors.ts +40 -1
  243. package/src/theme/types.ts +2 -2
  244. package/src/utils/animations.ts +0 -47
  245. package/src/utils/curatedIcons.ts +93 -801
  246. package/src/utils/haptics.ts +13 -208
  247. package/src/utils/icons.ts +27 -91
  248. package/src/utils/pressable.ts +10 -61
  249. package/dist/VirtualList.d.mts +0 -19
  250. package/dist/VirtualList.d.ts +0 -19
  251. package/dist/VirtualList.js +0 -38
  252. package/dist/VirtualList.mjs +0 -2
  253. package/dist/chunk-2BA3JMKK.mjs +0 -136
  254. package/dist/chunk-3DKJ2GIC.mjs +0 -30
  255. package/dist/chunk-7ELGZ66G.mjs +0 -164
  256. package/dist/chunk-DVK4G2GT.mjs +0 -59
  257. package/dist/chunk-EJ7ZPXOH.mjs +0 -163
  258. package/dist/chunk-KA7LTET3.mjs +0 -71
  259. package/dist/chunk-LNPKGWBG.mjs +0 -134
  260. package/dist/chunk-NC5ZTR2Y.mjs +0 -32
  261. package/dist/chunk-SAWUXP3A.mjs +0 -1114
  262. package/dist/chunk-YNROWHQJ.mjs +0 -46
  263. package/src/components/VirtualList/VirtualList.tsx +0 -60
  264. package/src/components/VirtualList/index.ts +0 -1
  265. package/src/utils/fontGuard.ts +0 -35
  266. package/src/utils/hover.ts +0 -25
  267. package/src/utils/useColorTransition.ts +0 -40
  268. 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<T> {
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
- /** Open the dialog, optionally with an associated value (e.g. the item to delete). */
16
- open: (target?: T) => void
17
- /** Handlers to pass directly to ConfirmDialog. */
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<T = undefined>(
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
- const open = useCallback((t?: T) => {
34
- setTarget(t ?? null)
35
- setVisible(true)
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 options.onConfirm()
39
+ await onConfirmRef.current()
40
+ } catch {
41
+ /* consumer handles error in onConfirm */
42
42
  } finally {
43
- setLoading(false)
44
- setVisible(false)
45
- setTarget(null)
43
+ if (mountedRef.current) {
44
+ setLoading(false)
45
+ setVisible(false)
46
+ }
46
47
  }
47
- }, [options])
48
+ }, [])
48
49
 
49
50
  const handleCancel = useCallback(() => {
50
51
  setVisible(false)
51
- setTarget(null)
52
- options.onCancel?.()
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, renderIcon, configureIconFamilies, getValidIconNames } from './utils/icons'
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
@@ -1,7 +1,4 @@
1
- // Hex color manipulation utilities for internal theme derivation.
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
@@ -1,5 +1,44 @@
1
1
  import { ThemeColors, ResolvedColors } from './types'
2
- import { mixWithBackground, withAlphaOnWhite, withAlphaOnDark, lighten, darken } from './colorUtils'
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:
@@ -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 // ~55% — body text, subtitles
30
- foregroundMuted: string // ~35% — captions, timestamps, placeholders
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
@@ -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,