@ankhorage/surface 0.1.4 → 0.1.6

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 (292) hide show
  1. package/CHANGELOG.md +12 -0
  2. package/README.md +23 -184
  3. package/dist/components/badge/Badge.js.map +1 -1
  4. package/dist/components/badge/index.js.map +1 -1
  5. package/dist/components/badge/types.js.map +1 -1
  6. package/dist/components/button/Button.js.map +1 -1
  7. package/dist/components/button/index.js.map +1 -1
  8. package/dist/components/button/types.js.map +1 -1
  9. package/dist/components/card/Card.js.map +1 -1
  10. package/dist/components/card/index.js.map +1 -1
  11. package/dist/components/card/types.js.map +1 -1
  12. package/dist/components/checkbox/Checkbox.js.map +1 -1
  13. package/dist/components/checkbox/index.js.map +1 -1
  14. package/dist/components/checkbox/types.js.map +1 -1
  15. package/dist/components/drawer/Drawer.js.map +1 -1
  16. package/dist/components/drawer/index.js.map +1 -1
  17. package/dist/components/drawer/types.js.map +1 -1
  18. package/dist/components/field/Field.js.map +1 -1
  19. package/dist/components/field/index.js.map +1 -1
  20. package/dist/components/field/types.js.map +1 -1
  21. package/dist/components/helper-text/HelperText.js.map +1 -1
  22. package/dist/components/helper-text/index.js.map +1 -1
  23. package/dist/components/helper-text/types.js.map +1 -1
  24. package/dist/components/icon-button/IconButton.js.map +1 -1
  25. package/dist/components/icon-button/index.js.map +1 -1
  26. package/dist/components/icon-button/types.js.map +1 -1
  27. package/dist/components/label/Label.js.map +1 -1
  28. package/dist/components/label/index.js.map +1 -1
  29. package/dist/components/label/types.js.map +1 -1
  30. package/dist/components/list-item/ListItem.js.map +1 -1
  31. package/dist/components/list-item/index.js.map +1 -1
  32. package/dist/components/list-item/types.js.map +1 -1
  33. package/dist/components/menu/Menu.js.map +1 -1
  34. package/dist/components/menu/index.js.map +1 -1
  35. package/dist/components/menu/navigation.js.map +1 -1
  36. package/dist/components/menu/types.js.map +1 -1
  37. package/dist/components/modal/Modal.js.map +1 -1
  38. package/dist/components/modal/index.js.map +1 -1
  39. package/dist/components/modal/types.js.map +1 -1
  40. package/dist/components/radio/Radio.js.map +1 -1
  41. package/dist/components/radio/index.js.map +1 -1
  42. package/dist/components/radio/types.js.map +1 -1
  43. package/dist/components/switch/Switch.js.map +1 -1
  44. package/dist/components/switch/index.js.map +1 -1
  45. package/dist/components/switch/types.js.map +1 -1
  46. package/dist/components/tabs/Tab.js.map +1 -1
  47. package/dist/components/tabs/TabList.js.map +1 -1
  48. package/dist/components/tabs/TabPanel.js.map +1 -1
  49. package/dist/components/tabs/Tabs.js.map +1 -1
  50. package/dist/components/tabs/a11y.js.map +1 -1
  51. package/dist/components/tabs/context.js.map +1 -1
  52. package/dist/components/tabs/index.js.map +1 -1
  53. package/dist/components/tabs/navigation.js.map +1 -1
  54. package/dist/components/tabs/types.js.map +1 -1
  55. package/dist/components/text-input/TextInput.js.map +1 -1
  56. package/dist/components/text-input/index.js.map +1 -1
  57. package/dist/components/text-input/types.js.map +1 -1
  58. package/dist/components/textarea/Textarea.js.map +1 -1
  59. package/dist/components/textarea/index.js.map +1 -1
  60. package/dist/components/textarea/types.js.map +1 -1
  61. package/dist/components/toast/Toast.js.map +1 -1
  62. package/dist/components/toast/ToastProvider.js.map +1 -1
  63. package/dist/components/toast/index.js.map +1 -1
  64. package/dist/components/toast/types.js.map +1 -1
  65. package/dist/components/tooltip/Tooltip.js.map +1 -1
  66. package/dist/components/tooltip/index.js.map +1 -1
  67. package/dist/components/tooltip/types.js.map +1 -1
  68. package/dist/context/FontContext.js.map +1 -1
  69. package/dist/context/TranslationContext.js.map +1 -1
  70. package/dist/core/responsive/ResponsiveProvider.js.map +1 -1
  71. package/dist/core/responsive/breakpoints.js.map +1 -1
  72. package/dist/core/responsive/getBreakpointFromWidth.js.map +1 -1
  73. package/dist/core/responsive/index.js.map +1 -1
  74. package/dist/core/responsive/resolve.js.map +1 -1
  75. package/dist/core/responsive/types.js.map +1 -1
  76. package/dist/core/responsive/useBreakpoint.js.map +1 -1
  77. package/dist/examples/DocsExamples.js.map +1 -1
  78. package/dist/index.js.map +1 -1
  79. package/dist/internal/focus/FocusScope.js.map +1 -1
  80. package/dist/internal/focus/useFocusManager.js.map +1 -1
  81. package/dist/internal/overlay/OverlayProvider.js.map +1 -1
  82. package/dist/internal/overlay/Portal.js.map +1 -1
  83. package/dist/internal/overlay/useOverlayStack.js.map +1 -1
  84. package/dist/internal/resolvers/index.js.map +1 -1
  85. package/dist/internal/resolvers/resolveControlSize.js.map +1 -1
  86. package/dist/internal/resolvers/resolveFieldPresentation.js.map +1 -1
  87. package/dist/internal/resolvers/resolveFieldState.js.map +1 -1
  88. package/dist/internal/resolvers/resolveFocusRingStyles.js.map +1 -1
  89. package/dist/internal/resolvers/resolveIconSize.js.map +1 -1
  90. package/dist/internal/resolvers/resolveIndicatorSize.js.map +1 -1
  91. package/dist/internal/resolvers/resolveInteractiveColors.js.map +1 -1
  92. package/dist/internal/resolvers/resolveInteractiveState.js.map +1 -1
  93. package/dist/internal/resolvers/resolveOverlayAnimation.js.map +1 -1
  94. package/dist/internal/resolvers/resolveOverlayZIndex.js.map +1 -1
  95. package/dist/internal/resolvers/resolveSelectionControlBehavior.js.map +1 -1
  96. package/dist/internal/resolvers/resolveSelectionControlColors.js.map +1 -1
  97. package/dist/internal/resolvers/resolveTextColor.js.map +1 -1
  98. package/dist/internal/resolvers/resolveTextStyles.js.map +1 -1
  99. package/dist/internal/resolvers/resolveTone.js.map +1 -1
  100. package/dist/internal/useControllableState.js.map +1 -1
  101. package/dist/layout/Box.js.map +1 -1
  102. package/dist/layout/Center.js.map +1 -1
  103. package/dist/layout/Container.js.map +1 -1
  104. package/dist/layout/Divider.js.map +1 -1
  105. package/dist/layout/Grid.js.map +1 -1
  106. package/dist/layout/Inline.js.map +1 -1
  107. package/dist/layout/Show.js.map +1 -1
  108. package/dist/layout/Spacer.js.map +1 -1
  109. package/dist/layout/Stack.js.map +1 -1
  110. package/dist/layout/Surface.js.map +1 -1
  111. package/dist/layout/Template.js.map +1 -1
  112. package/dist/layout/helpers.js.map +1 -1
  113. package/dist/layout/index.js.map +1 -1
  114. package/dist/primitives/button-base/ButtonBase.js.map +1 -1
  115. package/dist/primitives/button-base/index.js.map +1 -1
  116. package/dist/primitives/button-base/types.js.map +1 -1
  117. package/dist/primitives/heading/Heading.js.map +1 -1
  118. package/dist/primitives/heading/index.js.map +1 -1
  119. package/dist/primitives/heading/resolveHeadingStyle.js.map +1 -1
  120. package/dist/primitives/heading/types.js.map +1 -1
  121. package/dist/primitives/icon/Icon.js.map +1 -1
  122. package/dist/primitives/icon/index.js.map +1 -1
  123. package/dist/primitives/icon/resolveExpoIconComponent.js.map +1 -1
  124. package/dist/primitives/text/Text.js.map +1 -1
  125. package/dist/primitives/text/index.js.map +1 -1
  126. package/dist/primitives/text/types.js.map +1 -1
  127. package/dist/theme/ThemeContext.js.map +1 -1
  128. package/dist/theme/colorEngine.js.map +1 -1
  129. package/dist/theme/createTheme.js.map +1 -1
  130. package/dist/theme/index.js.map +1 -1
  131. package/dist/theme/resolveToken.js.map +1 -1
  132. package/dist/theme/types.js.map +1 -1
  133. package/dist/utils/deepEqual.js.map +1 -1
  134. package/dist/utils/deepMerge.js.map +1 -1
  135. package/package.json +4 -1
  136. package/src/components/badge/Badge.tsx +47 -0
  137. package/src/components/badge/index.ts +2 -0
  138. package/src/components/badge/types.ts +13 -0
  139. package/src/components/button/Button.tsx +104 -0
  140. package/src/components/button/index.ts +2 -0
  141. package/src/components/button/types.ts +26 -0
  142. package/src/components/card/Card.tsx +81 -0
  143. package/src/components/card/index.ts +2 -0
  144. package/src/components/card/types.ts +11 -0
  145. package/src/components/checkbox/Checkbox.tsx +111 -0
  146. package/src/components/checkbox/index.ts +2 -0
  147. package/src/components/checkbox/types.ts +19 -0
  148. package/src/components/drawer/Drawer.tsx +92 -0
  149. package/src/components/drawer/index.ts +2 -0
  150. package/src/components/drawer/types.ts +10 -0
  151. package/src/components/field/Field.tsx +43 -0
  152. package/src/components/field/index.ts +2 -0
  153. package/src/components/field/types.ts +13 -0
  154. package/src/components/helper-text/HelperText.tsx +12 -0
  155. package/src/components/helper-text/index.ts +2 -0
  156. package/src/components/helper-text/types.ts +9 -0
  157. package/src/components/icon-button/IconButton.tsx +60 -0
  158. package/src/components/icon-button/index.ts +2 -0
  159. package/src/components/icon-button/types.ts +19 -0
  160. package/src/components/label/Label.tsx +17 -0
  161. package/src/components/label/index.ts +2 -0
  162. package/src/components/label/types.ts +10 -0
  163. package/src/components/list-item/ListItem.tsx +72 -0
  164. package/src/components/list-item/index.ts +2 -0
  165. package/src/components/list-item/types.ts +11 -0
  166. package/src/components/menu/Menu.tsx +180 -0
  167. package/src/components/menu/index.ts +2 -0
  168. package/src/components/menu/navigation.test.ts +21 -0
  169. package/src/components/menu/navigation.ts +34 -0
  170. package/src/components/menu/types.ts +16 -0
  171. package/src/components/modal/Modal.tsx +87 -0
  172. package/src/components/modal/index.ts +2 -0
  173. package/src/components/modal/types.ts +9 -0
  174. package/src/components/radio/Radio.tsx +116 -0
  175. package/src/components/radio/index.ts +2 -0
  176. package/src/components/radio/types.ts +19 -0
  177. package/src/components/switch/Switch.tsx +116 -0
  178. package/src/components/switch/index.ts +2 -0
  179. package/src/components/switch/types.ts +19 -0
  180. package/src/components/tabs/Tab.tsx +82 -0
  181. package/src/components/tabs/TabList.tsx +51 -0
  182. package/src/components/tabs/TabPanel.tsx +29 -0
  183. package/src/components/tabs/Tabs.tsx +67 -0
  184. package/src/components/tabs/a11y.test.ts +15 -0
  185. package/src/components/tabs/a11y.ts +15 -0
  186. package/src/components/tabs/context.tsx +31 -0
  187. package/src/components/tabs/index.ts +5 -0
  188. package/src/components/tabs/navigation.test.ts +21 -0
  189. package/src/components/tabs/navigation.ts +32 -0
  190. package/src/components/tabs/types.ts +27 -0
  191. package/src/components/text-input/TextInput.tsx +116 -0
  192. package/src/components/text-input/index.ts +2 -0
  193. package/src/components/text-input/types.ts +32 -0
  194. package/src/components/textarea/Textarea.tsx +15 -0
  195. package/src/components/textarea/index.ts +2 -0
  196. package/src/components/textarea/types.ts +5 -0
  197. package/src/components/toast/Toast.tsx +54 -0
  198. package/src/components/toast/ToastProvider.tsx +114 -0
  199. package/src/components/toast/index.ts +3 -0
  200. package/src/components/toast/types.ts +16 -0
  201. package/src/components/tooltip/Tooltip.tsx +109 -0
  202. package/src/components/tooltip/index.ts +2 -0
  203. package/src/components/tooltip/types.ts +9 -0
  204. package/src/context/FontContext.tsx +59 -0
  205. package/src/context/TranslationContext.tsx +54 -0
  206. package/src/core/responsive/ResponsiveProvider.tsx +31 -0
  207. package/src/core/responsive/breakpoints.ts +9 -0
  208. package/src/core/responsive/getBreakpointFromWidth.test.ts +15 -0
  209. package/src/core/responsive/getBreakpointFromWidth.ts +10 -0
  210. package/src/core/responsive/index.ts +6 -0
  211. package/src/core/responsive/resolve.test.ts +25 -0
  212. package/src/core/responsive/resolve.ts +24 -0
  213. package/src/core/responsive/types.ts +10 -0
  214. package/src/core/responsive/useBreakpoint.ts +9 -0
  215. package/src/examples/DocsExamples.tsx +116 -0
  216. package/src/index.test.ts +64 -0
  217. package/src/index.ts +55 -0
  218. package/src/internal/focus/FocusScope.tsx +66 -0
  219. package/src/internal/focus/useFocusManager.test.ts +44 -0
  220. package/src/internal/focus/useFocusManager.ts +142 -0
  221. package/src/internal/overlay/OverlayProvider.tsx +74 -0
  222. package/src/internal/overlay/Portal.tsx +38 -0
  223. package/src/internal/overlay/useOverlayStack.test.ts +31 -0
  224. package/src/internal/overlay/useOverlayStack.ts +61 -0
  225. package/src/internal/resolvers/index.ts +15 -0
  226. package/src/internal/resolvers/resolveControlSize.test.ts +25 -0
  227. package/src/internal/resolvers/resolveControlSize.ts +45 -0
  228. package/src/internal/resolvers/resolveFieldPresentation.test.ts +31 -0
  229. package/src/internal/resolvers/resolveFieldPresentation.ts +30 -0
  230. package/src/internal/resolvers/resolveFieldState.test.ts +22 -0
  231. package/src/internal/resolvers/resolveFieldState.ts +36 -0
  232. package/src/internal/resolvers/resolveFocusRingStyles.ts +14 -0
  233. package/src/internal/resolvers/resolveIconSize.ts +6 -0
  234. package/src/internal/resolvers/resolveIndicatorSize.test.ts +19 -0
  235. package/src/internal/resolvers/resolveIndicatorSize.ts +47 -0
  236. package/src/internal/resolvers/resolveInteractiveColors.test.ts +57 -0
  237. package/src/internal/resolvers/resolveInteractiveColors.ts +134 -0
  238. package/src/internal/resolvers/resolveInteractiveState.test.ts +14 -0
  239. package/src/internal/resolvers/resolveInteractiveState.ts +15 -0
  240. package/src/internal/resolvers/resolveOverlayAnimation.test.ts +15 -0
  241. package/src/internal/resolvers/resolveOverlayAnimation.ts +24 -0
  242. package/src/internal/resolvers/resolveOverlayZIndex.test.ts +15 -0
  243. package/src/internal/resolvers/resolveOverlayZIndex.ts +13 -0
  244. package/src/internal/resolvers/resolveSelectionControlBehavior.test.ts +52 -0
  245. package/src/internal/resolvers/resolveSelectionControlBehavior.ts +23 -0
  246. package/src/internal/resolvers/resolveSelectionControlColors.test.ts +44 -0
  247. package/src/internal/resolvers/resolveSelectionControlColors.ts +81 -0
  248. package/src/internal/resolvers/resolveTextColor.test.ts +23 -0
  249. package/src/internal/resolvers/resolveTextColor.ts +40 -0
  250. package/src/internal/resolvers/resolveTextStyles.test.ts +27 -0
  251. package/src/internal/resolvers/resolveTextStyles.ts +95 -0
  252. package/src/internal/resolvers/resolveTone.ts +19 -0
  253. package/src/internal/useControllableState.ts +28 -0
  254. package/src/layout/Box.tsx +79 -0
  255. package/src/layout/Center.tsx +22 -0
  256. package/src/layout/Container.tsx +43 -0
  257. package/src/layout/Divider.tsx +26 -0
  258. package/src/layout/Grid.tsx +83 -0
  259. package/src/layout/Inline.tsx +9 -0
  260. package/src/layout/Show.tsx +15 -0
  261. package/src/layout/Spacer.tsx +22 -0
  262. package/src/layout/Stack.tsx +67 -0
  263. package/src/layout/Surface.tsx +70 -0
  264. package/src/layout/Template.tsx +85 -0
  265. package/src/layout/helpers.test.ts +71 -0
  266. package/src/layout/helpers.ts +208 -0
  267. package/src/layout/index.ts +22 -0
  268. package/src/primitives/button-base/ButtonBase.tsx +81 -0
  269. package/src/primitives/button-base/index.ts +2 -0
  270. package/src/primitives/button-base/types.ts +16 -0
  271. package/src/primitives/heading/Heading.tsx +60 -0
  272. package/src/primitives/heading/index.ts +2 -0
  273. package/src/primitives/heading/resolveHeadingStyle.test.ts +31 -0
  274. package/src/primitives/heading/resolveHeadingStyle.ts +17 -0
  275. package/src/primitives/heading/types.ts +13 -0
  276. package/src/primitives/icon/Icon.tsx +40 -0
  277. package/src/primitives/icon/index.ts +2 -0
  278. package/src/primitives/icon/resolveExpoIconComponent.test.ts +29 -0
  279. package/src/primitives/icon/resolveExpoIconComponent.ts +20 -0
  280. package/src/primitives/text/Text.tsx +66 -0
  281. package/src/primitives/text/index.ts +2 -0
  282. package/src/primitives/text/types.ts +18 -0
  283. package/src/theme/ThemeContext.tsx +95 -0
  284. package/src/theme/colorEngine.test.ts +114 -0
  285. package/src/theme/colorEngine.ts +480 -0
  286. package/src/theme/createTheme.ts +121 -0
  287. package/src/theme/index.ts +5 -0
  288. package/src/theme/resolveToken.ts +32 -0
  289. package/src/theme/types.ts +188 -0
  290. package/src/utils/deepEqual.ts +34 -0
  291. package/src/utils/deepMerge.test.ts +117 -0
  292. package/src/utils/deepMerge.ts +29 -0
@@ -0,0 +1,92 @@
1
+ import React from 'react';
2
+ import { Pressable, View } from 'react-native';
3
+
4
+ import { FocusScope } from '../../internal/focus/FocusScope';
5
+ import { Portal } from '../../internal/overlay/Portal';
6
+ import { resolveOverlayAnimation } from '../../internal/resolvers';
7
+ import { Box, Surface } from '../../layout';
8
+ import { useTheme } from '../../theme/ThemeContext';
9
+ import type { DrawerProps } from './types';
10
+
11
+ export function Drawer({
12
+ visible,
13
+ onDismiss,
14
+ position = 'right',
15
+ children,
16
+ closeOnBackdrop = true,
17
+ testID,
18
+ }: DrawerProps) {
19
+ const { theme } = useTheme();
20
+ const animation = resolveOverlayAnimation('drawer');
21
+
22
+ if (!visible) {
23
+ return null;
24
+ }
25
+
26
+ return (
27
+ <Portal layer="drawer" visible={visible}>
28
+ <View
29
+ pointerEvents="box-none"
30
+ style={{
31
+ bottom: 0,
32
+ left: 0,
33
+ position: 'absolute',
34
+ right: 0,
35
+ top: 0,
36
+ }}
37
+ >
38
+ <Pressable
39
+ onPress={closeOnBackdrop ? onDismiss : undefined}
40
+ style={{
41
+ backgroundColor: theme.semantics.neutral.text,
42
+ bottom: 0,
43
+ left: 0,
44
+ opacity: animation.backdropOpacity,
45
+ position: 'absolute',
46
+ right: 0,
47
+ top: 0,
48
+ }}
49
+ testID={testID ? `${testID}-backdrop` : undefined}
50
+ />
51
+ <FocusScope
52
+ active={visible}
53
+ onEscape={onDismiss}
54
+ testID={testID ? `${testID}-focus` : undefined}
55
+ >
56
+ <Box
57
+ pointerEvents="box-none"
58
+ style={{
59
+ flex: 1,
60
+ justifyContent: 'flex-start',
61
+ }}
62
+ >
63
+ <View
64
+ accessible
65
+ accessibilityViewIsModal
66
+ style={{
67
+ alignSelf: position === 'left' ? 'flex-start' : 'flex-end',
68
+ height: '100%',
69
+ maxWidth: 420,
70
+ width: '88%',
71
+ }}
72
+ >
73
+ <Surface
74
+ p="l"
75
+ style={{
76
+ height: '100%',
77
+ shadowOpacity: 0.14,
78
+ shadowRadius: 14,
79
+ shadowOffset: { width: position === 'left' ? 4 : -4, height: 0 },
80
+ }}
81
+ testID={testID}
82
+ variant="raised"
83
+ >
84
+ {children}
85
+ </Surface>
86
+ </View>
87
+ </Box>
88
+ </FocusScope>
89
+ </View>
90
+ </Portal>
91
+ );
92
+ }
@@ -0,0 +1,2 @@
1
+ export { Drawer } from './Drawer';
2
+ export type { DrawerProps } from './types';
@@ -0,0 +1,10 @@
1
+ import type React from 'react';
2
+
3
+ export interface DrawerProps {
4
+ visible: boolean;
5
+ onDismiss?: (() => void) | undefined;
6
+ position?: 'left' | 'right';
7
+ children?: React.ReactNode;
8
+ closeOnBackdrop?: boolean;
9
+ testID?: string;
10
+ }
@@ -0,0 +1,43 @@
1
+ import React from 'react';
2
+
3
+ import { resolveFieldPresentation, resolveFieldState } from '../../internal/resolvers';
4
+ import { Box, Stack } from '../../layout';
5
+ import { HelperText } from '../helper-text';
6
+ import { Label } from '../label';
7
+ import type { FieldProps } from './types';
8
+
9
+ export function Field({
10
+ children,
11
+ label,
12
+ helperText,
13
+ errorText,
14
+ required = false,
15
+ disabled = false,
16
+ invalid = false,
17
+ readOnly = false,
18
+ testID,
19
+ }: FieldProps) {
20
+ const hasErrorText = Boolean(errorText);
21
+ const fieldState = resolveFieldState({
22
+ disabled,
23
+ invalid: invalid || hasErrorText,
24
+ readOnly,
25
+ });
26
+ const presentation = resolveFieldPresentation(fieldState);
27
+
28
+ return (
29
+ <Stack gap="xs" testID={testID}>
30
+ {label ? (
31
+ <Label required={required} tone={presentation.labelTone}>
32
+ {label}
33
+ </Label>
34
+ ) : null}
35
+ <Box>{children}</Box>
36
+ {hasErrorText ? (
37
+ <HelperText tone={presentation.helperTone}>{errorText}</HelperText>
38
+ ) : helperText ? (
39
+ <HelperText tone={presentation.helperTone}>{helperText}</HelperText>
40
+ ) : null}
41
+ </Stack>
42
+ );
43
+ }
@@ -0,0 +1,2 @@
1
+ export { Field } from './Field';
2
+ export type { FieldProps } from './types';
@@ -0,0 +1,13 @@
1
+ import type React from 'react';
2
+
3
+ export interface FieldProps {
4
+ children?: React.ReactNode;
5
+ label?: React.ReactNode;
6
+ helperText?: React.ReactNode;
7
+ errorText?: React.ReactNode;
8
+ required?: boolean;
9
+ disabled?: boolean;
10
+ invalid?: boolean;
11
+ readOnly?: boolean;
12
+ testID?: string;
13
+ }
@@ -0,0 +1,12 @@
1
+ import React from 'react';
2
+
3
+ import { Text } from '../../primitives/text';
4
+ import type { HelperTextProps } from './types';
5
+
6
+ export function HelperText({ children, tone = 'muted', testID }: HelperTextProps) {
7
+ return (
8
+ <Text testID={testID} tone={tone} variant="caption">
9
+ {children}
10
+ </Text>
11
+ );
12
+ }
@@ -0,0 +1,2 @@
1
+ export { HelperText } from './HelperText';
2
+ export type { HelperTextProps } from './types';
@@ -0,0 +1,9 @@
1
+ import type React from 'react';
2
+
3
+ import type { TextTone } from '../../internal/resolvers/resolveTextColor';
4
+
5
+ export interface HelperTextProps {
6
+ children?: React.ReactNode;
7
+ tone?: Extract<TextTone, 'default' | 'muted' | 'danger'>;
8
+ testID?: string;
9
+ }
@@ -0,0 +1,60 @@
1
+ import React from 'react';
2
+
3
+ import { resolveButtonColors, resolveControlSize, resolveIconSize } from '../../internal/resolvers';
4
+ import { Box } from '../../layout';
5
+ import { ButtonBase } from '../../primitives/button-base';
6
+ import { Icon } from '../../primitives/icon';
7
+ import { useTheme } from '../../theme/ThemeContext';
8
+ import type { IconButtonProps } from './types';
9
+
10
+ export function IconButton({
11
+ icon,
12
+ accessibilityLabel,
13
+ variant = 'ghost',
14
+ tone = 'primary',
15
+ size = 'm',
16
+ disabled = false,
17
+ onPress,
18
+ testID,
19
+ ...props
20
+ }: IconButtonProps) {
21
+ const { theme } = useTheme();
22
+ const controlSize = resolveControlSize(theme, size);
23
+
24
+ return (
25
+ <ButtonBase
26
+ {...props}
27
+ accessibilityLabel={accessibilityLabel}
28
+ disabled={disabled}
29
+ onPress={onPress}
30
+ testID={testID}
31
+ >
32
+ {(state) => {
33
+ const colors = resolveButtonColors(theme, { state, tone, variant });
34
+
35
+ return (
36
+ <Box
37
+ radius={controlSize.borderRadius}
38
+ style={{
39
+ minHeight: controlSize.minHeight,
40
+ minWidth: controlSize.minHeight,
41
+ alignItems: 'center',
42
+ justifyContent: 'center',
43
+ backgroundColor: colors.backgroundColor,
44
+ borderColor: colors.borderColor,
45
+ borderWidth: variant === 'solid' ? 0 : 1,
46
+ opacity: colors.opacity,
47
+ }}
48
+ >
49
+ <Icon
50
+ color={colors.contentColor}
51
+ name={icon.name}
52
+ provider={icon.provider}
53
+ size={resolveIconSize(theme, size)}
54
+ />
55
+ </Box>
56
+ );
57
+ }}
58
+ </ButtonBase>
59
+ );
60
+ }
@@ -0,0 +1,2 @@
1
+ export { IconButton } from './IconButton';
2
+ export type { IconButtonProps } from './types';
@@ -0,0 +1,19 @@
1
+ import type { ControlSize } from '../../internal/resolvers/resolveControlSize';
2
+ import type { ButtonVariant } from '../../internal/resolvers/resolveInteractiveColors';
3
+ import type { ComponentTone } from '../../internal/resolvers/resolveTone';
4
+ import type { ButtonBaseProps } from '../../primitives/button-base';
5
+ import type { IconProps } from '../../primitives/icon';
6
+
7
+ export interface IconButtonProps extends Omit<
8
+ ButtonBaseProps,
9
+ 'children' | 'style' | 'accessibilityLabel'
10
+ > {
11
+ icon: {
12
+ name: IconProps['name'];
13
+ provider?: IconProps['provider'];
14
+ };
15
+ accessibilityLabel: string;
16
+ variant?: ButtonVariant;
17
+ tone?: ComponentTone;
18
+ size?: ControlSize;
19
+ }
@@ -0,0 +1,17 @@
1
+ import React from 'react';
2
+
3
+ import { Text } from '../../primitives/text';
4
+ import type { LabelProps } from './types';
5
+
6
+ export function Label({ children, required = false, tone = 'default', testID }: LabelProps) {
7
+ return (
8
+ <Text testID={testID} tone={tone} variant="label" weight="medium">
9
+ {children}
10
+ {required ? (
11
+ <Text tone="danger" variant="label" weight="medium">
12
+ {' *'}
13
+ </Text>
14
+ ) : null}
15
+ </Text>
16
+ );
17
+ }
@@ -0,0 +1,2 @@
1
+ export { Label } from './Label';
2
+ export type { LabelProps } from './types';
@@ -0,0 +1,10 @@
1
+ import type React from 'react';
2
+
3
+ import type { TextTone } from '../../internal/resolvers/resolveTextColor';
4
+
5
+ export interface LabelProps {
6
+ children?: React.ReactNode;
7
+ required?: boolean;
8
+ tone?: Extract<TextTone, 'default' | 'muted' | 'danger'>;
9
+ testID?: string;
10
+ }
@@ -0,0 +1,72 @@
1
+ import React from 'react';
2
+ import { View } from 'react-native';
3
+
4
+ import { Box } from '../../layout';
5
+ import { ButtonBase } from '../../primitives/button-base';
6
+ import { Text } from '../../primitives/text';
7
+ import { useTheme } from '../../theme/ThemeContext';
8
+ import type { ListItemProps } from './types';
9
+
10
+ function ListItemContent({
11
+ title,
12
+ description,
13
+ leading,
14
+ trailing,
15
+ testID,
16
+ }: Omit<ListItemProps, 'disabled' | 'onPress'>) {
17
+ const { theme } = useTheme();
18
+
19
+ return (
20
+ <Box
21
+ px="m"
22
+ py="m"
23
+ testID={testID}
24
+ style={{
25
+ flexDirection: 'row',
26
+ alignItems: 'center',
27
+ }}
28
+ >
29
+ {leading ? <View style={{ marginRight: theme.spacing.m }}>{leading}</View> : null}
30
+ <Box flex={1}>
31
+ {title ? (
32
+ <Text numberOfLines={1} variant="body" weight="medium">
33
+ {title}
34
+ </Text>
35
+ ) : null}
36
+ {description ? (
37
+ <Text numberOfLines={2} tone="muted" variant="bodySmall">
38
+ {description}
39
+ </Text>
40
+ ) : null}
41
+ </Box>
42
+ {trailing ? <View style={{ marginLeft: theme.spacing.m }}>{trailing}</View> : null}
43
+ </Box>
44
+ );
45
+ }
46
+
47
+ export function ListItem({ onPress, disabled = false, testID, ...content }: ListItemProps) {
48
+ const { theme } = useTheme();
49
+
50
+ if (!onPress) {
51
+ return <ListItemContent {...content} testID={testID} />;
52
+ }
53
+
54
+ return (
55
+ <ButtonBase accessibilityRole="button" disabled={disabled} onPress={onPress} testID={testID}>
56
+ {(state) => (
57
+ <Box
58
+ style={{
59
+ backgroundColor: state.pressed
60
+ ? theme.semantics.neutral.surfaceActive
61
+ : state.hovered
62
+ ? theme.semantics.neutral.surfaceHover
63
+ : 'transparent',
64
+ opacity: state.disabled ? 0.72 : 1,
65
+ }}
66
+ >
67
+ <ListItemContent {...content} />
68
+ </Box>
69
+ )}
70
+ </ButtonBase>
71
+ );
72
+ }
@@ -0,0 +1,2 @@
1
+ export { ListItem } from './ListItem';
2
+ export type { ListItemProps } from './types';
@@ -0,0 +1,11 @@
1
+ import type React from 'react';
2
+
3
+ export interface ListItemProps {
4
+ title?: React.ReactNode;
5
+ description?: React.ReactNode;
6
+ leading?: React.ReactNode;
7
+ trailing?: React.ReactNode;
8
+ onPress?: (() => void) | undefined;
9
+ disabled?: boolean;
10
+ testID?: string;
11
+ }
@@ -0,0 +1,180 @@
1
+ import React from 'react';
2
+ import { type LayoutRectangle, Pressable, View } from 'react-native';
3
+
4
+ import { FocusScope } from '../../internal/focus/FocusScope';
5
+ import { useFocusManager } from '../../internal/focus/useFocusManager';
6
+ import { Portal } from '../../internal/overlay/Portal';
7
+ import { resolveOverlayAnimation } from '../../internal/resolvers';
8
+ import { Box, Surface } from '../../layout';
9
+ import { ButtonBase } from '../../primitives/button-base';
10
+ import { Text } from '../../primitives/text';
11
+ import { useTheme } from '../../theme/ThemeContext';
12
+ import { resolveNextMenuIndex } from './navigation';
13
+ import type { MenuProps } from './types';
14
+
15
+ interface MeasurableNode {
16
+ measureInWindow?: (
17
+ callback: (x: number, y: number, width: number, height: number) => void,
18
+ ) => void;
19
+ }
20
+
21
+ function measureNode(node: unknown, callback: (layout: LayoutRectangle) => void) {
22
+ const measurableNode = node as MeasurableNode | null;
23
+ measurableNode?.measureInWindow?.((x, y, width, height) => {
24
+ callback({ height, width, x, y });
25
+ });
26
+ }
27
+
28
+ export function Menu({ trigger, items, onDismiss, closeOnSelect = true, testID }: MenuProps) {
29
+ const { theme } = useTheme();
30
+ const { bindKeydown } = useFocusManager();
31
+ const animation = resolveOverlayAnimation('menu');
32
+ const anchorRef = React.useRef<View | null>(null);
33
+ const [open, setOpen] = React.useState(false);
34
+ const [layout, setLayout] = React.useState<LayoutRectangle | null>(null);
35
+ const [activeIndex, setActiveIndex] = React.useState(0);
36
+
37
+ const closeMenu = React.useCallback(() => {
38
+ setOpen(false);
39
+ onDismiss?.();
40
+ }, [onDismiss]);
41
+
42
+ const openMenu = React.useCallback(() => {
43
+ measureNode(anchorRef.current, setLayout);
44
+ const firstEnabledIndex = items.findIndex((item) => !item.disabled);
45
+ setActiveIndex(firstEnabledIndex === -1 ? 0 : firstEnabledIndex);
46
+ setOpen(true);
47
+ }, [items]);
48
+
49
+ React.useEffect(() => {
50
+ if (!open) {
51
+ return undefined;
52
+ }
53
+
54
+ return bindKeydown((event) => {
55
+ const { key } = event;
56
+ if (key === 'ArrowDown' || key === 'ArrowUp' || key === 'Home' || key === 'End') {
57
+ event.preventDefault();
58
+ setActiveIndex((current) => resolveNextMenuIndex(items, current, key));
59
+ }
60
+
61
+ if (event.key === 'Enter') {
62
+ event.preventDefault();
63
+ const activeItem = items[activeIndex];
64
+ if (activeItem && !activeItem.disabled) {
65
+ activeItem.onPress?.();
66
+ if (closeOnSelect) {
67
+ closeMenu();
68
+ }
69
+ }
70
+ }
71
+
72
+ if (event.key === 'Escape') {
73
+ event.preventDefault();
74
+ closeMenu();
75
+ }
76
+ });
77
+ }, [activeIndex, bindKeydown, closeMenu, closeOnSelect, items, open]);
78
+
79
+ return (
80
+ <View collapsable={false} ref={anchorRef}>
81
+ <ButtonBase
82
+ onPress={open ? closeMenu : openMenu}
83
+ testID={testID ? `${testID}-trigger` : undefined}
84
+ >
85
+ {trigger}
86
+ </ButtonBase>
87
+ <Portal layer="menu" visible={open && Boolean(layout)}>
88
+ <View
89
+ pointerEvents="box-none"
90
+ style={{
91
+ bottom: 0,
92
+ left: 0,
93
+ position: 'absolute',
94
+ right: 0,
95
+ top: 0,
96
+ }}
97
+ >
98
+ <Pressable
99
+ onPress={closeMenu}
100
+ style={{
101
+ bottom: 0,
102
+ left: 0,
103
+ position: 'absolute',
104
+ right: 0,
105
+ top: 0,
106
+ }}
107
+ />
108
+ <FocusScope active={open} onEscape={closeMenu}>
109
+ <View
110
+ style={{
111
+ left: layout?.x ?? 0,
112
+ position: 'absolute',
113
+ top: (layout?.y ?? 0) + (layout?.height ?? 0) + animation.offset,
114
+ }}
115
+ >
116
+ <Surface
117
+ accessibilityRole="menu"
118
+ p="xs"
119
+ style={{
120
+ minWidth: Math.max(layout?.width ?? 0, 180),
121
+ shadowOpacity: 0.12,
122
+ shadowRadius: 12,
123
+ shadowOffset: { width: 0, height: 6 },
124
+ }}
125
+ testID={testID}
126
+ variant="raised"
127
+ >
128
+ {items.map((item, index) => {
129
+ const selected = index === activeIndex;
130
+
131
+ return (
132
+ <Pressable
133
+ accessibilityRole="menuitem"
134
+ accessibilityState={{ disabled: item.disabled, selected }}
135
+ disabled={item.disabled}
136
+ key={item.id}
137
+ onPress={() => {
138
+ if (item.disabled) {
139
+ return;
140
+ }
141
+
142
+ item.onPress?.();
143
+ if (closeOnSelect) {
144
+ closeMenu();
145
+ }
146
+ }}
147
+ >
148
+ <Box
149
+ px="m"
150
+ py="s"
151
+ style={{
152
+ backgroundColor: selected
153
+ ? theme.semantics.action.neutral.softBg
154
+ : 'transparent',
155
+ opacity: item.disabled ? 0.56 : 1,
156
+ }}
157
+ testID={testID ? `${testID}-item-${item.id}` : undefined}
158
+ >
159
+ <Text
160
+ color={
161
+ selected
162
+ ? theme.semantics.action.neutral.base
163
+ : theme.semantics.content.default
164
+ }
165
+ variant="bodySmall"
166
+ >
167
+ {item.label}
168
+ </Text>
169
+ </Box>
170
+ </Pressable>
171
+ );
172
+ })}
173
+ </Surface>
174
+ </View>
175
+ </FocusScope>
176
+ </View>
177
+ </Portal>
178
+ </View>
179
+ );
180
+ }
@@ -0,0 +1,2 @@
1
+ export { Menu } from './Menu';
2
+ export type { MenuItem, MenuProps } from './types';
@@ -0,0 +1,21 @@
1
+ import { describe, expect, it } from 'bun:test';
2
+
3
+ import { resolveNextMenuIndex } from './navigation';
4
+
5
+ describe('resolveNextMenuIndex', () => {
6
+ const items = [{}, { disabled: true }, {}, {}];
7
+
8
+ it('cycles down and up through the menu items', () => {
9
+ expect(resolveNextMenuIndex(items, 0, 'ArrowDown')).toBe(2);
10
+ expect(resolveNextMenuIndex(items, 0, 'ArrowUp')).toBe(3);
11
+ });
12
+
13
+ it('supports home and end navigation', () => {
14
+ expect(resolveNextMenuIndex(items, 2, 'Home')).toBe(0);
15
+ expect(resolveNextMenuIndex(items, 0, 'End')).toBe(3);
16
+ });
17
+
18
+ it('returns -1 when all items are disabled', () => {
19
+ expect(resolveNextMenuIndex([{ disabled: true }], 0, 'ArrowDown')).toBe(-1);
20
+ });
21
+ });
@@ -0,0 +1,34 @@
1
+ export function resolveNextMenuIndex(
2
+ items: { disabled?: boolean }[],
3
+ currentIndex: number,
4
+ key: 'ArrowDown' | 'ArrowUp' | 'Home' | 'End',
5
+ ): number {
6
+ const enabledIndexes = items.reduce<number[]>((indexes, item, index) => {
7
+ if (!item.disabled) {
8
+ indexes.push(index);
9
+ }
10
+ return indexes;
11
+ }, []);
12
+
13
+ if (enabledIndexes.length === 0) {
14
+ return -1;
15
+ }
16
+
17
+ if (key === 'Home') {
18
+ return enabledIndexes[0] ?? -1;
19
+ }
20
+
21
+ if (key === 'End') {
22
+ return enabledIndexes[enabledIndexes.length - 1] ?? -1;
23
+ }
24
+
25
+ const activeEnabledIndex = enabledIndexes.indexOf(currentIndex);
26
+ const nextEnabledIndex =
27
+ activeEnabledIndex === -1
28
+ ? 0
29
+ : key === 'ArrowUp'
30
+ ? (activeEnabledIndex - 1 + enabledIndexes.length) % enabledIndexes.length
31
+ : (activeEnabledIndex + 1) % enabledIndexes.length;
32
+
33
+ return enabledIndexes[nextEnabledIndex] ?? -1;
34
+ }
@@ -0,0 +1,16 @@
1
+ import type React from 'react';
2
+
3
+ export interface MenuItem {
4
+ id: string;
5
+ label: React.ReactNode;
6
+ disabled?: boolean;
7
+ onPress?: (() => void) | undefined;
8
+ }
9
+
10
+ export interface MenuProps {
11
+ trigger?: React.ReactNode;
12
+ items: MenuItem[];
13
+ onDismiss?: (() => void) | undefined;
14
+ closeOnSelect?: boolean;
15
+ testID?: string;
16
+ }