@idealyst/components 1.0.82 → 1.0.84

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 (316) hide show
  1. package/CLAUDE.md +199 -232
  2. package/README.md +5 -5
  3. package/package.json +25 -7
  4. package/plugin/README.md +272 -0
  5. package/plugin/test-cases.jsx +112 -0
  6. package/plugin/web-legacy.js +320 -0
  7. package/plugin/web.js +422 -124
  8. package/src/Accordion/Accordion.native.tsx +182 -0
  9. package/src/Accordion/Accordion.styles.tsx +260 -0
  10. package/src/Accordion/Accordion.web.tsx +147 -0
  11. package/src/Accordion/index.native.tsx +3 -0
  12. package/src/Accordion/index.ts +3 -0
  13. package/src/Accordion/index.web.tsx +3 -0
  14. package/src/Accordion/types.ts +23 -0
  15. package/src/ActivityIndicator/ActivityIndicator.native.tsx +17 -12
  16. package/src/ActivityIndicator/ActivityIndicator.styles.tsx +83 -109
  17. package/src/ActivityIndicator/ActivityIndicator.web.tsx +23 -17
  18. package/src/ActivityIndicator/index.ts +5 -2
  19. package/src/ActivityIndicator/index.web.ts +5 -2
  20. package/src/ActivityIndicator/types.ts +15 -10
  21. package/src/Alert/Alert.native.tsx +113 -0
  22. package/src/Alert/Alert.styles.tsx +304 -0
  23. package/src/Alert/Alert.web.tsx +123 -0
  24. package/src/Alert/index.native.ts +5 -0
  25. package/src/Alert/index.ts +5 -0
  26. package/src/Alert/index.web.ts +5 -0
  27. package/src/Alert/types.ts +21 -0
  28. package/src/Avatar/Avatar.native.tsx +8 -6
  29. package/src/Avatar/Avatar.styles.tsx +64 -58
  30. package/src/Avatar/Avatar.web.tsx +13 -8
  31. package/src/Avatar/index.ts +5 -2
  32. package/src/Avatar/index.web.ts +5 -2
  33. package/src/Avatar/types.ts +19 -13
  34. package/src/Badge/Badge.native.tsx +59 -14
  35. package/src/Badge/Badge.styles.tsx +125 -139
  36. package/src/Badge/Badge.web.tsx +72 -16
  37. package/src/Badge/index.ts +5 -2
  38. package/src/Badge/index.web.ts +5 -2
  39. package/src/Badge/types.ts +23 -11
  40. package/src/Breadcrumb/Breadcrumb.native.tsx +225 -0
  41. package/src/Breadcrumb/Breadcrumb.styles.tsx +234 -0
  42. package/src/Breadcrumb/Breadcrumb.web.tsx +268 -0
  43. package/src/Breadcrumb/index.native.ts +5 -0
  44. package/src/Breadcrumb/index.ts +5 -0
  45. package/src/Breadcrumb/index.web.ts +5 -0
  46. package/src/Breadcrumb/types.ts +56 -0
  47. package/src/Button/Button.native.tsx +75 -24
  48. package/src/Button/Button.styles.tsx +248 -205
  49. package/src/Button/Button.web.tsx +82 -25
  50. package/src/Button/index.ts +5 -5
  51. package/src/Button/index.web.ts +5 -3
  52. package/src/Button/types.ts +32 -15
  53. package/src/Card/Card.native.tsx +14 -11
  54. package/src/Card/Card.styles.tsx +146 -220
  55. package/src/Card/Card.web.tsx +20 -21
  56. package/src/Card/index.ts +5 -5
  57. package/src/Card/index.web.ts +5 -3
  58. package/src/Card/types.ts +24 -17
  59. package/src/Checkbox/Checkbox.native.tsx +24 -34
  60. package/src/Checkbox/Checkbox.styles.tsx +223 -275
  61. package/src/Checkbox/Checkbox.web.tsx +30 -37
  62. package/src/Checkbox/index.ts +5 -5
  63. package/src/Checkbox/index.web.ts +5 -3
  64. package/src/Checkbox/types.ts +26 -20
  65. package/src/Chip/Chip.native.tsx +126 -0
  66. package/src/Chip/Chip.styles.tsx +138 -0
  67. package/src/Chip/Chip.web.tsx +154 -0
  68. package/src/Chip/index.native.ts +5 -0
  69. package/src/Chip/index.ts +5 -0
  70. package/src/Chip/index.web.ts +5 -0
  71. package/src/Chip/types.ts +51 -0
  72. package/src/Dialog/Dialog.native.tsx +65 -12
  73. package/src/Dialog/Dialog.styles.tsx +154 -136
  74. package/src/Dialog/Dialog.web.tsx +16 -11
  75. package/src/Dialog/index.ts +5 -2
  76. package/src/Dialog/index.web.ts +5 -2
  77. package/src/Dialog/types.ts +22 -16
  78. package/src/Divider/Divider.native.tsx +19 -14
  79. package/src/Divider/Divider.styles.tsx +273 -595
  80. package/src/Divider/Divider.web.tsx +19 -12
  81. package/src/Divider/index.ts +5 -5
  82. package/src/Divider/index.web.ts +5 -3
  83. package/src/Divider/types.ts +28 -19
  84. package/src/Icon/Icon.native.tsx +17 -24
  85. package/src/Icon/Icon.styles.tsx +64 -48
  86. package/src/Icon/Icon.web.tsx +14 -11
  87. package/src/Icon/IconSvg/IconSvg.native.tsx +42 -0
  88. package/src/Icon/IconSvg/IconSvg.web.tsx +40 -0
  89. package/src/Icon/IconSvg/index.native.ts +1 -0
  90. package/src/Icon/IconSvg/index.ts +1 -0
  91. package/src/Icon/icon-resolver.native.ts +27 -0
  92. package/src/Icon/icon-resolver.ts +70 -0
  93. package/src/Icon/index.ts +5 -5
  94. package/src/Icon/index.web.ts +5 -3
  95. package/src/Icon/types.ts +17 -11
  96. package/src/Image/Image.native.tsx +86 -0
  97. package/src/Image/Image.styles.tsx +57 -0
  98. package/src/Image/Image.web.tsx +92 -0
  99. package/src/Image/index.native.ts +5 -0
  100. package/src/Image/index.ts +5 -0
  101. package/src/Image/types.ts +21 -0
  102. package/src/Input/Input.native.tsx +103 -26
  103. package/src/Input/Input.styles.tsx +240 -177
  104. package/src/Input/Input.web.tsx +141 -38
  105. package/src/Input/index.ts +5 -5
  106. package/src/Input/index.web.ts +5 -3
  107. package/src/Input/types.ts +43 -20
  108. package/src/List/List.native.tsx +56 -0
  109. package/src/List/List.styles.tsx +257 -0
  110. package/src/List/List.web.tsx +43 -0
  111. package/src/List/ListContext.tsx +16 -0
  112. package/src/List/ListItem.native.tsx +111 -0
  113. package/src/List/ListItem.web.tsx +110 -0
  114. package/src/List/ListSection.native.tsx +31 -0
  115. package/src/List/ListSection.web.tsx +33 -0
  116. package/src/List/index.native.tsx +5 -0
  117. package/src/List/index.ts +5 -0
  118. package/src/List/index.web.tsx +5 -0
  119. package/src/List/types.ts +42 -0
  120. package/src/Menu/Menu.native.tsx +150 -0
  121. package/src/Menu/Menu.styles.tsx +185 -0
  122. package/src/Menu/Menu.web.tsx +99 -0
  123. package/src/Menu/MenuItem.native.tsx +66 -0
  124. package/src/Menu/MenuItem.styles.tsx +119 -0
  125. package/src/Menu/MenuItem.web.tsx +67 -0
  126. package/src/Menu/index.native.ts +3 -0
  127. package/src/Menu/index.ts +3 -0
  128. package/src/Menu/index.web.ts +3 -0
  129. package/src/Menu/types.ts +30 -0
  130. package/src/Popover/Popover.native.tsx +102 -32
  131. package/src/Popover/Popover.styles.tsx +100 -67
  132. package/src/Popover/Popover.web.tsx +36 -260
  133. package/src/Popover/index.ts +5 -2
  134. package/src/Popover/index.web.ts +5 -2
  135. package/src/Popover/types.ts +14 -13
  136. package/src/Pressable/Pressable.native.tsx +7 -6
  137. package/src/Pressable/Pressable.web.tsx +8 -6
  138. package/src/Pressable/index.ts +5 -2
  139. package/src/Pressable/index.web.ts +5 -2
  140. package/src/Pressable/types.ts +11 -10
  141. package/src/Progress/Progress.native.tsx +179 -0
  142. package/src/Progress/Progress.styles.tsx +164 -0
  143. package/src/Progress/Progress.web.tsx +144 -0
  144. package/src/Progress/index.native.ts +1 -0
  145. package/src/Progress/index.ts +5 -0
  146. package/src/Progress/index.web.ts +5 -0
  147. package/src/Progress/types.ts +21 -0
  148. package/src/RadioButton/RadioButton.native.tsx +88 -0
  149. package/src/RadioButton/RadioButton.styles.tsx +163 -0
  150. package/src/RadioButton/RadioButton.web.tsx +85 -0
  151. package/src/RadioButton/RadioGroup.native.tsx +43 -0
  152. package/src/RadioButton/RadioGroup.web.tsx +49 -0
  153. package/src/RadioButton/index.native.ts +2 -0
  154. package/src/RadioButton/index.ts +2 -0
  155. package/src/RadioButton/index.web.ts +2 -0
  156. package/src/RadioButton/types.ts +29 -0
  157. package/src/SVGImage/SVGImage.native.tsx +9 -7
  158. package/src/SVGImage/SVGImage.styles.tsx +63 -55
  159. package/src/SVGImage/SVGImage.web.tsx +16 -13
  160. package/src/SVGImage/index.ts +5 -5
  161. package/src/SVGImage/index.web.ts +5 -2
  162. package/src/SVGImage/types.ts +7 -3
  163. package/src/Screen/Screen.native.tsx +43 -17
  164. package/src/Screen/Screen.styles.tsx +58 -54
  165. package/src/Screen/Screen.web.tsx +11 -5
  166. package/src/Screen/index.ts +5 -2
  167. package/src/Screen/index.web.ts +5 -2
  168. package/src/Screen/types.ts +23 -9
  169. package/src/Select/Select.native.tsx +347 -0
  170. package/src/Select/Select.styles.tsx +335 -0
  171. package/src/Select/Select.web.tsx +276 -0
  172. package/src/Select/index.native.ts +2 -0
  173. package/src/Select/index.ts +5 -0
  174. package/src/Select/index.web.ts +5 -0
  175. package/src/Select/types.ts +124 -0
  176. package/src/Skeleton/Skeleton.native.tsx +139 -0
  177. package/src/Skeleton/Skeleton.styles.tsx +59 -0
  178. package/src/Skeleton/Skeleton.web.tsx +112 -0
  179. package/src/Skeleton/index.native.ts +4 -0
  180. package/src/Skeleton/index.ts +5 -0
  181. package/src/Skeleton/index.web.ts +5 -0
  182. package/src/Skeleton/types.ts +75 -0
  183. package/src/Slider/Slider.native.tsx +248 -0
  184. package/src/Slider/Slider.styles.tsx +241 -0
  185. package/src/Slider/Slider.web.tsx +226 -0
  186. package/src/Slider/index.native.ts +3 -0
  187. package/src/Slider/index.ts +5 -0
  188. package/src/Slider/index.web.ts +5 -0
  189. package/src/Slider/types.ts +31 -0
  190. package/src/Switch/Switch.native.tsx +131 -0
  191. package/src/Switch/Switch.styles.tsx +169 -0
  192. package/src/Switch/Switch.web.tsx +121 -0
  193. package/src/Switch/index.native.ts +3 -0
  194. package/src/Switch/index.ts +5 -0
  195. package/src/Switch/index.web.ts +5 -0
  196. package/src/Switch/types.ts +21 -0
  197. package/src/TabBar/TabBar.native.tsx +142 -0
  198. package/src/TabBar/TabBar.styles.tsx +399 -0
  199. package/src/TabBar/TabBar.web.tsx +205 -0
  200. package/src/TabBar/index.native.tsx +3 -0
  201. package/src/TabBar/index.ts +3 -0
  202. package/src/TabBar/index.web.tsx +3 -0
  203. package/src/TabBar/types.ts +26 -0
  204. package/src/Table/Table.native.tsx +122 -0
  205. package/src/Table/Table.styles.tsx +283 -0
  206. package/src/Table/Table.web.tsx +112 -0
  207. package/src/Table/index.native.tsx +3 -0
  208. package/src/Table/index.ts +3 -0
  209. package/src/Table/index.web.tsx +3 -0
  210. package/src/Table/types.ts +28 -0
  211. package/src/Text/Text.native.tsx +12 -11
  212. package/src/Text/Text.styles.tsx +76 -64
  213. package/src/Text/Text.web.tsx +14 -9
  214. package/src/Text/index.ts +5 -5
  215. package/src/Text/index.web.ts +5 -3
  216. package/src/Text/types.ts +20 -13
  217. package/src/TextArea/TextArea.native.tsx +134 -0
  218. package/src/TextArea/TextArea.styles.tsx +175 -0
  219. package/src/TextArea/TextArea.web.tsx +156 -0
  220. package/src/TextArea/index.native.ts +3 -0
  221. package/src/TextArea/index.ts +3 -0
  222. package/src/TextArea/index.web.ts +3 -0
  223. package/src/TextArea/types.ts +30 -0
  224. package/src/Tooltip/Tooltip.native.tsx +165 -0
  225. package/src/Tooltip/Tooltip.styles.tsx +73 -0
  226. package/src/Tooltip/Tooltip.web.tsx +87 -0
  227. package/src/Tooltip/index.native.ts +3 -0
  228. package/src/Tooltip/index.ts +3 -0
  229. package/src/Tooltip/types.ts +18 -0
  230. package/src/Video/Video.native.tsx +105 -0
  231. package/src/Video/Video.styles.tsx +39 -0
  232. package/src/Video/Video.web.tsx +115 -0
  233. package/src/Video/index.native.ts +5 -0
  234. package/src/Video/index.ts +5 -0
  235. package/src/Video/types.ts +29 -0
  236. package/src/View/View.native.tsx +9 -14
  237. package/src/View/View.styles.tsx +101 -93
  238. package/src/View/View.web.tsx +16 -17
  239. package/src/View/index.ts +5 -5
  240. package/src/View/index.web.ts +5 -3
  241. package/src/View/types.ts +29 -21
  242. package/src/examples/AccordionExamples.tsx +126 -0
  243. package/src/examples/AlertExamples.tsx +280 -0
  244. package/src/examples/AvatarExamples.tsx +23 -23
  245. package/src/examples/BadgeExamples.tsx +109 -41
  246. package/src/examples/BreadcrumbExamples.tsx +312 -0
  247. package/src/examples/ButtonExamples.tsx +160 -33
  248. package/src/examples/CardExamples.tsx +40 -40
  249. package/src/examples/CheckboxExamples.tsx +12 -12
  250. package/src/examples/ChipExamples.tsx +197 -0
  251. package/src/examples/DialogExamples.tsx +22 -22
  252. package/src/examples/DividerExamples.tsx +49 -49
  253. package/src/examples/IconExamples.tsx +270 -54
  254. package/src/examples/ImageExamples.tsx +174 -0
  255. package/src/examples/InputExamples.tsx +75 -17
  256. package/src/examples/ListExamples.tsx +288 -0
  257. package/src/examples/MenuExamples.tsx +144 -0
  258. package/src/examples/PopoverExamples.tsx +69 -73
  259. package/src/examples/ProgressExamples.tsx +137 -0
  260. package/src/examples/RadioButtonExamples.tsx +161 -0
  261. package/src/examples/SVGImageExamples.tsx +19 -17
  262. package/src/examples/ScreenExamples.tsx +31 -31
  263. package/src/examples/SelectExamples.tsx +423 -0
  264. package/src/examples/SkeletonExamples.tsx +206 -0
  265. package/src/examples/SliderExamples.tsx +200 -0
  266. package/src/examples/SwitchExamples.tsx +182 -0
  267. package/src/examples/TabBarExamples.tsx +143 -0
  268. package/src/examples/TableExamples.tsx +280 -0
  269. package/src/examples/TextAreaExamples.tsx +173 -0
  270. package/src/examples/TextExamples.tsx +28 -32
  271. package/src/examples/ThemeExtensionExamples.tsx +10 -10
  272. package/src/examples/TooltipExamples.tsx +126 -0
  273. package/src/examples/VideoExamples.tsx +144 -0
  274. package/src/examples/ViewExamples.tsx +64 -56
  275. package/src/examples/index.ts +18 -3
  276. package/src/hooks/useMergeRefs.ts +16 -0
  277. package/src/hooks/useSmartPosition.native.ts +169 -0
  278. package/src/index.native.ts +80 -9
  279. package/src/index.ts +75 -1
  280. package/src/internal/BoundedModalContent.native.tsx +58 -0
  281. package/src/internal/PositionedPortal.tsx +254 -0
  282. package/src/internal/SafeAreaDebugOverlay.native.tsx +173 -0
  283. package/src/unistyles.d.ts +6 -0
  284. package/src/utils/buildSizeVariants.ts +16 -0
  285. package/src/utils/deepMerge.ts +43 -0
  286. package/src/utils/positionUtils.native.ts +280 -0
  287. package/src/utils/styleHelpers.ts +48 -0
  288. package/LLM-ACCESS-GUIDE.md +0 -143
  289. package/src/ActivityIndicator/README.md +0 -132
  290. package/src/Avatar/README.md +0 -139
  291. package/src/Badge/README.md +0 -170
  292. package/src/Button/Button.types.ts +0 -12
  293. package/src/Button/README.md +0 -262
  294. package/src/Card/README.md +0 -258
  295. package/src/Checkbox/README.md +0 -102
  296. package/src/Dialog/README.md +0 -210
  297. package/src/Divider/README.md +0 -108
  298. package/src/Icon/README.md +0 -81
  299. package/src/Input/README.md +0 -100
  300. package/src/SVGImage/README.md +0 -209
  301. package/src/Screen/README.md +0 -86
  302. package/src/Text/README.md +0 -94
  303. package/src/View/README.md +0 -107
  304. package/src/examples/AllExamples.tsx +0 -84
  305. package/src/examples/README.md +0 -136
  306. package/src/examples/ValidationExamples.tsx +0 -95
  307. package/src/examples/extendedTheme.ts +0 -329
  308. package/src/theme/breakpoints.ts +0 -8
  309. package/src/theme/colorResolver.ts +0 -218
  310. package/src/theme/colors.ts +0 -315
  311. package/src/theme/defaultThemes.ts +0 -326
  312. package/src/theme/index.ts +0 -188
  313. package/src/theme/themeBuilder.ts +0 -602
  314. package/src/theme/unistyles.d.ts +0 -6
  315. package/src/theme/variantHelpers.ts +0 -584
  316. package/src/theme/variants.ts +0 -56
@@ -0,0 +1,335 @@
1
+ import { StyleSheet } from 'react-native-unistyles';
2
+ import { Theme, StylesheetStyles, Intent, Size } from '@idealyst/theme';
3
+ import { buildSizeVariants } from '../utils/buildSizeVariants';
4
+ import { SelectIntentVariant } from './types';
5
+
6
+ // Type definitions
7
+ type SelectSize = Size;
8
+ type SelectType = 'outlined' | 'filled';
9
+
10
+ type SelectTriggerVariants = {
11
+ type: SelectType;
12
+ size: SelectSize;
13
+ intent: SelectIntentVariant;
14
+ disabled: boolean;
15
+ error: boolean;
16
+ focused: boolean;
17
+ }
18
+
19
+
20
+ export type ExpandedSelectTriggerStyles = StylesheetStyles<keyof SelectTriggerVariants>;
21
+ export type ExpandedSelectStyles = StylesheetStyles<never>;
22
+
23
+ function createTriggerTypeVariants(theme: Theme) {
24
+ return {
25
+ outlined: {
26
+ backgroundColor: theme.colors.surface.primary,
27
+ borderColor: theme.colors.border.primary,
28
+ _web: {
29
+ border: `1px solid ${theme.colors.border.primary}`,
30
+ },
31
+ },
32
+ filled: {
33
+ backgroundColor: theme.colors.surface.secondary,
34
+ borderColor: 'transparent',
35
+ _web: {
36
+ border: '1px solid transparent',
37
+ },
38
+ },
39
+ } as const;
40
+ }
41
+
42
+ function createTriggerSizeVariants(theme: Theme) {
43
+ return buildSizeVariants(theme, 'select', (size) => ({
44
+ paddingHorizontal: size.paddingHorizontal,
45
+ minHeight: size.minHeight,
46
+ }));
47
+ }
48
+
49
+ function createIntentVariants(theme: Theme, type: SelectType, intent: SelectIntentVariant) {
50
+ if (intent === 'neutral') {
51
+ return {};
52
+ }
53
+
54
+ const intentValue = (theme.intents as any)[intent];
55
+
56
+ if (type === 'outlined') {
57
+ return {
58
+ borderColor: intentValue.primary,
59
+ _web: {
60
+ border: `1px solid ${intentValue.primary}`,
61
+ },
62
+ } as const;
63
+ }
64
+
65
+ return {} as const;
66
+ }
67
+
68
+ function buildDynamicTriggerStyles(theme: Theme) {
69
+ return ({ type, intent }: Partial<SelectTriggerVariants>) => {
70
+ const intentStyles = createIntentVariants(theme, type, intent);
71
+
72
+ return {
73
+ position: 'relative',
74
+ flexDirection: 'row',
75
+ alignItems: 'center',
76
+ justifyContent: 'space-between',
77
+ borderRadius: 8,
78
+ borderWidth: 1,
79
+ borderStyle: 'solid',
80
+ ...intentStyles,
81
+ variants: {
82
+ type: createTriggerTypeVariants(theme),
83
+ size: createTriggerSizeVariants(theme),
84
+ disabled: {
85
+ true: {
86
+ opacity: 0.6,
87
+ _web: {
88
+ cursor: 'not-allowed',
89
+ },
90
+ },
91
+ false: {
92
+ _web: {
93
+ cursor: 'pointer',
94
+ _hover: {
95
+ opacity: 0.9,
96
+ },
97
+ _active: {
98
+ opacity: 0.8,
99
+ },
100
+ },
101
+ },
102
+ },
103
+ error: {
104
+ true: {
105
+ borderColor: theme.intents.error.primary,
106
+ _web: {
107
+ border: `1px solid ${theme.intents.error.primary}`,
108
+ },
109
+ },
110
+ false: {},
111
+ },
112
+ focused: {
113
+ true: {
114
+ borderColor: theme.intents.primary.primary,
115
+ _web: {
116
+ border: `2px solid ${theme.intents.primary.primary}`,
117
+ outline: 'none',
118
+ },
119
+ },
120
+ false: {},
121
+ },
122
+ },
123
+ _web: {
124
+ display: 'flex',
125
+ boxSizing: 'border-box',
126
+ _focus: {
127
+ outline: 'none',
128
+ },
129
+ },
130
+ } as const;
131
+ }
132
+ }
133
+
134
+ // Styles are inlined here instead of in @idealyst/theme because Unistyles' Babel
135
+ // transform on native cannot resolve function calls to extract variant structures.
136
+ export const selectStyles = StyleSheet.create((theme: Theme) => {
137
+ return {
138
+ container: {
139
+ position: 'relative',
140
+ backgroundColor: theme.colors.surface.primary,
141
+ },
142
+ label: {
143
+ fontSize: 14,
144
+ fontWeight: '500',
145
+ color: theme.colors.text.primary,
146
+ marginBottom: 4,
147
+ },
148
+ trigger: buildDynamicTriggerStyles(theme),
149
+ triggerContent: {
150
+ flex: 1,
151
+ flexDirection: 'row',
152
+ alignItems: 'center',
153
+ },
154
+ triggerText: {
155
+ color: theme.colors.text.primary,
156
+ flex: 1,
157
+ variants: {
158
+ size: buildSizeVariants(theme, 'select', (size: any) => ({
159
+ fontSize: size.fontSize,
160
+ })),
161
+ },
162
+ },
163
+ placeholder: {
164
+ color: theme.colors.text.secondary,
165
+ variants: {
166
+ size: buildSizeVariants(theme, 'select', (size: any) => ({
167
+ fontSize: size.fontSize,
168
+ })),
169
+ },
170
+ },
171
+ icon: {
172
+ marginLeft: 4,
173
+ color: theme.colors.text.secondary,
174
+ },
175
+ chevron: {
176
+ display: 'flex',
177
+ alignItems: 'center',
178
+ justifyContent: 'center',
179
+ marginLeft: 4,
180
+ color: theme.colors.text.secondary,
181
+ variants: {
182
+ size: buildSizeVariants(theme, 'select', (size: any) => ({
183
+ width: size.iconSize,
184
+ height: size.iconSize,
185
+ })),
186
+ },
187
+ _web: {
188
+ transition: 'transform 0.2s ease',
189
+ },
190
+ },
191
+ chevronOpen: {
192
+ _web: {
193
+ transform: 'rotate(180deg)',
194
+ }
195
+ },
196
+ dropdown: {
197
+ position: 'absolute',
198
+ top: '100%',
199
+ left: 0,
200
+ right: 0,
201
+ backgroundColor: theme.colors.surface.primary,
202
+ borderRadius: 8,
203
+ borderWidth: 1,
204
+ borderStyle: 'solid',
205
+ borderColor: theme.colors.border.primary,
206
+ shadowColor: '#000',
207
+ shadowOffset: { width: 0, height: 8 },
208
+ shadowOpacity: 0.1,
209
+ shadowRadius: 24,
210
+ elevation: 8,
211
+ zIndex: 9999,
212
+ maxHeight: 240,
213
+ minWidth: 200,
214
+ overflow: 'hidden',
215
+ _web: {
216
+ border: `1px solid ${theme.colors.border.primary}`,
217
+ boxShadow: '0 8px 24px rgba(0, 0, 0, 0.1), 0 4px 8px rgba(0, 0, 0, 0.06)',
218
+ overflowY: 'auto',
219
+ },
220
+ },
221
+ searchContainer: {
222
+ padding: 8,
223
+ borderBottomWidth: 1,
224
+ borderBottomStyle: 'solid',
225
+ borderBottomColor: theme.colors.border.primary,
226
+ _web: {
227
+ borderBottom: `1px solid ${theme.colors.border.primary}`,
228
+ },
229
+ },
230
+ searchInput: {
231
+ padding: 4,
232
+ borderRadius: 4,
233
+ borderWidth: 1,
234
+ borderStyle: 'solid',
235
+ borderColor: theme.colors.border.primary,
236
+ backgroundColor: theme.colors.surface.primary,
237
+ variants: {
238
+ size: buildSizeVariants(theme, 'select', (size: any) => ({
239
+ fontSize: size.fontSize,
240
+ })),
241
+ },
242
+ _web: {
243
+ border: `1px solid ${theme.colors.border.primary}`,
244
+ outline: 'none',
245
+ _focus: {
246
+ borderColor: theme.intents.primary.primary,
247
+ },
248
+ },
249
+ },
250
+ optionsList: {
251
+ paddingVertical: 4,
252
+ },
253
+ option: {
254
+ paddingHorizontal: 8,
255
+ paddingVertical: 4,
256
+ flexDirection: 'row',
257
+ alignItems: 'center',
258
+ minHeight: 36,
259
+ variants: {
260
+ selected: {
261
+ true: {
262
+ backgroundColor: theme.intents.primary.light,
263
+ },
264
+ false: {},
265
+ },
266
+ disabled: {
267
+ true: {
268
+ opacity: 0.5,
269
+ _web: {
270
+ cursor: 'not-allowed',
271
+ },
272
+ },
273
+ false: {
274
+ _web: {
275
+ cursor: 'pointer',
276
+ _hover: {
277
+ backgroundColor: theme.colors.surface.secondary,
278
+ },
279
+ _active: {
280
+ opacity: 0.8,
281
+ },
282
+ },
283
+ },
284
+ },
285
+ },
286
+ _web: {
287
+ display: 'flex',
288
+ },
289
+ },
290
+ optionContent: {
291
+ flexDirection: 'row',
292
+ alignItems: 'center',
293
+ flex: 1,
294
+ },
295
+ optionIcon: {
296
+ marginRight: 4,
297
+ },
298
+ optionText: {
299
+ color: theme.colors.text.primary,
300
+ flex: 1,
301
+ variants: {
302
+ size: buildSizeVariants(theme, 'select', (size: any) => ({
303
+ fontSize: size.fontSize,
304
+ })),
305
+ },
306
+ },
307
+ optionTextDisabled: {
308
+ color: theme.colors.text.secondary,
309
+ },
310
+ helperText: {
311
+ fontSize: 12,
312
+ marginTop: 4,
313
+ color: theme.colors.text.secondary,
314
+ variants: {
315
+ error: {
316
+ true: {
317
+ color: theme.intents.error.primary,
318
+ },
319
+ false: {},
320
+ },
321
+ },
322
+ },
323
+ overlay: {
324
+ position: 'absolute',
325
+ top: 0,
326
+ left: 0,
327
+ right: 0,
328
+ bottom: 0,
329
+ zIndex: 999,
330
+ _web: {
331
+ position: 'fixed',
332
+ },
333
+ },
334
+ } as const;
335
+ });
@@ -0,0 +1,276 @@
1
+ import React, { useState, useRef, useEffect, forwardRef } from 'react';
2
+ // @ts-ignore - web-specific import
3
+ import { getWebProps } from 'react-native-unistyles/web';
4
+ import { SelectProps, SelectOption } from './types';
5
+ import { selectStyles } from './Select.styles';
6
+ import { IconSvg } from '../Icon/IconSvg/IconSvg.web';
7
+ import { resolveIconPath } from '../Icon/icon-resolver';
8
+ import { PositionedPortal } from '../internal/PositionedPortal';
9
+ import useMergeRefs from '../hooks/useMergeRefs';
10
+
11
+ const Select = forwardRef<HTMLDivElement, SelectProps>(({
12
+ options,
13
+ value,
14
+ onValueChange,
15
+ placeholder = 'Select an option',
16
+ disabled = false,
17
+ error = false,
18
+ helperText,
19
+ label,
20
+ type = 'outlined',
21
+ intent = 'neutral',
22
+ size = 'md',
23
+ searchable = false,
24
+ filterOption,
25
+ maxHeight = 240,
26
+ style,
27
+ testID,
28
+ accessibilityLabel,
29
+ }, ref) => {
30
+ const [isOpen, setIsOpen] = useState(false);
31
+ const [searchTerm, setSearchTerm] = useState('');
32
+ const [focusedIndex, setFocusedIndex] = useState(-1);
33
+ const triggerRef = useRef<HTMLDivElement>(null);
34
+ const searchInputRef = useRef<HTMLInputElement>(null);
35
+
36
+ const selectedOption = options.find(option => option.value === value);
37
+
38
+ // Filter options based on search term
39
+ const filteredOptions = searchable && searchTerm
40
+ ? options.filter(option => {
41
+ if (filterOption) {
42
+ return filterOption(option, searchTerm);
43
+ }
44
+ return option.label.toLowerCase().includes(searchTerm.toLowerCase());
45
+ })
46
+ : options;
47
+
48
+ // Apply styles with variants
49
+ selectStyles.useVariants({
50
+ type,
51
+ size,
52
+ disabled,
53
+ error,
54
+ focused: isOpen,
55
+ });
56
+
57
+
58
+ // Handle keyboard navigation
59
+ const handleKeyDown = (event: KeyboardEvent) => {
60
+ if (!isOpen) return;
61
+
62
+ switch (event.key) {
63
+ case 'ArrowDown':
64
+ event.preventDefault();
65
+ setFocusedIndex(prev =>
66
+ prev < filteredOptions.length - 1 ? prev + 1 : 0
67
+ );
68
+ break;
69
+ case 'ArrowUp':
70
+ event.preventDefault();
71
+ setFocusedIndex(prev =>
72
+ prev > 0 ? prev - 1 : filteredOptions.length - 1
73
+ );
74
+ break;
75
+ case 'Enter':
76
+ case ' ':
77
+ event.preventDefault();
78
+ if (focusedIndex >= 0 && focusedIndex < filteredOptions.length) {
79
+ const option = filteredOptions[focusedIndex];
80
+ if (!option.disabled) {
81
+ handleOptionSelect(option);
82
+ }
83
+ }
84
+ break;
85
+ }
86
+ };
87
+
88
+ useEffect(() => {
89
+ if (!isOpen) return;
90
+ document.addEventListener('keydown', handleKeyDown);
91
+ return () => document.removeEventListener('keydown', handleKeyDown);
92
+ }, [isOpen, focusedIndex, filteredOptions]);
93
+
94
+ // Focus search input when dropdown opens
95
+ useEffect(() => {
96
+ if (isOpen && searchable && searchInputRef.current) {
97
+ // Delay to ensure dropdown is positioned
98
+ setTimeout(() => {
99
+ searchInputRef.current?.focus();
100
+ }, 50);
101
+ }
102
+ }, [isOpen, searchable]);
103
+
104
+ const handleTriggerClick = () => {
105
+ if (!disabled) {
106
+ setIsOpen(!isOpen);
107
+ setSearchTerm('');
108
+ setFocusedIndex(-1);
109
+ }
110
+ };
111
+
112
+ const handleOptionSelect = (option: SelectOption) => {
113
+ if (!option.disabled) {
114
+ onValueChange(option.value);
115
+ setIsOpen(false);
116
+ setSearchTerm('');
117
+ triggerRef.current?.focus();
118
+ }
119
+ };
120
+
121
+ const handleSearchChange = (e: React.ChangeEvent<HTMLInputElement>) => {
122
+ setSearchTerm(e.target.value);
123
+ setFocusedIndex(0);
124
+ };
125
+
126
+ const containerWebProps = getWebProps([
127
+ selectStyles.container,
128
+ style as any
129
+ ]);
130
+
131
+ const triggerWebProps = getWebProps([
132
+ selectStyles.trigger({ type, intent }),
133
+ ]);
134
+
135
+ const handleClose = () => {
136
+ setIsOpen(false);
137
+ triggerRef.current?.focus();
138
+ };
139
+
140
+ const mergedRef = useMergeRefs(ref, containerWebProps.ref);
141
+
142
+ return (
143
+ <div {...containerWebProps} ref={mergedRef} data-testid={testID}>
144
+ {label && (
145
+ <label {...getWebProps([selectStyles.label])}>
146
+ {label}
147
+ </label>
148
+ )}
149
+
150
+ <div ref={triggerRef}>
151
+ <button
152
+ {...triggerWebProps}
153
+ onClick={handleTriggerClick}
154
+ disabled={disabled}
155
+ aria-label={accessibilityLabel || label}
156
+ aria-expanded={isOpen}
157
+ aria-haspopup="listbox"
158
+ type="button"
159
+ >
160
+ <div {...getWebProps([selectStyles.triggerContent])}>
161
+ {selectedOption?.icon && (
162
+ <span {...getWebProps([selectStyles.icon])}>
163
+ {selectedOption.icon}
164
+ </span>
165
+ )}
166
+ <span
167
+ {...getWebProps([
168
+ selectedOption ? selectStyles.triggerText : selectStyles.placeholder
169
+ ])}
170
+ >
171
+ {selectedOption ? selectedOption.label : placeholder}
172
+ </span>
173
+ </div>
174
+
175
+ <IconSvg
176
+ path={resolveIconPath('chevron-down')}
177
+ {...getWebProps([
178
+ selectStyles.chevron,
179
+ isOpen && selectStyles.chevronOpen
180
+ ])}
181
+ aria-label="chevron-down"
182
+ />
183
+ </button>
184
+ </div>
185
+ <PositionedPortal
186
+ open={isOpen}
187
+ anchor={triggerRef}
188
+ placement="bottom-start"
189
+ offset={4}
190
+ onClickOutside={handleClose}
191
+ onEscapeKey={handleClose}
192
+ matchWidth={false}
193
+ zIndex={1000}
194
+ >
195
+ <div
196
+ {...getWebProps([selectStyles.dropdown])}
197
+ style={{
198
+ maxHeight: maxHeight,
199
+ // Override positioning since PositionedPortal handles it
200
+ position: 'relative',
201
+ top: 'auto',
202
+ left: 'auto',
203
+ right: 'auto',
204
+ }}
205
+ role="listbox"
206
+ >
207
+ {searchable && (
208
+ <div {...getWebProps([selectStyles.searchContainer])}>
209
+ <input
210
+ ref={searchInputRef}
211
+ type="text"
212
+ placeholder="Search options..."
213
+ value={searchTerm}
214
+ onChange={handleSearchChange}
215
+ {...getWebProps([selectStyles.searchInput])}
216
+ />
217
+ </div>
218
+ )}
219
+
220
+ <div {...getWebProps([selectStyles.optionsList])}>
221
+ {filteredOptions.map((option, index) => {
222
+ const isSelected = option.value === value;
223
+
224
+ return (
225
+ <div
226
+ key={option.value}
227
+ onClick={() => handleOptionSelect(option)}
228
+ role="option"
229
+ aria-selected={isSelected}
230
+ onMouseEnter={() => setFocusedIndex(index)}
231
+ {...getWebProps([selectStyles.option])}
232
+ >
233
+ <div {...getWebProps([selectStyles.optionContent])}>
234
+ {option.icon && (
235
+ <span {...getWebProps([selectStyles.optionIcon])}>
236
+ {option.icon}
237
+ </span>
238
+ )}
239
+ <span {...getWebProps([
240
+ selectStyles.optionText,
241
+ option.disabled && selectStyles.optionTextDisabled
242
+ ])}>
243
+ {option.label}
244
+ </span>
245
+ </div>
246
+ </div>
247
+ );
248
+ })}
249
+
250
+ {filteredOptions.length === 0 && (
251
+ <div {...getWebProps([selectStyles.option])} style={{ cursor: 'default' }}>
252
+ <span {...getWebProps([selectStyles.optionText])}>
253
+ No options found
254
+ </span>
255
+ </div>
256
+ )}
257
+ </div>
258
+ </div>
259
+ </PositionedPortal>
260
+
261
+ {helperText && (
262
+ <div
263
+ {...getWebProps([
264
+ selectStyles.helperText,
265
+ ])}
266
+ >
267
+ {helperText}
268
+ </div>
269
+ )}
270
+ </div>
271
+ );
272
+ });
273
+
274
+ Select.displayName = 'Select';
275
+
276
+ export default Select;
@@ -0,0 +1,2 @@
1
+ export { default } from './Select.native';
2
+ export * from './types';
@@ -0,0 +1,5 @@
1
+ import SelectComponent from './Select.web';
2
+
3
+ export default SelectComponent;
4
+ export { SelectComponent as Select };
5
+ export * from './types';
@@ -0,0 +1,5 @@
1
+ import SelectComponent from './Select.web';
2
+
3
+ export default SelectComponent;
4
+ export { SelectComponent as Select };
5
+ export * from './types';