@retray-dev/ui-kit 12.2.0 → 13.2.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 (296) hide show
  1. package/CONSUMER.md +26 -11
  2. package/DESIGN.md +2 -2
  3. package/README.md +15 -11
  4. package/{COMPONENTS.md → SKILL.md} +374 -996
  5. package/dist/Accordion.d.mts +2 -0
  6. package/dist/Accordion.d.ts +2 -0
  7. package/dist/Accordion.js +49 -210
  8. package/dist/Accordion.mjs +6 -6
  9. package/dist/AlertBanner.js +29 -153
  10. package/dist/AlertBanner.mjs +3 -4
  11. package/dist/AppHeader.d.mts +5 -2
  12. package/dist/AppHeader.d.ts +5 -2
  13. package/dist/AppHeader.js +45 -239
  14. package/dist/AppHeader.mjs +6 -8
  15. package/dist/Avatar.d.mts +17 -1
  16. package/dist/Avatar.d.ts +17 -1
  17. package/dist/Avatar.js +80 -115
  18. package/dist/Avatar.mjs +2 -3
  19. package/dist/Badge.js +24 -149
  20. package/dist/Badge.mjs +3 -4
  21. package/dist/Button.js +79 -267
  22. package/dist/Button.mjs +6 -7
  23. package/dist/ButtonGroup.mjs +0 -1
  24. package/dist/Card.js +15 -200
  25. package/dist/Card.mjs +4 -6
  26. package/dist/CategoryStrip.d.mts +0 -5
  27. package/dist/CategoryStrip.d.ts +0 -5
  28. package/dist/CategoryStrip.js +47 -265
  29. package/dist/CategoryStrip.mjs +6 -7
  30. package/dist/Checkbox.d.mts +2 -1
  31. package/dist/Checkbox.d.ts +2 -1
  32. package/dist/Checkbox.js +18 -201
  33. package/dist/Checkbox.mjs +5 -6
  34. package/dist/Chip.js +44 -236
  35. package/dist/Chip.mjs +7 -7
  36. package/dist/ConfirmDialog.d.mts +2 -1
  37. package/dist/ConfirmDialog.d.ts +2 -1
  38. package/dist/ConfirmDialog.js +110 -300
  39. package/dist/ConfirmDialog.mjs +7 -8
  40. package/dist/CurrencyDisplay.js +1 -114
  41. package/dist/CurrencyDisplay.mjs +2 -3
  42. package/dist/CurrencyInput.js +35 -162
  43. package/dist/CurrencyInput.mjs +5 -6
  44. package/dist/DetailRow.js +25 -150
  45. package/dist/DetailRow.mjs +3 -4
  46. package/dist/EmptyState.js +80 -268
  47. package/dist/EmptyState.mjs +7 -8
  48. package/dist/ErrorBoundary.js +32 -199
  49. package/dist/ErrorBoundary.mjs +4 -5
  50. package/dist/Form.js +1 -114
  51. package/dist/Form.mjs +2 -3
  52. package/dist/HolographicCard.d.mts +0 -28
  53. package/dist/HolographicCard.d.ts +0 -28
  54. package/dist/HolographicCard.js +20 -130
  55. package/dist/HolographicCard.mjs +9 -33
  56. package/dist/IconButton.js +36 -234
  57. package/dist/IconButton.mjs +5 -7
  58. package/dist/IconPicker.js +222 -929
  59. package/dist/IconPicker.mjs +5 -6
  60. package/dist/ImageUpload.d.mts +3 -3
  61. package/dist/ImageUpload.d.ts +3 -3
  62. package/dist/ImageUpload.js +49 -238
  63. package/dist/ImageUpload.mjs +5 -7
  64. package/dist/ImageViewer.js +75 -266
  65. package/dist/ImageViewer.mjs +8 -9
  66. package/dist/Input.d.mts +1 -1
  67. package/dist/Input.d.ts +1 -1
  68. package/dist/Input.js +35 -162
  69. package/dist/Input.mjs +4 -5
  70. package/dist/LabelValue.js +24 -149
  71. package/dist/LabelValue.mjs +3 -4
  72. package/dist/ListGroup.js +1 -114
  73. package/dist/ListGroup.mjs +2 -3
  74. package/dist/ListItem.d.mts +2 -1
  75. package/dist/ListItem.d.ts +2 -1
  76. package/dist/ListItem.js +41 -236
  77. package/dist/ListItem.mjs +5 -7
  78. package/dist/MediaCard.d.mts +0 -14
  79. package/dist/MediaCard.d.ts +0 -14
  80. package/dist/MediaCard.js +69 -315
  81. package/dist/MediaCard.mjs +5 -7
  82. package/dist/MenuGroup.js +1 -114
  83. package/dist/MenuGroup.mjs +2 -3
  84. package/dist/MenuItem.d.mts +2 -1
  85. package/dist/MenuItem.d.ts +2 -1
  86. package/dist/MenuItem.js +39 -235
  87. package/dist/MenuItem.mjs +5 -7
  88. package/dist/MonthPicker.js +8 -163
  89. package/dist/MonthPicker.mjs +3 -4
  90. package/dist/NumberStepper.d.mts +2 -1
  91. package/dist/NumberStepper.d.ts +2 -1
  92. package/dist/NumberStepper.js +44 -239
  93. package/dist/NumberStepper.mjs +5 -7
  94. package/dist/PagerDots.d.mts +1 -1
  95. package/dist/PagerDots.d.ts +1 -1
  96. package/dist/PagerDots.js +69 -224
  97. package/dist/PagerDots.mjs +6 -6
  98. package/dist/Pressable.js +14 -85
  99. package/dist/Pressable.mjs +4 -5
  100. package/dist/PricingCard.js +87 -272
  101. package/dist/PricingCard.mjs +8 -9
  102. package/dist/Progress.js +3 -123
  103. package/dist/Progress.mjs +3 -4
  104. package/dist/RadioGroup.js +52 -265
  105. package/dist/RadioGroup.mjs +5 -6
  106. package/dist/RetrayProvider.js +3 -6
  107. package/dist/RetrayProvider.mjs +3 -4
  108. package/dist/Select.d.mts +3 -1
  109. package/dist/Select.d.ts +3 -1
  110. package/dist/Select.js +27 -233
  111. package/dist/Select.mjs +4 -6
  112. package/dist/SelectableCard.js +33 -209
  113. package/dist/SelectableCard.mjs +5 -6
  114. package/dist/SelectableGrid.d.mts +0 -21
  115. package/dist/SelectableGrid.d.ts +0 -21
  116. package/dist/SelectableGrid.js +49 -272
  117. package/dist/SelectableGrid.mjs +5 -7
  118. package/dist/Separator.js +1 -114
  119. package/dist/Separator.mjs +2 -3
  120. package/dist/Sheet.d.mts +1 -1
  121. package/dist/Sheet.d.ts +1 -1
  122. package/dist/Sheet.js +33 -175
  123. package/dist/Sheet.mjs +3 -4
  124. package/dist/SheetSelect.js +39 -236
  125. package/dist/SheetSelect.mjs +6 -7
  126. package/dist/Skeleton.js +4 -124
  127. package/dist/Skeleton.mjs +3 -4
  128. package/dist/Slider.d.mts +2 -1
  129. package/dist/Slider.d.ts +2 -1
  130. package/dist/Slider.js +8 -161
  131. package/dist/Slider.mjs +3 -4
  132. package/dist/Spinner.js +3 -116
  133. package/dist/Spinner.mjs +2 -3
  134. package/dist/Stats.js +36 -234
  135. package/dist/Stats.mjs +5 -7
  136. package/dist/Switch.d.mts +2 -1
  137. package/dist/Switch.d.ts +2 -1
  138. package/dist/Switch.js +26 -176
  139. package/dist/Switch.mjs +5 -5
  140. package/dist/TabBar.js +43 -200
  141. package/dist/TabBar.mjs +5 -5
  142. package/dist/Tabs.js +15 -199
  143. package/dist/Tabs.mjs +5 -6
  144. package/dist/Text.js +9 -130
  145. package/dist/Text.mjs +2 -3
  146. package/dist/Textarea.d.mts +2 -1
  147. package/dist/Textarea.d.ts +2 -1
  148. package/dist/Textarea.js +71 -219
  149. package/dist/Textarea.mjs +4 -5
  150. package/dist/Toast.d.mts +12 -10
  151. package/dist/Toast.d.ts +12 -10
  152. package/dist/Toast.js +1 -114
  153. package/dist/Toast.mjs +2 -3
  154. package/dist/Toggle.js +39 -236
  155. package/dist/Toggle.mjs +6 -7
  156. package/dist/{chunk-ELGEOM7I.mjs → chunk-2QXJDRVU.mjs} +13 -10
  157. package/dist/{chunk-LIS6I5UP.mjs → chunk-2VIDP72N.mjs} +3 -3
  158. package/dist/{chunk-NHDI3VQB.mjs → chunk-422IVD3H.mjs} +16 -12
  159. package/dist/{chunk-DF7JA72E.mjs → chunk-4NQFTHN3.mjs} +13 -7
  160. package/dist/{chunk-3XCFYSX4.mjs → chunk-5MYNAAFE.mjs} +13 -17
  161. package/dist/{chunk-E7NEHHXV.mjs → chunk-62BBSSUF.mjs} +3 -3
  162. package/dist/{chunk-UBUXUMER.mjs → chunk-77UOVFIS.mjs} +7 -5
  163. package/dist/{chunk-M3C7XM2M.mjs → chunk-7BZJRB77.mjs} +28 -18
  164. package/dist/chunk-ARONDO7M.mjs +40 -0
  165. package/dist/{chunk-GH67YXG6.mjs → chunk-AZV7KNJI.mjs} +3 -3
  166. package/dist/{chunk-2P2CB235.mjs → chunk-BULKGOIZ.mjs} +7 -8
  167. package/dist/{chunk-RJNLAH76.mjs → chunk-C5ZRMR2E.mjs} +4 -2
  168. package/dist/chunk-CM2DG4MR.mjs +142 -0
  169. package/dist/{chunk-UQ4742ET.mjs → chunk-COA2YZOX.mjs} +8 -6
  170. package/dist/{chunk-EDLCGYIO.mjs → chunk-CZN6L2QU.mjs} +11 -8
  171. package/dist/{chunk-TS7DGUIR.mjs → chunk-DBHSUUKU.mjs} +2 -2
  172. package/dist/{chunk-57V2LXCK.mjs → chunk-DE25XTVQ.mjs} +3 -3
  173. package/dist/{chunk-RMRS44MQ.mjs → chunk-E2PONRJG.mjs} +13 -9
  174. package/dist/{chunk-GUTDFUNF.mjs → chunk-EHGBHFMH.mjs} +9 -17
  175. package/dist/{chunk-ZIMY2QUM.mjs → chunk-ERWJPVX7.mjs} +2 -2
  176. package/dist/{chunk-NLZY4TXU.mjs → chunk-ESQDPO5E.mjs} +7 -7
  177. package/dist/{chunk-VJBUCITV.mjs → chunk-EW2FIDSM.mjs} +1 -1
  178. package/dist/{chunk-HC4VVCWY.mjs → chunk-FTTI6T5Q.mjs} +4 -4
  179. package/dist/{chunk-MVMGPZN6.mjs → chunk-H6MQL7PS.mjs} +12 -7
  180. package/dist/{chunk-CF27NBXO.mjs → chunk-HHOOFDBA.mjs} +38 -41
  181. package/dist/{chunk-2HFD4IHU.mjs → chunk-HUSSF6TF.mjs} +1 -1
  182. package/dist/{chunk-HEDQPK4I.mjs → chunk-IDVUZIVY.mjs} +16 -22
  183. package/dist/chunk-IFYMBOEN.mjs +14 -0
  184. package/dist/{chunk-QOLWA2PW.mjs → chunk-IGU223UM.mjs} +80 -4
  185. package/dist/chunk-IJCMPVW5.mjs +121 -0
  186. package/dist/{chunk-AENAVIKT.mjs → chunk-ITG4JQM3.mjs} +4 -4
  187. package/dist/{chunk-E5UKLSJZ.mjs → chunk-K3QX2M26.mjs} +11 -8
  188. package/dist/{chunk-4OORJ2DY.mjs → chunk-K7TKID3V.mjs} +8 -7
  189. package/dist/{chunk-2LG326TT.mjs → chunk-KAGADD2O.mjs} +4 -4
  190. package/dist/{chunk-IVSRW4HS.mjs → chunk-KC5QDYGZ.mjs} +4 -4
  191. package/dist/{chunk-7AFZWSCI.mjs → chunk-KPTY7UYQ.mjs} +1 -1
  192. package/dist/{chunk-YTXRIXNZ.mjs → chunk-KSSVIFYR.mjs} +9 -12
  193. package/dist/chunk-L3YKPTJQ.mjs +119 -0
  194. package/dist/chunk-M53LC4Q7.mjs +35 -0
  195. package/dist/chunk-MZ6WRTD2.mjs +40 -0
  196. package/dist/chunk-NGEN2EES.mjs +581 -0
  197. package/dist/{chunk-ZR6HSEAB.mjs → chunk-NPCBNGNE.mjs} +17 -26
  198. package/dist/{chunk-C43HRKXH.mjs → chunk-OBV72JD4.mjs} +1 -1
  199. package/dist/{chunk-LPV4NJJK.mjs → chunk-PGQ6FMXS.mjs} +6 -5
  200. package/dist/{chunk-MEPSKGBO.mjs → chunk-PI6RULJX.mjs} +1 -1
  201. package/dist/{chunk-F3YTWO3T.mjs → chunk-RA6SAAFE.mjs} +9 -8
  202. package/dist/{chunk-UNNRUJTM.mjs → chunk-RRKM4MKB.mjs} +7 -7
  203. package/dist/{chunk-ULGNQPNE.mjs → chunk-S2VGME7X.mjs} +1 -1
  204. package/dist/{chunk-OLVJFKXS.mjs → chunk-S44XWTTC.mjs} +35 -25
  205. package/dist/{chunk-YMYIEVZP.mjs → chunk-SZEKQAOY.mjs} +1 -1
  206. package/dist/{chunk-BXF4AMHY.mjs → chunk-TMH263OK.mjs} +5 -4
  207. package/dist/{chunk-NJG7DHVF.mjs → chunk-U6DEBYU5.mjs} +10 -9
  208. package/dist/{chunk-QXDGGOLC.mjs → chunk-UMZTPUB3.mjs} +33 -21
  209. package/dist/{chunk-KSUWPU2F.mjs → chunk-WIPEDNSD.mjs} +7 -7
  210. package/dist/{chunk-QDAZGZUF.mjs → chunk-XCIG6HT2.mjs} +3 -3
  211. package/dist/{chunk-4J2PXL36.mjs → chunk-Y6YS33GM.mjs} +40 -38
  212. package/dist/{chunk-4XOB5TTD.mjs → chunk-ZKDKKQCE.mjs} +5 -5
  213. package/dist/{chunk-LOBLCFMN.mjs → chunk-ZTPYUU5C.mjs} +5 -5
  214. package/dist/fonts.mjs +0 -2
  215. package/dist/index.d.mts +13 -73
  216. package/dist/index.d.ts +13 -73
  217. package/dist/index.js +1149 -1892
  218. package/dist/index.mjs +81 -86
  219. package/package.json +20 -20
  220. package/src/components/Accordion/Accordion.tsx +15 -9
  221. package/src/components/AlertBanner/AlertBanner.tsx +7 -6
  222. package/src/components/AppHeader/AppHeader.tsx +25 -10
  223. package/src/components/Avatar/Avatar.tsx +92 -1
  224. package/src/components/Avatar/index.ts +2 -2
  225. package/src/components/Badge/Badge.tsx +2 -2
  226. package/src/components/Button/Button.tsx +50 -46
  227. package/src/components/Card/Card.tsx +1 -0
  228. package/src/components/CategoryStrip/CategoryStrip.tsx +36 -49
  229. package/src/components/Checkbox/Checkbox.tsx +3 -0
  230. package/src/components/Chip/Chip.tsx +5 -4
  231. package/src/components/ConfirmDialog/ConfirmDialog.tsx +33 -17
  232. package/src/components/DetailRow/DetailRow.tsx +3 -3
  233. package/src/components/EmptyState/EmptyState.tsx +2 -2
  234. package/src/components/ErrorBoundary/ErrorBoundary.tsx +6 -6
  235. package/src/components/HolographicCard/HolographicCard.tsx +14 -95
  236. package/src/components/IconButton/IconButton.tsx +2 -2
  237. package/src/components/IconPicker/IconPicker.tsx +13 -12
  238. package/src/components/ImageUpload/ImageUpload.tsx +43 -46
  239. package/src/components/ImageViewer/ImageViewer.tsx +3 -3
  240. package/src/components/Input/Input.tsx +11 -5
  241. package/src/components/LabelValue/LabelValue.tsx +2 -2
  242. package/src/components/ListItem/ListItem.tsx +7 -4
  243. package/src/components/MediaCard/MediaCard.tsx +21 -59
  244. package/src/components/MenuItem/MenuItem.tsx +5 -2
  245. package/src/components/MonthPicker/MonthPicker.tsx +2 -2
  246. package/src/components/NumberStepper/NumberStepper.tsx +10 -6
  247. package/src/components/PagerDots/PagerDots.tsx +38 -28
  248. package/src/components/PricingCard/PricingCard.tsx +6 -6
  249. package/src/components/RadioGroup/RadioGroup.tsx +18 -31
  250. package/src/components/Select/Select.tsx +35 -39
  251. package/src/components/SelectableCard/SelectableCard.tsx +4 -6
  252. package/src/components/SelectableGrid/SelectableGrid.tsx +37 -72
  253. package/src/components/Sheet/Sheet.tsx +28 -15
  254. package/src/components/Sheet/index.ts +2 -2
  255. package/src/components/SheetSelect/SheetSelect.tsx +3 -3
  256. package/src/components/Skeleton/Skeleton.tsx +1 -1
  257. package/src/components/Slider/Slider.tsx +3 -0
  258. package/src/components/Spinner/Spinner.tsx +2 -2
  259. package/src/components/Stats/Stats.tsx +2 -2
  260. package/src/components/Switch/Switch.tsx +12 -7
  261. package/src/components/TabBar/TabBar.tsx +9 -8
  262. package/src/components/Text/Text.tsx +13 -1
  263. package/src/components/Textarea/Textarea.tsx +18 -32
  264. package/src/components/Toggle/Toggle.tsx +3 -3
  265. package/src/hooks/useConfirmDialog.ts +31 -42
  266. package/src/index.ts +3 -4
  267. package/src/theme/ThemeProvider.tsx +1 -4
  268. package/src/theme/colorUtils.ts +1 -72
  269. package/src/theme/colors.ts +40 -1
  270. package/src/theme/types.ts +2 -2
  271. package/src/utils/animations.ts +0 -47
  272. package/src/utils/curatedIcons.ts +93 -801
  273. package/src/utils/haptics.ts +13 -208
  274. package/src/utils/icons.ts +27 -91
  275. package/src/utils/pressable.ts +10 -61
  276. package/dist/VirtualList.d.mts +0 -19
  277. package/dist/VirtualList.d.ts +0 -19
  278. package/dist/VirtualList.js +0 -38
  279. package/dist/VirtualList.mjs +0 -2
  280. package/dist/chunk-2BA3JMKK.mjs +0 -136
  281. package/dist/chunk-3DKJ2GIC.mjs +0 -30
  282. package/dist/chunk-7ELGZ66G.mjs +0 -164
  283. package/dist/chunk-DVK4G2GT.mjs +0 -59
  284. package/dist/chunk-EJ7ZPXOH.mjs +0 -163
  285. package/dist/chunk-KA7LTET3.mjs +0 -71
  286. package/dist/chunk-LNPKGWBG.mjs +0 -134
  287. package/dist/chunk-NC5ZTR2Y.mjs +0 -32
  288. package/dist/chunk-SAWUXP3A.mjs +0 -1114
  289. package/dist/chunk-Y6FXYEAI.mjs +0 -8
  290. package/dist/chunk-YNROWHQJ.mjs +0 -46
  291. package/src/components/VirtualList/VirtualList.tsx +0 -60
  292. package/src/components/VirtualList/index.ts +0 -1
  293. package/src/utils/fontGuard.ts +0 -35
  294. package/src/utils/hover.ts +0 -25
  295. package/src/utils/useColorTransition.ts +0 -40
  296. package/src/utils/usePressScale.ts +0 -75
@@ -1,15 +1,10 @@
1
1
  import React, { useState } from 'react'
2
2
  import { TextInput, View, Text, StyleSheet, TextInputProps, ViewStyle, Platform } from 'react-native'
3
- import Animated, {
4
- useAnimatedStyle,
5
- interpolateColor,
6
- interpolate,
7
- } from 'react-native-reanimated'
3
+ import { EaseView } from 'react-native-ease'
8
4
  import { useTheme } from '../../theme'
9
5
  import { s, vs, ms } from '../../utils/scaling'
10
- import { renderIcon } from '../../utils/icons'
11
- import { useColorTransition } from '../../utils/useColorTransition'
12
- import { TIMINGS } from '../../utils/animations'
6
+ import { Icon } from '../../utils/icons'
7
+ import { COLOR_TRANSITION } from '../../utils/animations'
13
8
 
14
9
  const webInputResetStyle: Record<string, unknown> =
15
10
  Platform.OS === 'web'
@@ -20,6 +15,7 @@ export interface TextareaProps extends TextInputProps {
20
15
  label?: string
21
16
  error?: string
22
17
  hint?: string
18
+ disabled?: boolean
23
19
  rows?: number
24
20
  prefixIcon?: string
25
21
  prefixIconNode?: React.ReactNode
@@ -31,6 +27,7 @@ export function Textarea({
31
27
  label,
32
28
  error,
33
29
  hint,
30
+ disabled,
34
31
  rows = 4,
35
32
  prefixIcon,
36
33
  prefixIconNode,
@@ -44,46 +41,36 @@ export function Textarea({
44
41
  }: TextareaProps) {
45
42
  const { colors } = useTheme()
46
43
  const [focused, setFocused] = useState(false)
47
- const focusProgress = useColorTransition(focused, {
48
- duration: focused ? TIMINGS.focusIn.duration : TIMINGS.focusOut.duration,
49
- })
50
44
 
51
45
  const resolvedPrefixIcon = prefixIcon
52
- ? renderIcon(prefixIcon, ms(16), prefixIconColor ?? colors.foregroundMuted)
46
+ ? <Icon name={prefixIcon} size={ms(16)} color={prefixIconColor ?? colors.foregroundMuted} />
53
47
  : prefixIconNode
54
48
 
55
- // Border drawn on an absolute overlay (mirrors Input.tsx) so the 1px→2px
56
- // focus weight change never resizes the box / reflows content.
57
- const borderAnimStyle = useAnimatedStyle(() => ({
58
- borderColor: error
59
- ? colors.destructive
60
- : interpolateColor(focusProgress.value, [0, 1], [colors.border, colors.primary]),
61
- borderWidth: error
62
- ? 2
63
- : interpolate(focusProgress.value, [0, 1], [1, 2]),
64
- }))
49
+ const borderColor = error ? colors.destructive : (focused ? colors.primary : colors.border)
65
50
 
66
51
  return (
67
52
  <View style={[styles.container, containerStyle]}>
68
53
  {label ? <Text style={[styles.label, { color: colors.foreground }]} allowFontScaling={true}>{label}</Text> : null}
69
- <Animated.View
70
- style={[
71
- styles.inputWrapper,
72
- { backgroundColor: colors.background },
73
- ]}
74
- >
75
- <Animated.View style={[styles.borderOverlay, borderAnimStyle]} pointerEvents="none" />
54
+ <View style={[styles.inputWrapper, { backgroundColor: colors.background }]}>
55
+ <EaseView
56
+ style={[styles.borderOverlay, { borderWidth: error ? 2 : 1 }]}
57
+ animate={{ borderColor }}
58
+ transition={COLOR_TRANSITION}
59
+ pointerEvents="none"
60
+ />
76
61
  {resolvedPrefixIcon ? <View style={styles.prefixIcon}>{resolvedPrefixIcon}</View> : null}
77
62
  <TextInput
78
63
  multiline
79
64
  numberOfLines={rows}
80
65
  textAlignVertical="top"
66
+ editable={!disabled}
81
67
  style={[
82
68
  styles.input,
83
69
  {
84
70
  color: colors.foreground,
85
71
  minHeight: rows * vs(30),
86
72
  },
73
+ disabled && { opacity: 0.45 },
87
74
  webInputResetStyle,
88
75
  style,
89
76
  ]}
@@ -98,9 +85,10 @@ export function Textarea({
98
85
  placeholderTextColor={colors.foregroundMuted}
99
86
  allowFontScaling={true}
100
87
  accessibilityLabel={accessibilityLabel ?? label}
88
+ accessibilityState={{ disabled: !!disabled }}
101
89
  {...props}
102
90
  />
103
- </Animated.View>
91
+ </View>
104
92
  {error ? (
105
93
  <Text
106
94
  style={[styles.helperText, { color: colors.destructive }]}
@@ -128,8 +116,6 @@ const styles = StyleSheet.create({
128
116
  marginBottom: vs(2),
129
117
  },
130
118
  inputWrapper: {
131
- // Border lives on borderOverlay (absolute); wrapper carries none so the
132
- // focus weight change never reflows content.
133
119
  borderRadius: 8,
134
120
  paddingHorizontal: s(14),
135
121
  paddingVertical: vs(11),
@@ -5,7 +5,7 @@ import { FontAwesome5 } from '@expo/vector-icons'
5
5
  import { selectionAsync as hapticSelection } from '../../utils/haptics'
6
6
  import { useTheme } from '../../theme'
7
7
  import { s, vs, ms } from '../../utils/scaling'
8
- import { renderIcon } from '../../utils/icons'
8
+ import { Icon } from '../../utils/icons'
9
9
  import { COLOR_TRANSITION } from '../../utils/animations'
10
10
  import { PressableButton } from '../../utils/pressable'
11
11
 
@@ -30,13 +30,13 @@ function ToggleIcon({ pressed, iconName, activeIconName, icon, activeIcon, iconC
30
30
  }
31
31
 
32
32
  if (pressed) {
33
- if (activeIconName) return <>{renderIcon(activeIconName, iconSize, activeIconColor ?? primaryColor)}</>
33
+ if (activeIconName) return <Icon name={activeIconName} size={iconSize} color={activeIconColor ?? primaryColor} />
34
34
  const active = renderProp(activeIcon)
35
35
  if (active) return <>{active}</>
36
36
  return <FontAwesome5 name="check-circle" size={iconSize} color={primaryColor} />
37
37
  }
38
38
 
39
- if (iconName) return <>{renderIcon(iconName, iconSize, iconColor ?? mutedColor)}</>
39
+ if (iconName) return <Icon name={iconName} size={iconSize} color={iconColor ?? mutedColor} />
40
40
  const custom = renderProp(icon)
41
41
  if (custom) return <>{custom}</>
42
42
 
@@ -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,