@ankhorage/surface 0.1.5 → 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 (291) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/dist/components/badge/Badge.js.map +1 -1
  3. package/dist/components/badge/index.js.map +1 -1
  4. package/dist/components/badge/types.js.map +1 -1
  5. package/dist/components/button/Button.js.map +1 -1
  6. package/dist/components/button/index.js.map +1 -1
  7. package/dist/components/button/types.js.map +1 -1
  8. package/dist/components/card/Card.js.map +1 -1
  9. package/dist/components/card/index.js.map +1 -1
  10. package/dist/components/card/types.js.map +1 -1
  11. package/dist/components/checkbox/Checkbox.js.map +1 -1
  12. package/dist/components/checkbox/index.js.map +1 -1
  13. package/dist/components/checkbox/types.js.map +1 -1
  14. package/dist/components/drawer/Drawer.js.map +1 -1
  15. package/dist/components/drawer/index.js.map +1 -1
  16. package/dist/components/drawer/types.js.map +1 -1
  17. package/dist/components/field/Field.js.map +1 -1
  18. package/dist/components/field/index.js.map +1 -1
  19. package/dist/components/field/types.js.map +1 -1
  20. package/dist/components/helper-text/HelperText.js.map +1 -1
  21. package/dist/components/helper-text/index.js.map +1 -1
  22. package/dist/components/helper-text/types.js.map +1 -1
  23. package/dist/components/icon-button/IconButton.js.map +1 -1
  24. package/dist/components/icon-button/index.js.map +1 -1
  25. package/dist/components/icon-button/types.js.map +1 -1
  26. package/dist/components/label/Label.js.map +1 -1
  27. package/dist/components/label/index.js.map +1 -1
  28. package/dist/components/label/types.js.map +1 -1
  29. package/dist/components/list-item/ListItem.js.map +1 -1
  30. package/dist/components/list-item/index.js.map +1 -1
  31. package/dist/components/list-item/types.js.map +1 -1
  32. package/dist/components/menu/Menu.js.map +1 -1
  33. package/dist/components/menu/index.js.map +1 -1
  34. package/dist/components/menu/navigation.js.map +1 -1
  35. package/dist/components/menu/types.js.map +1 -1
  36. package/dist/components/modal/Modal.js.map +1 -1
  37. package/dist/components/modal/index.js.map +1 -1
  38. package/dist/components/modal/types.js.map +1 -1
  39. package/dist/components/radio/Radio.js.map +1 -1
  40. package/dist/components/radio/index.js.map +1 -1
  41. package/dist/components/radio/types.js.map +1 -1
  42. package/dist/components/switch/Switch.js.map +1 -1
  43. package/dist/components/switch/index.js.map +1 -1
  44. package/dist/components/switch/types.js.map +1 -1
  45. package/dist/components/tabs/Tab.js.map +1 -1
  46. package/dist/components/tabs/TabList.js.map +1 -1
  47. package/dist/components/tabs/TabPanel.js.map +1 -1
  48. package/dist/components/tabs/Tabs.js.map +1 -1
  49. package/dist/components/tabs/a11y.js.map +1 -1
  50. package/dist/components/tabs/context.js.map +1 -1
  51. package/dist/components/tabs/index.js.map +1 -1
  52. package/dist/components/tabs/navigation.js.map +1 -1
  53. package/dist/components/tabs/types.js.map +1 -1
  54. package/dist/components/text-input/TextInput.js.map +1 -1
  55. package/dist/components/text-input/index.js.map +1 -1
  56. package/dist/components/text-input/types.js.map +1 -1
  57. package/dist/components/textarea/Textarea.js.map +1 -1
  58. package/dist/components/textarea/index.js.map +1 -1
  59. package/dist/components/textarea/types.js.map +1 -1
  60. package/dist/components/toast/Toast.js.map +1 -1
  61. package/dist/components/toast/ToastProvider.js.map +1 -1
  62. package/dist/components/toast/index.js.map +1 -1
  63. package/dist/components/toast/types.js.map +1 -1
  64. package/dist/components/tooltip/Tooltip.js.map +1 -1
  65. package/dist/components/tooltip/index.js.map +1 -1
  66. package/dist/components/tooltip/types.js.map +1 -1
  67. package/dist/context/FontContext.js.map +1 -1
  68. package/dist/context/TranslationContext.js.map +1 -1
  69. package/dist/core/responsive/ResponsiveProvider.js.map +1 -1
  70. package/dist/core/responsive/breakpoints.js.map +1 -1
  71. package/dist/core/responsive/getBreakpointFromWidth.js.map +1 -1
  72. package/dist/core/responsive/index.js.map +1 -1
  73. package/dist/core/responsive/resolve.js.map +1 -1
  74. package/dist/core/responsive/types.js.map +1 -1
  75. package/dist/core/responsive/useBreakpoint.js.map +1 -1
  76. package/dist/examples/DocsExamples.js.map +1 -1
  77. package/dist/index.js.map +1 -1
  78. package/dist/internal/focus/FocusScope.js.map +1 -1
  79. package/dist/internal/focus/useFocusManager.js.map +1 -1
  80. package/dist/internal/overlay/OverlayProvider.js.map +1 -1
  81. package/dist/internal/overlay/Portal.js.map +1 -1
  82. package/dist/internal/overlay/useOverlayStack.js.map +1 -1
  83. package/dist/internal/resolvers/index.js.map +1 -1
  84. package/dist/internal/resolvers/resolveControlSize.js.map +1 -1
  85. package/dist/internal/resolvers/resolveFieldPresentation.js.map +1 -1
  86. package/dist/internal/resolvers/resolveFieldState.js.map +1 -1
  87. package/dist/internal/resolvers/resolveFocusRingStyles.js.map +1 -1
  88. package/dist/internal/resolvers/resolveIconSize.js.map +1 -1
  89. package/dist/internal/resolvers/resolveIndicatorSize.js.map +1 -1
  90. package/dist/internal/resolvers/resolveInteractiveColors.js.map +1 -1
  91. package/dist/internal/resolvers/resolveInteractiveState.js.map +1 -1
  92. package/dist/internal/resolvers/resolveOverlayAnimation.js.map +1 -1
  93. package/dist/internal/resolvers/resolveOverlayZIndex.js.map +1 -1
  94. package/dist/internal/resolvers/resolveSelectionControlBehavior.js.map +1 -1
  95. package/dist/internal/resolvers/resolveSelectionControlColors.js.map +1 -1
  96. package/dist/internal/resolvers/resolveTextColor.js.map +1 -1
  97. package/dist/internal/resolvers/resolveTextStyles.js.map +1 -1
  98. package/dist/internal/resolvers/resolveTone.js.map +1 -1
  99. package/dist/internal/useControllableState.js.map +1 -1
  100. package/dist/layout/Box.js.map +1 -1
  101. package/dist/layout/Center.js.map +1 -1
  102. package/dist/layout/Container.js.map +1 -1
  103. package/dist/layout/Divider.js.map +1 -1
  104. package/dist/layout/Grid.js.map +1 -1
  105. package/dist/layout/Inline.js.map +1 -1
  106. package/dist/layout/Show.js.map +1 -1
  107. package/dist/layout/Spacer.js.map +1 -1
  108. package/dist/layout/Stack.js.map +1 -1
  109. package/dist/layout/Surface.js.map +1 -1
  110. package/dist/layout/Template.js.map +1 -1
  111. package/dist/layout/helpers.js.map +1 -1
  112. package/dist/layout/index.js.map +1 -1
  113. package/dist/primitives/button-base/ButtonBase.js.map +1 -1
  114. package/dist/primitives/button-base/index.js.map +1 -1
  115. package/dist/primitives/button-base/types.js.map +1 -1
  116. package/dist/primitives/heading/Heading.js.map +1 -1
  117. package/dist/primitives/heading/index.js.map +1 -1
  118. package/dist/primitives/heading/resolveHeadingStyle.js.map +1 -1
  119. package/dist/primitives/heading/types.js.map +1 -1
  120. package/dist/primitives/icon/Icon.js.map +1 -1
  121. package/dist/primitives/icon/index.js.map +1 -1
  122. package/dist/primitives/icon/resolveExpoIconComponent.js.map +1 -1
  123. package/dist/primitives/text/Text.js.map +1 -1
  124. package/dist/primitives/text/index.js.map +1 -1
  125. package/dist/primitives/text/types.js.map +1 -1
  126. package/dist/theme/ThemeContext.js.map +1 -1
  127. package/dist/theme/colorEngine.js.map +1 -1
  128. package/dist/theme/createTheme.js.map +1 -1
  129. package/dist/theme/index.js.map +1 -1
  130. package/dist/theme/resolveToken.js.map +1 -1
  131. package/dist/theme/types.js.map +1 -1
  132. package/dist/utils/deepEqual.js.map +1 -1
  133. package/dist/utils/deepMerge.js.map +1 -1
  134. package/package.json +4 -1
  135. package/src/components/badge/Badge.tsx +47 -0
  136. package/src/components/badge/index.ts +2 -0
  137. package/src/components/badge/types.ts +13 -0
  138. package/src/components/button/Button.tsx +104 -0
  139. package/src/components/button/index.ts +2 -0
  140. package/src/components/button/types.ts +26 -0
  141. package/src/components/card/Card.tsx +81 -0
  142. package/src/components/card/index.ts +2 -0
  143. package/src/components/card/types.ts +11 -0
  144. package/src/components/checkbox/Checkbox.tsx +111 -0
  145. package/src/components/checkbox/index.ts +2 -0
  146. package/src/components/checkbox/types.ts +19 -0
  147. package/src/components/drawer/Drawer.tsx +92 -0
  148. package/src/components/drawer/index.ts +2 -0
  149. package/src/components/drawer/types.ts +10 -0
  150. package/src/components/field/Field.tsx +43 -0
  151. package/src/components/field/index.ts +2 -0
  152. package/src/components/field/types.ts +13 -0
  153. package/src/components/helper-text/HelperText.tsx +12 -0
  154. package/src/components/helper-text/index.ts +2 -0
  155. package/src/components/helper-text/types.ts +9 -0
  156. package/src/components/icon-button/IconButton.tsx +60 -0
  157. package/src/components/icon-button/index.ts +2 -0
  158. package/src/components/icon-button/types.ts +19 -0
  159. package/src/components/label/Label.tsx +17 -0
  160. package/src/components/label/index.ts +2 -0
  161. package/src/components/label/types.ts +10 -0
  162. package/src/components/list-item/ListItem.tsx +72 -0
  163. package/src/components/list-item/index.ts +2 -0
  164. package/src/components/list-item/types.ts +11 -0
  165. package/src/components/menu/Menu.tsx +180 -0
  166. package/src/components/menu/index.ts +2 -0
  167. package/src/components/menu/navigation.test.ts +21 -0
  168. package/src/components/menu/navigation.ts +34 -0
  169. package/src/components/menu/types.ts +16 -0
  170. package/src/components/modal/Modal.tsx +87 -0
  171. package/src/components/modal/index.ts +2 -0
  172. package/src/components/modal/types.ts +9 -0
  173. package/src/components/radio/Radio.tsx +116 -0
  174. package/src/components/radio/index.ts +2 -0
  175. package/src/components/radio/types.ts +19 -0
  176. package/src/components/switch/Switch.tsx +116 -0
  177. package/src/components/switch/index.ts +2 -0
  178. package/src/components/switch/types.ts +19 -0
  179. package/src/components/tabs/Tab.tsx +82 -0
  180. package/src/components/tabs/TabList.tsx +51 -0
  181. package/src/components/tabs/TabPanel.tsx +29 -0
  182. package/src/components/tabs/Tabs.tsx +67 -0
  183. package/src/components/tabs/a11y.test.ts +15 -0
  184. package/src/components/tabs/a11y.ts +15 -0
  185. package/src/components/tabs/context.tsx +31 -0
  186. package/src/components/tabs/index.ts +5 -0
  187. package/src/components/tabs/navigation.test.ts +21 -0
  188. package/src/components/tabs/navigation.ts +32 -0
  189. package/src/components/tabs/types.ts +27 -0
  190. package/src/components/text-input/TextInput.tsx +116 -0
  191. package/src/components/text-input/index.ts +2 -0
  192. package/src/components/text-input/types.ts +32 -0
  193. package/src/components/textarea/Textarea.tsx +15 -0
  194. package/src/components/textarea/index.ts +2 -0
  195. package/src/components/textarea/types.ts +5 -0
  196. package/src/components/toast/Toast.tsx +54 -0
  197. package/src/components/toast/ToastProvider.tsx +114 -0
  198. package/src/components/toast/index.ts +3 -0
  199. package/src/components/toast/types.ts +16 -0
  200. package/src/components/tooltip/Tooltip.tsx +109 -0
  201. package/src/components/tooltip/index.ts +2 -0
  202. package/src/components/tooltip/types.ts +9 -0
  203. package/src/context/FontContext.tsx +59 -0
  204. package/src/context/TranslationContext.tsx +54 -0
  205. package/src/core/responsive/ResponsiveProvider.tsx +31 -0
  206. package/src/core/responsive/breakpoints.ts +9 -0
  207. package/src/core/responsive/getBreakpointFromWidth.test.ts +15 -0
  208. package/src/core/responsive/getBreakpointFromWidth.ts +10 -0
  209. package/src/core/responsive/index.ts +6 -0
  210. package/src/core/responsive/resolve.test.ts +25 -0
  211. package/src/core/responsive/resolve.ts +24 -0
  212. package/src/core/responsive/types.ts +10 -0
  213. package/src/core/responsive/useBreakpoint.ts +9 -0
  214. package/src/examples/DocsExamples.tsx +116 -0
  215. package/src/index.test.ts +64 -0
  216. package/src/index.ts +55 -0
  217. package/src/internal/focus/FocusScope.tsx +66 -0
  218. package/src/internal/focus/useFocusManager.test.ts +44 -0
  219. package/src/internal/focus/useFocusManager.ts +142 -0
  220. package/src/internal/overlay/OverlayProvider.tsx +74 -0
  221. package/src/internal/overlay/Portal.tsx +38 -0
  222. package/src/internal/overlay/useOverlayStack.test.ts +31 -0
  223. package/src/internal/overlay/useOverlayStack.ts +61 -0
  224. package/src/internal/resolvers/index.ts +15 -0
  225. package/src/internal/resolvers/resolveControlSize.test.ts +25 -0
  226. package/src/internal/resolvers/resolveControlSize.ts +45 -0
  227. package/src/internal/resolvers/resolveFieldPresentation.test.ts +31 -0
  228. package/src/internal/resolvers/resolveFieldPresentation.ts +30 -0
  229. package/src/internal/resolvers/resolveFieldState.test.ts +22 -0
  230. package/src/internal/resolvers/resolveFieldState.ts +36 -0
  231. package/src/internal/resolvers/resolveFocusRingStyles.ts +14 -0
  232. package/src/internal/resolvers/resolveIconSize.ts +6 -0
  233. package/src/internal/resolvers/resolveIndicatorSize.test.ts +19 -0
  234. package/src/internal/resolvers/resolveIndicatorSize.ts +47 -0
  235. package/src/internal/resolvers/resolveInteractiveColors.test.ts +57 -0
  236. package/src/internal/resolvers/resolveInteractiveColors.ts +134 -0
  237. package/src/internal/resolvers/resolveInteractiveState.test.ts +14 -0
  238. package/src/internal/resolvers/resolveInteractiveState.ts +15 -0
  239. package/src/internal/resolvers/resolveOverlayAnimation.test.ts +15 -0
  240. package/src/internal/resolvers/resolveOverlayAnimation.ts +24 -0
  241. package/src/internal/resolvers/resolveOverlayZIndex.test.ts +15 -0
  242. package/src/internal/resolvers/resolveOverlayZIndex.ts +13 -0
  243. package/src/internal/resolvers/resolveSelectionControlBehavior.test.ts +52 -0
  244. package/src/internal/resolvers/resolveSelectionControlBehavior.ts +23 -0
  245. package/src/internal/resolvers/resolveSelectionControlColors.test.ts +44 -0
  246. package/src/internal/resolvers/resolveSelectionControlColors.ts +81 -0
  247. package/src/internal/resolvers/resolveTextColor.test.ts +23 -0
  248. package/src/internal/resolvers/resolveTextColor.ts +40 -0
  249. package/src/internal/resolvers/resolveTextStyles.test.ts +27 -0
  250. package/src/internal/resolvers/resolveTextStyles.ts +95 -0
  251. package/src/internal/resolvers/resolveTone.ts +19 -0
  252. package/src/internal/useControllableState.ts +28 -0
  253. package/src/layout/Box.tsx +79 -0
  254. package/src/layout/Center.tsx +22 -0
  255. package/src/layout/Container.tsx +43 -0
  256. package/src/layout/Divider.tsx +26 -0
  257. package/src/layout/Grid.tsx +83 -0
  258. package/src/layout/Inline.tsx +9 -0
  259. package/src/layout/Show.tsx +15 -0
  260. package/src/layout/Spacer.tsx +22 -0
  261. package/src/layout/Stack.tsx +67 -0
  262. package/src/layout/Surface.tsx +70 -0
  263. package/src/layout/Template.tsx +85 -0
  264. package/src/layout/helpers.test.ts +71 -0
  265. package/src/layout/helpers.ts +208 -0
  266. package/src/layout/index.ts +22 -0
  267. package/src/primitives/button-base/ButtonBase.tsx +81 -0
  268. package/src/primitives/button-base/index.ts +2 -0
  269. package/src/primitives/button-base/types.ts +16 -0
  270. package/src/primitives/heading/Heading.tsx +60 -0
  271. package/src/primitives/heading/index.ts +2 -0
  272. package/src/primitives/heading/resolveHeadingStyle.test.ts +31 -0
  273. package/src/primitives/heading/resolveHeadingStyle.ts +17 -0
  274. package/src/primitives/heading/types.ts +13 -0
  275. package/src/primitives/icon/Icon.tsx +40 -0
  276. package/src/primitives/icon/index.ts +2 -0
  277. package/src/primitives/icon/resolveExpoIconComponent.test.ts +29 -0
  278. package/src/primitives/icon/resolveExpoIconComponent.ts +20 -0
  279. package/src/primitives/text/Text.tsx +66 -0
  280. package/src/primitives/text/index.ts +2 -0
  281. package/src/primitives/text/types.ts +18 -0
  282. package/src/theme/ThemeContext.tsx +95 -0
  283. package/src/theme/colorEngine.test.ts +114 -0
  284. package/src/theme/colorEngine.ts +480 -0
  285. package/src/theme/createTheme.ts +121 -0
  286. package/src/theme/index.ts +5 -0
  287. package/src/theme/resolveToken.ts +32 -0
  288. package/src/theme/types.ts +188 -0
  289. package/src/utils/deepEqual.ts +34 -0
  290. package/src/utils/deepMerge.test.ts +117 -0
  291. package/src/utils/deepMerge.ts +29 -0
@@ -0,0 +1,47 @@
1
+ import React from 'react';
2
+
3
+ import { resolveButtonColors, resolveControlSize } from '../../internal/resolvers';
4
+ import { Box } from '../../layout';
5
+ import { Text } from '../../primitives/text';
6
+ import { useTheme } from '../../theme/ThemeContext';
7
+ import type { BadgeProps } from './types';
8
+
9
+ export function Badge({
10
+ content,
11
+ variant = 'soft',
12
+ tone = 'primary',
13
+ size = 's',
14
+ testID,
15
+ }: BadgeProps) {
16
+ const { theme } = useTheme();
17
+ const controlSize = resolveControlSize(theme, size);
18
+ const colors = resolveButtonColors(theme, {
19
+ variant,
20
+ tone,
21
+ state: {
22
+ disabled: false,
23
+ focused: false,
24
+ hovered: false,
25
+ pressed: false,
26
+ },
27
+ });
28
+
29
+ return (
30
+ <Box
31
+ px={Math.max(theme.spacing.xs, controlSize.paddingHorizontal - 4)}
32
+ py={Math.max(4, controlSize.paddingVertical - 2)}
33
+ radius={controlSize.borderRadius}
34
+ style={{
35
+ alignSelf: 'flex-start',
36
+ backgroundColor: colors.backgroundColor,
37
+ borderColor: colors.borderColor,
38
+ borderWidth: variant === 'solid' ? 0 : 1,
39
+ }}
40
+ testID={testID}
41
+ >
42
+ <Text color={colors.contentColor} variant="bodySmall" weight="medium">
43
+ {content}
44
+ </Text>
45
+ </Box>
46
+ );
47
+ }
@@ -0,0 +1,2 @@
1
+ export { Badge } from './Badge';
2
+ export type { BadgeProps } from './types';
@@ -0,0 +1,13 @@
1
+ import type React from 'react';
2
+
3
+ import type { ControlSize } from '../../internal/resolvers/resolveControlSize';
4
+ import type { ButtonVariant } from '../../internal/resolvers/resolveInteractiveColors';
5
+ import type { ComponentTone } from '../../internal/resolvers/resolveTone';
6
+
7
+ export interface BadgeProps {
8
+ content?: React.ReactNode;
9
+ variant?: Extract<ButtonVariant, 'solid' | 'soft' | 'outline'>;
10
+ tone?: Extract<ComponentTone, 'primary' | 'neutral' | 'danger' | 'success' | 'warning'>;
11
+ size?: ControlSize;
12
+ testID?: string;
13
+ }
@@ -0,0 +1,104 @@
1
+ import React from 'react';
2
+ import { ActivityIndicator, View } from 'react-native';
3
+
4
+ import { resolveButtonColors, resolveControlSize, resolveIconSize } from '../../internal/resolvers';
5
+ import { Box } from '../../layout';
6
+ import { ButtonBase } from '../../primitives/button-base';
7
+ import { Icon } from '../../primitives/icon';
8
+ import { Text } from '../../primitives/text';
9
+ import { useTheme } from '../../theme/ThemeContext';
10
+ import type { ButtonProps } from './types';
11
+
12
+ export function Button({
13
+ children,
14
+ variant = 'solid',
15
+ tone = 'primary',
16
+ size = 'm',
17
+ disabled = false,
18
+ loading = false,
19
+ leadingIcon,
20
+ trailingIcon,
21
+ fullWidth = false,
22
+ onPress,
23
+ testID,
24
+ ...props
25
+ }: ButtonProps) {
26
+ const { theme } = useTheme();
27
+ const controlSize = resolveControlSize(theme, size);
28
+
29
+ return (
30
+ <ButtonBase
31
+ {...props}
32
+ disabled={disabled || loading}
33
+ onPress={onPress}
34
+ style={[
35
+ {
36
+ alignSelf: fullWidth ? 'stretch' : 'flex-start',
37
+ },
38
+ ]}
39
+ testID={testID}
40
+ >
41
+ {(state) => {
42
+ const colors = resolveButtonColors(theme, {
43
+ variant,
44
+ tone,
45
+ state,
46
+ });
47
+ const iconSize = resolveIconSize(theme, size);
48
+
49
+ return (
50
+ <Box
51
+ px={controlSize.paddingHorizontal}
52
+ py={controlSize.paddingVertical}
53
+ radius={controlSize.borderRadius}
54
+ style={{
55
+ minHeight: controlSize.minHeight,
56
+ width: fullWidth ? '100%' : undefined,
57
+ flexDirection: 'row',
58
+ alignItems: 'center',
59
+ justifyContent: 'center',
60
+ backgroundColor: colors.backgroundColor,
61
+ borderColor: colors.borderColor,
62
+ borderWidth: variant === 'solid' ? 0 : 1,
63
+ opacity: colors.opacity,
64
+ }}
65
+ >
66
+ {loading ? (
67
+ <ActivityIndicator color={colors.contentColor} size="small" />
68
+ ) : (
69
+ <>
70
+ {leadingIcon ? (
71
+ <View style={{ marginRight: theme.spacing.s }}>
72
+ <Icon
73
+ color={colors.contentColor}
74
+ name={leadingIcon.name}
75
+ provider={leadingIcon.provider}
76
+ size={iconSize}
77
+ />
78
+ </View>
79
+ ) : null}
80
+ <Text
81
+ color={colors.contentColor}
82
+ variant={controlSize.textVariant}
83
+ weight="semiBold"
84
+ >
85
+ {children}
86
+ </Text>
87
+ {trailingIcon ? (
88
+ <View style={{ marginLeft: theme.spacing.s }}>
89
+ <Icon
90
+ color={colors.contentColor}
91
+ name={trailingIcon.name}
92
+ provider={trailingIcon.provider}
93
+ size={iconSize}
94
+ />
95
+ </View>
96
+ ) : null}
97
+ </>
98
+ )}
99
+ </Box>
100
+ );
101
+ }}
102
+ </ButtonBase>
103
+ );
104
+ }
@@ -0,0 +1,2 @@
1
+ export { Button } from './Button';
2
+ export type { ButtonIconSpec, ButtonProps } from './types';
@@ -0,0 +1,26 @@
1
+ import type React from 'react';
2
+
3
+ import type { ControlSize } from '../../internal/resolvers/resolveControlSize';
4
+ import type { ButtonVariant } from '../../internal/resolvers/resolveInteractiveColors';
5
+ import type { ComponentTone } from '../../internal/resolvers/resolveTone';
6
+ import type { ButtonBaseProps } from '../../primitives/button-base';
7
+ import type { IconProps } from '../../primitives/icon';
8
+
9
+ export interface ButtonIconSpec {
10
+ name: IconProps['name'];
11
+ provider?: IconProps['provider'];
12
+ }
13
+
14
+ export interface ButtonProps extends Omit<
15
+ ButtonBaseProps,
16
+ 'accessibilityLabel' | 'children' | 'style'
17
+ > {
18
+ children?: React.ReactNode;
19
+ variant?: ButtonVariant;
20
+ tone?: ComponentTone;
21
+ size?: ControlSize;
22
+ loading?: boolean;
23
+ leadingIcon?: ButtonIconSpec;
24
+ trailingIcon?: ButtonIconSpec;
25
+ fullWidth?: boolean;
26
+ }
@@ -0,0 +1,81 @@
1
+ import React from 'react';
2
+
3
+ import { Surface } from '../../layout';
4
+ import { ButtonBase } from '../../primitives/button-base';
5
+ import { useTheme } from '../../theme/ThemeContext';
6
+ import type { CardProps } from './types';
7
+
8
+ function resolveCardStateBackground(
9
+ variant: NonNullable<CardProps['variant']>,
10
+ pressed: boolean,
11
+ hovered: boolean,
12
+ themeBackgrounds: {
13
+ base: string;
14
+ subtle: string;
15
+ hover: string;
16
+ active: string;
17
+ },
18
+ ) {
19
+ if (variant === 'outline') {
20
+ return pressed ? themeBackgrounds.active : hovered ? themeBackgrounds.hover : 'transparent';
21
+ }
22
+
23
+ if (variant === 'subtle') {
24
+ return pressed
25
+ ? themeBackgrounds.active
26
+ : hovered
27
+ ? themeBackgrounds.hover
28
+ : themeBackgrounds.subtle;
29
+ }
30
+
31
+ return pressed
32
+ ? themeBackgrounds.active
33
+ : hovered
34
+ ? themeBackgrounds.hover
35
+ : themeBackgrounds.base;
36
+ }
37
+
38
+ export function Card({
39
+ children,
40
+ variant = 'default',
41
+ onPress,
42
+ disabled = false,
43
+ testID,
44
+ style,
45
+ ...props
46
+ }: CardProps) {
47
+ const { theme } = useTheme();
48
+
49
+ if (!onPress) {
50
+ return (
51
+ <Surface {...props} style={style} testID={testID} variant={variant}>
52
+ {children}
53
+ </Surface>
54
+ );
55
+ }
56
+
57
+ return (
58
+ <ButtonBase accessibilityRole="button" disabled={disabled} onPress={onPress} testID={testID}>
59
+ {(state) => (
60
+ <Surface
61
+ {...props}
62
+ style={[
63
+ {
64
+ backgroundColor: resolveCardStateBackground(variant, state.pressed, state.hovered, {
65
+ active: theme.semantics.neutral.surfaceActive,
66
+ base: theme.semantics.surface.default,
67
+ hover: theme.semantics.neutral.surfaceHover,
68
+ subtle: theme.semantics.surface.subtle,
69
+ }),
70
+ opacity: state.disabled ? 0.72 : 1,
71
+ },
72
+ style,
73
+ ]}
74
+ variant={variant}
75
+ >
76
+ {children}
77
+ </Surface>
78
+ )}
79
+ </ButtonBase>
80
+ );
81
+ }
@@ -0,0 +1,2 @@
1
+ export { Card } from './Card';
2
+ export type { CardProps } from './types';
@@ -0,0 +1,11 @@
1
+ import type React from 'react';
2
+
3
+ import type { SurfaceProps, SurfaceVariant } from '../../layout';
4
+
5
+ export interface CardProps extends Omit<SurfaceProps, 'children' | 'variant'> {
6
+ children?: React.ReactNode;
7
+ variant?: SurfaceVariant;
8
+ onPress?: (() => void) | undefined;
9
+ disabled?: boolean;
10
+ testID?: string;
11
+ }
@@ -0,0 +1,111 @@
1
+ import React from 'react';
2
+
3
+ import {
4
+ resolveFieldState,
5
+ resolveIndicatorSize,
6
+ resolveSelectionControlColors,
7
+ resolveSelectionControlNextChecked,
8
+ } from '../../internal/resolvers';
9
+ import { useControllableState } from '../../internal/useControllableState';
10
+ import { Box } from '../../layout';
11
+ import { ButtonBase } from '../../primitives/button-base';
12
+ import { Text } from '../../primitives/text';
13
+ import { useTheme } from '../../theme/ThemeContext';
14
+ import type { CheckboxProps } from './types';
15
+
16
+ export function Checkbox({
17
+ children,
18
+ checked,
19
+ defaultChecked = false,
20
+ onCheckedChange,
21
+ tone = 'primary',
22
+ size = 'm',
23
+ disabled = false,
24
+ invalid = false,
25
+ readOnly = false,
26
+ accessibilityLabel,
27
+ testID,
28
+ ...props
29
+ }: CheckboxProps) {
30
+ const { theme } = useTheme();
31
+ const [isChecked, setChecked] = useControllableState<boolean>({
32
+ value: checked,
33
+ defaultValue: defaultChecked,
34
+ onChange: onCheckedChange,
35
+ });
36
+ const indicatorSize = resolveIndicatorSize(size);
37
+ const nextChecked = resolveSelectionControlNextChecked({
38
+ checked: isChecked,
39
+ disabled,
40
+ kind: 'checkbox',
41
+ readOnly,
42
+ });
43
+
44
+ return (
45
+ <ButtonBase
46
+ {...props}
47
+ accessibilityLabel={accessibilityLabel}
48
+ accessibilityRole="checkbox"
49
+ accessibilityState={{ checked: isChecked }}
50
+ disabled={disabled}
51
+ onPress={
52
+ nextChecked === null
53
+ ? undefined
54
+ : () => {
55
+ setChecked(nextChecked);
56
+ }
57
+ }
58
+ testID={testID}
59
+ >
60
+ {(interactionState) => {
61
+ const fieldState = resolveFieldState({
62
+ disabled,
63
+ focused: interactionState.focused,
64
+ invalid,
65
+ readOnly,
66
+ });
67
+ const colors = resolveSelectionControlColors(theme, {
68
+ checked: isChecked,
69
+ fieldState,
70
+ hovered: interactionState.hovered,
71
+ pressed: interactionState.pressed,
72
+ tone,
73
+ });
74
+
75
+ return (
76
+ <Box
77
+ style={{
78
+ alignItems: 'center',
79
+ flexDirection: 'row',
80
+ opacity: colors.opacity,
81
+ }}
82
+ >
83
+ <Box
84
+ radius="s"
85
+ style={{
86
+ alignItems: 'center',
87
+ backgroundColor: colors.backgroundColor,
88
+ borderColor: colors.borderColor,
89
+ borderWidth: 1.5,
90
+ height: indicatorSize.checkbox,
91
+ justifyContent: 'center',
92
+ width: indicatorSize.checkbox,
93
+ }}
94
+ >
95
+ {isChecked ? (
96
+ <Text color={colors.indicatorColor} variant="caption" weight="bold">
97
+
98
+ </Text>
99
+ ) : null}
100
+ </Box>
101
+ {children ? (
102
+ <Box ml="s">
103
+ <Text color={colors.labelColor}>{children}</Text>
104
+ </Box>
105
+ ) : null}
106
+ </Box>
107
+ );
108
+ }}
109
+ </ButtonBase>
110
+ );
111
+ }
@@ -0,0 +1,2 @@
1
+ export { Checkbox } from './Checkbox';
2
+ export type { CheckboxProps } from './types';
@@ -0,0 +1,19 @@
1
+ import type React from 'react';
2
+
3
+ import type { ControlSize } from '../../internal/resolvers/resolveControlSize';
4
+ import type { ComponentTone } from '../../internal/resolvers/resolveTone';
5
+ import type { ButtonBaseProps } from '../../primitives/button-base';
6
+
7
+ export interface CheckboxProps extends Omit<
8
+ ButtonBaseProps,
9
+ 'accessibilityRole' | 'accessibilityState' | 'children' | 'onPress' | 'style'
10
+ > {
11
+ children?: React.ReactNode;
12
+ checked?: boolean;
13
+ defaultChecked?: boolean;
14
+ onCheckedChange?: ((checked: boolean) => void) | undefined;
15
+ tone?: ComponentTone;
16
+ size?: ControlSize;
17
+ invalid?: boolean;
18
+ readOnly?: boolean;
19
+ }
@@ -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
+ }