@dxos/react-ui 0.8.4-main.28f8d3d → 0.8.4-main.2c6827d

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 (253) hide show
  1. package/dist/lib/browser/{chunk-BKNGMGOK.mjs → chunk-N5GDJTT2.mjs} +640 -306
  2. package/dist/lib/browser/chunk-N5GDJTT2.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +12 -1
  4. package/dist/lib/browser/index.mjs.map +2 -2
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/testing/index.mjs +57 -29
  7. package/dist/lib/browser/testing/index.mjs.map +4 -4
  8. package/dist/lib/node-esm/{chunk-D2HZS6F4.mjs → chunk-SP7VQH72.mjs} +640 -306
  9. package/dist/lib/node-esm/chunk-SP7VQH72.mjs.map +7 -0
  10. package/dist/lib/node-esm/index.mjs +12 -1
  11. package/dist/lib/node-esm/index.mjs.map +2 -2
  12. package/dist/lib/node-esm/meta.json +1 -1
  13. package/dist/lib/node-esm/testing/index.mjs +57 -29
  14. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  15. package/dist/types/src/components/Avatars/Avatar.stories.d.ts +5 -31
  16. package/dist/types/src/components/Avatars/Avatar.stories.d.ts.map +1 -1
  17. package/dist/types/src/components/Avatars/AvatarGroup.stories.d.ts +5 -11
  18. package/dist/types/src/components/Avatars/AvatarGroup.stories.d.ts.map +1 -1
  19. package/dist/types/src/components/Breadcrumb/Breadcrumb.stories.d.ts +8 -20
  20. package/dist/types/src/components/Breadcrumb/Breadcrumb.stories.d.ts.map +1 -1
  21. package/dist/types/src/components/{Buttons → Button}/Button.d.ts +1 -1
  22. package/dist/types/src/components/Button/Button.d.ts.map +1 -0
  23. package/dist/types/src/components/Button/Button.stories.d.ts +17 -0
  24. package/dist/types/src/components/Button/Button.stories.d.ts.map +1 -0
  25. package/dist/types/src/components/{Buttons → Button}/IconButton.d.ts +1 -2
  26. package/dist/types/src/components/Button/IconButton.d.ts.map +1 -0
  27. package/dist/types/src/components/Button/IconButton.stories.d.ts +13 -0
  28. package/dist/types/src/components/Button/IconButton.stories.d.ts.map +1 -0
  29. package/dist/types/src/components/Button/Toggle.d.ts.map +1 -0
  30. package/dist/types/src/components/Button/Toggle.stories.d.ts +16 -0
  31. package/dist/types/src/components/Button/Toggle.stories.d.ts.map +1 -0
  32. package/dist/types/src/components/{Buttons → Button}/ToggleGroup.d.ts +6 -7
  33. package/dist/types/src/components/Button/ToggleGroup.d.ts.map +1 -0
  34. package/dist/types/src/components/Button/ToggleGroup.stories.d.ts +27 -0
  35. package/dist/types/src/components/Button/ToggleGroup.stories.d.ts.map +1 -0
  36. package/dist/types/src/components/Button/index.d.ts.map +1 -0
  37. package/dist/types/src/components/Clipboard/CopyButton.d.ts +1 -1
  38. package/dist/types/src/components/Clipboard/CopyButton.d.ts.map +1 -1
  39. package/dist/types/src/components/Dialog/AlertDialog.d.ts.map +1 -0
  40. package/dist/types/src/components/Dialog/AlertDialog.stories.d.ts +11 -0
  41. package/dist/types/src/components/Dialog/AlertDialog.stories.d.ts.map +1 -0
  42. package/dist/types/src/components/Dialog/Dialog.d.ts.map +1 -0
  43. package/dist/types/src/components/Dialog/Dialog.stories.d.ts +25 -0
  44. package/dist/types/src/components/Dialog/Dialog.stories.d.ts.map +1 -0
  45. package/dist/types/src/components/Dialog/index.d.ts.map +1 -0
  46. package/dist/types/src/components/Icon/Icon.d.ts +1 -1
  47. package/dist/types/src/components/Icon/Icon.d.ts.map +1 -1
  48. package/dist/types/src/components/Icon/Icon.stories.d.ts +17 -0
  49. package/dist/types/src/components/Icon/Icon.stories.d.ts.map +1 -0
  50. package/dist/types/src/components/Input/Input.d.ts +0 -2
  51. package/dist/types/src/components/Input/Input.d.ts.map +1 -1
  52. package/dist/types/src/components/Input/Input.stories.d.ts +9 -10
  53. package/dist/types/src/components/Input/Input.stories.d.ts.map +1 -1
  54. package/dist/types/src/components/Link/Link.stories.d.ts +8 -8
  55. package/dist/types/src/components/Link/Link.stories.d.ts.map +1 -1
  56. package/dist/types/src/components/List/List.d.ts.map +1 -0
  57. package/dist/types/src/components/List/List.stories.d.ts +14 -0
  58. package/dist/types/src/components/List/List.stories.d.ts.map +1 -0
  59. package/dist/types/src/components/List/ListDropIndicator.d.ts.map +1 -0
  60. package/dist/types/src/components/List/Tree.d.ts.map +1 -0
  61. package/dist/types/src/components/List/Tree.stories.d.ts +15 -0
  62. package/dist/types/src/components/List/Tree.stories.d.ts.map +1 -0
  63. package/dist/types/src/components/List/TreeDropIndicator.d.ts.map +1 -0
  64. package/dist/types/src/components/{Lists → List}/Treegrid.d.ts.map +1 -1
  65. package/dist/types/src/components/List/Treegrid.stories.d.ts +12 -0
  66. package/dist/types/src/components/List/Treegrid.stories.d.ts.map +1 -0
  67. package/dist/types/src/components/List/index.d.ts.map +1 -0
  68. package/dist/types/src/components/Main/Main.d.ts +9 -18
  69. package/dist/types/src/components/Main/Main.d.ts.map +1 -1
  70. package/dist/types/src/components/Main/Main.stories.d.ts +6 -7
  71. package/dist/types/src/components/Main/Main.stories.d.ts.map +1 -1
  72. package/dist/types/src/components/Menus/ContextMenu.d.ts.map +1 -1
  73. package/dist/types/src/components/Menus/ContextMenu.stories.d.ts +6 -44
  74. package/dist/types/src/components/Menus/ContextMenu.stories.d.ts.map +1 -1
  75. package/dist/types/src/components/Menus/DropdownMenu.d.ts +7 -7
  76. package/dist/types/src/components/Menus/DropdownMenu.d.ts.map +1 -1
  77. package/dist/types/src/components/Menus/DropdownMenu.stories.d.ts +6 -41
  78. package/dist/types/src/components/Menus/DropdownMenu.stories.d.ts.map +1 -1
  79. package/dist/types/src/components/Message/Message.stories.d.ts +7 -16
  80. package/dist/types/src/components/Message/Message.stories.d.ts.map +1 -1
  81. package/dist/types/src/components/Popover/Popover.d.ts +1 -1
  82. package/dist/types/src/components/Popover/Popover.d.ts.map +1 -1
  83. package/dist/types/src/components/Popover/Popover.stories.d.ts +6 -34
  84. package/dist/types/src/components/Popover/Popover.stories.d.ts.map +1 -1
  85. package/dist/types/src/components/ScrollArea/ScrollArea.stories.d.ts +6 -32
  86. package/dist/types/src/components/ScrollArea/ScrollArea.stories.d.ts.map +1 -1
  87. package/dist/types/src/components/ScrollContainer/ScrollContainer.d.ts +37 -0
  88. package/dist/types/src/components/ScrollContainer/ScrollContainer.d.ts.map +1 -0
  89. package/dist/types/src/components/ScrollContainer/ScrollContainer.stories.d.ts +18 -0
  90. package/dist/types/src/components/ScrollContainer/ScrollContainer.stories.d.ts.map +1 -0
  91. package/dist/types/src/components/ScrollContainer/index.d.ts +2 -0
  92. package/dist/types/src/components/ScrollContainer/index.d.ts.map +1 -0
  93. package/dist/types/src/components/Select/Select.d.ts +1 -1
  94. package/dist/types/src/components/Select/Select.d.ts.map +1 -1
  95. package/dist/types/src/components/Select/Select.stories.d.ts +4 -9
  96. package/dist/types/src/components/Select/Select.stories.d.ts.map +1 -1
  97. package/dist/types/src/components/Status/Status.stories.d.ts +2 -8
  98. package/dist/types/src/components/Status/Status.stories.d.ts.map +1 -1
  99. package/dist/types/src/components/Tag/Tag.stories.d.ts +12 -12
  100. package/dist/types/src/components/Tag/Tag.stories.d.ts.map +1 -1
  101. package/dist/types/src/components/Toast/Toast.stories.d.ts +6 -44
  102. package/dist/types/src/components/Toast/Toast.stories.d.ts.map +1 -1
  103. package/dist/types/src/components/Toolbar/Toolbar.d.ts +12 -12
  104. package/dist/types/src/components/Toolbar/Toolbar.d.ts.map +1 -1
  105. package/dist/types/src/components/Toolbar/Toolbar.stories.d.ts +6 -50
  106. package/dist/types/src/components/Toolbar/Toolbar.stories.d.ts.map +1 -1
  107. package/dist/types/src/components/Tooltip/Tooltip.d.ts +1 -3
  108. package/dist/types/src/components/Tooltip/Tooltip.d.ts.map +1 -1
  109. package/dist/types/src/components/Tooltip/Tooltip.stories.d.ts +8 -61
  110. package/dist/types/src/components/Tooltip/Tooltip.stories.d.ts.map +1 -1
  111. package/dist/types/src/components/index.d.ts +4 -3
  112. package/dist/types/src/components/index.d.ts.map +1 -1
  113. package/dist/types/src/hooks/useSafeArea.d.ts.map +1 -1
  114. package/dist/types/src/hooks/useVisualViewport.d.ts +2 -2
  115. package/dist/types/src/hooks/useVisualViewport.d.ts.map +1 -1
  116. package/dist/types/src/index.d.ts +1 -1
  117. package/dist/types/src/index.d.ts.map +1 -1
  118. package/dist/types/src/playground/Controls.stories.d.ts +5 -9
  119. package/dist/types/src/playground/Controls.stories.d.ts.map +1 -1
  120. package/dist/types/src/playground/Custom.stories.d.ts +12 -4
  121. package/dist/types/src/playground/Custom.stories.d.ts.map +1 -1
  122. package/dist/types/src/playground/Typography.stories.d.ts +5 -11
  123. package/dist/types/src/playground/Typography.stories.d.ts.map +1 -1
  124. package/dist/types/src/testing/decorators/index.d.ts +2 -1
  125. package/dist/types/src/testing/decorators/index.d.ts.map +1 -1
  126. package/dist/types/src/testing/decorators/withLayout.d.ts +15 -0
  127. package/dist/types/src/testing/decorators/withLayout.d.ts.map +1 -0
  128. package/dist/types/src/testing/decorators/{withSurfaceVariantsLayout.d.ts → withLayoutVariants.d.ts} +2 -2
  129. package/dist/types/src/testing/decorators/withLayoutVariants.d.ts.map +1 -0
  130. package/dist/types/src/testing/decorators/withTheme.d.ts +3 -0
  131. package/dist/types/src/testing/decorators/withTheme.d.ts.map +1 -1
  132. package/dist/types/src/util/domino.d.ts +18 -0
  133. package/dist/types/src/util/domino.d.ts.map +1 -0
  134. package/dist/types/src/util/index.d.ts +2 -0
  135. package/dist/types/src/util/index.d.ts.map +1 -1
  136. package/dist/types/src/util/usePx.d.ts +8 -0
  137. package/dist/types/src/util/usePx.d.ts.map +1 -0
  138. package/dist/types/tsconfig.tsbuildinfo +1 -1
  139. package/package.json +24 -23
  140. package/src/components/Avatars/Avatar.stories.tsx +17 -9
  141. package/src/components/Avatars/AvatarGroup.stories.tsx +8 -5
  142. package/src/components/Breadcrumb/Breadcrumb.stories.tsx +16 -12
  143. package/src/components/{Buttons → Button}/Button.stories.tsx +5 -7
  144. package/src/components/{Buttons → Button}/IconButton.stories.tsx +9 -7
  145. package/src/components/{Buttons → Button}/IconButton.tsx +14 -13
  146. package/src/components/{Buttons → Button}/Toggle.stories.tsx +13 -10
  147. package/src/components/{Buttons → Button}/ToggleGroup.stories.tsx +8 -6
  148. package/src/components/{Buttons → Button}/ToggleGroup.tsx +14 -1
  149. package/src/components/Clipboard/CopyButton.tsx +1 -1
  150. package/src/components/{Dialogs → Dialog}/AlertDialog.stories.tsx +13 -11
  151. package/src/components/{Dialogs → Dialog}/Dialog.stories.tsx +15 -14
  152. package/src/components/Icon/Icon.stories.tsx +113 -0
  153. package/src/components/Icon/Icon.tsx +1 -1
  154. package/src/components/Input/Input.stories.tsx +8 -11
  155. package/src/components/Input/Input.tsx +3 -3
  156. package/src/components/Link/Link.stories.tsx +8 -5
  157. package/src/components/{Lists → List}/List.stories.tsx +13 -12
  158. package/src/components/{Lists → List}/List.tsx +5 -2
  159. package/src/components/{Lists → List}/ListDropIndicator.tsx +1 -1
  160. package/src/components/{Lists → List}/Tree.stories.tsx +9 -7
  161. package/src/components/{Lists → List}/Treegrid.stories.tsx +10 -5
  162. package/src/components/{Lists → List}/Treegrid.tsx +57 -16
  163. package/src/components/Main/Main.stories.tsx +15 -8
  164. package/src/components/Main/Main.tsx +36 -22
  165. package/src/components/Menus/ContextMenu.stories.tsx +9 -7
  166. package/src/components/Menus/ContextMenu.tsx +1 -0
  167. package/src/components/Menus/DropdownMenu.stories.tsx +10 -8
  168. package/src/components/Menus/DropdownMenu.tsx +37 -8
  169. package/src/components/Message/Message.stories.tsx +9 -6
  170. package/src/components/Popover/Popover.stories.tsx +10 -8
  171. package/src/components/Popover/Popover.tsx +22 -5
  172. package/src/components/ScrollArea/ScrollArea.stories.tsx +10 -8
  173. package/src/components/ScrollContainer/ScrollContainer.stories.tsx +69 -0
  174. package/src/components/ScrollContainer/ScrollContainer.tsx +231 -0
  175. package/src/components/ScrollContainer/index.ts +5 -0
  176. package/src/components/Select/Select.stories.tsx +14 -12
  177. package/src/components/Select/Select.tsx +9 -8
  178. package/src/components/Status/Status.stories.tsx +7 -5
  179. package/src/components/Tag/Tag.stories.tsx +16 -8
  180. package/src/components/Toast/Toast.stories.tsx +10 -8
  181. package/src/components/Toolbar/Toolbar.stories.tsx +12 -11
  182. package/src/components/Toolbar/Toolbar.tsx +18 -6
  183. package/src/components/Tooltip/Tooltip.stories.tsx +14 -11
  184. package/src/components/Tooltip/Tooltip.tsx +2 -1
  185. package/src/components/index.ts +4 -3
  186. package/src/hooks/useSafeArea.ts +3 -2
  187. package/src/hooks/useVisualViewport.ts +4 -4
  188. package/src/index.ts +1 -1
  189. package/src/playground/Controls.stories.tsx +12 -8
  190. package/src/playground/Custom.stories.tsx +11 -22
  191. package/src/playground/Typography.stories.tsx +8 -6
  192. package/src/testing/decorators/index.ts +2 -1
  193. package/src/testing/decorators/withLayout.tsx +56 -0
  194. package/src/testing/decorators/{withSurfaceVariantsLayout.tsx → withLayoutVariants.tsx} +2 -2
  195. package/src/testing/decorators/withTheme.tsx +31 -0
  196. package/src/util/domino.ts +53 -0
  197. package/src/util/index.ts +2 -0
  198. package/src/util/usePx.ts +61 -0
  199. package/dist/lib/browser/chunk-BKNGMGOK.mjs.map +0 -7
  200. package/dist/lib/node-esm/chunk-D2HZS6F4.mjs.map +0 -7
  201. package/dist/types/src/components/Buttons/Button.d.ts.map +0 -1
  202. package/dist/types/src/components/Buttons/Button.stories.d.ts +0 -12
  203. package/dist/types/src/components/Buttons/Button.stories.d.ts.map +0 -1
  204. package/dist/types/src/components/Buttons/IconButton.d.ts.map +0 -1
  205. package/dist/types/src/components/Buttons/IconButton.stories.d.ts +0 -22
  206. package/dist/types/src/components/Buttons/IconButton.stories.d.ts.map +0 -1
  207. package/dist/types/src/components/Buttons/Toggle.d.ts.map +0 -1
  208. package/dist/types/src/components/Buttons/Toggle.stories.d.ts +0 -19
  209. package/dist/types/src/components/Buttons/Toggle.stories.d.ts.map +0 -1
  210. package/dist/types/src/components/Buttons/ToggleGroup.d.ts.map +0 -1
  211. package/dist/types/src/components/Buttons/ToggleGroup.stories.d.ts +0 -39
  212. package/dist/types/src/components/Buttons/ToggleGroup.stories.d.ts.map +0 -1
  213. package/dist/types/src/components/Buttons/index.d.ts.map +0 -1
  214. package/dist/types/src/components/Dialogs/AlertDialog.d.ts.map +0 -1
  215. package/dist/types/src/components/Dialogs/AlertDialog.stories.d.ts +0 -43
  216. package/dist/types/src/components/Dialogs/AlertDialog.stories.d.ts.map +0 -1
  217. package/dist/types/src/components/Dialogs/Dialog.d.ts.map +0 -1
  218. package/dist/types/src/components/Dialogs/Dialog.stories.d.ts +0 -48
  219. package/dist/types/src/components/Dialogs/Dialog.stories.d.ts.map +0 -1
  220. package/dist/types/src/components/Dialogs/index.d.ts.map +0 -1
  221. package/dist/types/src/components/Lists/List.d.ts.map +0 -1
  222. package/dist/types/src/components/Lists/List.stories.d.ts +0 -37
  223. package/dist/types/src/components/Lists/List.stories.d.ts.map +0 -1
  224. package/dist/types/src/components/Lists/ListDropIndicator.d.ts.map +0 -1
  225. package/dist/types/src/components/Lists/Tree.d.ts.map +0 -1
  226. package/dist/types/src/components/Lists/Tree.stories.d.ts +0 -41
  227. package/dist/types/src/components/Lists/Tree.stories.d.ts.map +0 -1
  228. package/dist/types/src/components/Lists/TreeDropIndicator.d.ts.map +0 -1
  229. package/dist/types/src/components/Lists/Treegrid.stories.d.ts +0 -10
  230. package/dist/types/src/components/Lists/Treegrid.stories.d.ts.map +0 -1
  231. package/dist/types/src/components/Lists/index.d.ts.map +0 -1
  232. package/dist/types/src/testing/decorators/withSurfaceVariantsLayout.d.ts.map +0 -1
  233. package/src/testing/decorators/withTheme.ts +0 -25
  234. /package/dist/types/src/components/{Buttons → Button}/Toggle.d.ts +0 -0
  235. /package/dist/types/src/components/{Buttons → Button}/index.d.ts +0 -0
  236. /package/dist/types/src/components/{Dialogs → Dialog}/AlertDialog.d.ts +0 -0
  237. /package/dist/types/src/components/{Dialogs → Dialog}/Dialog.d.ts +0 -0
  238. /package/dist/types/src/components/{Dialogs → Dialog}/index.d.ts +0 -0
  239. /package/dist/types/src/components/{Lists → List}/List.d.ts +0 -0
  240. /package/dist/types/src/components/{Lists → List}/ListDropIndicator.d.ts +0 -0
  241. /package/dist/types/src/components/{Lists → List}/Tree.d.ts +0 -0
  242. /package/dist/types/src/components/{Lists → List}/TreeDropIndicator.d.ts +0 -0
  243. /package/dist/types/src/components/{Lists → List}/Treegrid.d.ts +0 -0
  244. /package/dist/types/src/components/{Lists → List}/index.d.ts +0 -0
  245. /package/src/components/{Buttons → Button}/Button.tsx +0 -0
  246. /package/src/components/{Buttons → Button}/Toggle.tsx +0 -0
  247. /package/src/components/{Buttons → Button}/index.ts +0 -0
  248. /package/src/components/{Dialogs → Dialog}/AlertDialog.tsx +0 -0
  249. /package/src/components/{Dialogs → Dialog}/Dialog.tsx +0 -0
  250. /package/src/components/{Dialogs → Dialog}/index.ts +0 -0
  251. /package/src/components/{Lists → List}/Tree.tsx +0 -0
  252. /package/src/components/{Lists → List}/TreeDropIndicator.tsx +0 -0
  253. /package/src/components/{Lists → List}/index.ts +0 -0
@@ -0,0 +1,231 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { useState } from '@preact-signals/safe-react/react';
6
+ import { createContext } from '@radix-ui/react-context';
7
+ import React, {
8
+ type HTMLAttributes,
9
+ type PropsWithChildren,
10
+ forwardRef,
11
+ useCallback,
12
+ useEffect,
13
+ useImperativeHandle,
14
+ useMemo,
15
+ useRef,
16
+ } from 'react';
17
+
18
+ // TODO(burdon): Move these deps to @dxos/dom-util.
19
+ import { addEventListener, combine } from '@dxos/async';
20
+ import { invariant } from '@dxos/invariant';
21
+ import { useForwardedRef } from '@dxos/react-hooks';
22
+ import { mx } from '@dxos/react-ui-theme';
23
+
24
+ import { type ThemedClassName } from '../../util';
25
+ import { IconButton } from '../Button';
26
+
27
+ const isBottom = (el: HTMLElement | null) => {
28
+ return !!(el && el.scrollHeight - el.scrollTop === el.clientHeight);
29
+ };
30
+
31
+ export interface ScrollController {
32
+ viewport: HTMLDivElement | null;
33
+ scrollToTop: (behavior?: ScrollBehavior) => void;
34
+ scrollToBottom: (behavior?: ScrollBehavior) => void;
35
+ }
36
+
37
+ type ScrollContainerContextValue = {
38
+ scrollToBottom: (behavior?: ScrollBehavior) => void;
39
+ controller?: ScrollController;
40
+ pinned?: boolean;
41
+ };
42
+
43
+ const [ScrollContainerProvider, useScrollContainerContext] =
44
+ createContext<ScrollContainerContextValue>('ScrollContainer');
45
+
46
+ //
47
+ // Root
48
+ //
49
+
50
+ type RootProps = ThemedClassName<
51
+ PropsWithChildren<{
52
+ pin?: boolean;
53
+ fade?: boolean;
54
+ }>
55
+ >;
56
+
57
+ /**
58
+ * Scroll container that automatically scrolls to the bottom when new content is added.
59
+ */
60
+ const Root = forwardRef<ScrollController, RootProps>(({ children, classNames, pin, fade }, forwardedRef) => {
61
+ const scrollerRef = useRef<HTMLDivElement>(null);
62
+ const autoScrollRef = useRef(false);
63
+ const [overflow, setOverflow] = useState(false);
64
+ const [pinned, setPinned] = useState(pin);
65
+
66
+ const timeoutRef = useRef<NodeJS.Timeout>(undefined);
67
+ const scrollToBottom = useCallback((behavior: ScrollBehavior = 'instant') => {
68
+ if (scrollerRef.current) {
69
+ // Temporarily hide scrollbar to prevent flicker.
70
+ autoScrollRef.current = true;
71
+ scrollerRef.current.classList.add('scrollbar-none');
72
+ scrollerRef.current.scrollTo({
73
+ top: scrollerRef.current.scrollHeight,
74
+ behavior,
75
+ });
76
+
77
+ clearTimeout(timeoutRef.current);
78
+ if (behavior !== 'instant') {
79
+ timeoutRef.current = setTimeout(() => {
80
+ scrollerRef.current?.classList.remove('scrollbar-none');
81
+ autoScrollRef.current = false;
82
+ }, 500);
83
+ }
84
+ setPinned(true);
85
+ }
86
+ }, []);
87
+
88
+ const controller = useMemo(
89
+ () => ({
90
+ viewport: scrollerRef.current,
91
+ scrollToTop: () => {
92
+ invariant(scrollerRef.current);
93
+ scrollerRef.current.scrollTo({ top: 0, behavior: 'smooth' });
94
+ setPinned(false);
95
+ },
96
+ scrollToBottom: () => {
97
+ scrollToBottom('smooth');
98
+ },
99
+ }),
100
+ [scrollToBottom, scrollerRef.current],
101
+ );
102
+
103
+ // Scroll controller imperative ref.
104
+ useImperativeHandle(forwardedRef, () => controller, [controller]);
105
+
106
+ // Listen for scroll events.
107
+ useEffect(() => {
108
+ if (!scrollerRef.current) {
109
+ return;
110
+ }
111
+
112
+ return combine(
113
+ // Check if user scrolls.
114
+ addEventListener(scrollerRef.current, 'wheel', () => {
115
+ setPinned(isBottom(scrollerRef.current));
116
+ }),
117
+ // Check if scrolls.
118
+ addEventListener(scrollerRef.current, 'scroll', () => {
119
+ setOverflow((scrollerRef.current?.scrollTop ?? 0) > 0);
120
+ }),
121
+ );
122
+ }, []);
123
+
124
+ return (
125
+ <ScrollContainerProvider pinned={pinned} controller={controller} scrollToBottom={scrollToBottom}>
126
+ <div className='relative grid flex-1 min-bs-0 overflow-hidden'>
127
+ {fade && (
128
+ <div
129
+ role='none'
130
+ data-visible={overflow}
131
+ className={mx(
132
+ // NOTE: Gradients may not be visible with dark reader extensions.
133
+ 'z-10 absolute block-start-0 inset-inline-0 bs-24 is-full',
134
+ 'opacity-0 duration-200 transition-opacity data-[visible="true"]:opacity-100',
135
+ 'bg-gradient-to-b from-[--surface-bg] to-transparent pointer-events-none',
136
+ )}
137
+ />
138
+ )}
139
+ <div className={mx('flex flex-col min-bs-0 overflow-y-auto scrollbar-thin', classNames)} ref={scrollerRef}>
140
+ {children}
141
+ </div>
142
+ </div>
143
+ </ScrollContainerProvider>
144
+ );
145
+ });
146
+
147
+ Root.displayName = 'ScrollContainer.Root';
148
+
149
+ //
150
+ // Viewport
151
+ //
152
+
153
+ type ViewportProps = ThemedClassName<PropsWithChildren<Omit<HTMLAttributes<HTMLDivElement>, 'className'>>>;
154
+
155
+ const Viewport = forwardRef<HTMLDivElement, ViewportProps>(({ classNames, children, ...props }, forwardedRef) => {
156
+ const contentRef = useForwardedRef(forwardedRef);
157
+ const { pinned, scrollToBottom } = useScrollContainerContext(Viewport.displayName!);
158
+
159
+ useEffect(() => {
160
+ if (!pinned || !contentRef.current) {
161
+ return;
162
+ }
163
+
164
+ // Setup resize observer to detect content changes.
165
+ const resizeObserver = new ResizeObserver(() => scrollToBottom());
166
+ scrollToBottom('instant');
167
+
168
+ resizeObserver.observe(contentRef.current);
169
+ return () => {
170
+ resizeObserver.disconnect();
171
+ };
172
+ }, [pinned, scrollToBottom]);
173
+
174
+ return (
175
+ <div className={mx('is-full', classNames)} {...props} ref={contentRef}>
176
+ {children}
177
+ </div>
178
+ );
179
+ });
180
+
181
+ Viewport.displayName = 'ScrollContainer.Viewport';
182
+
183
+ //
184
+ // ScrollDownButton
185
+ //
186
+
187
+ type ScrollDownButtonProps = ThemedClassName;
188
+
189
+ const ScrollDownButton = ({ classNames }: ScrollDownButtonProps) => {
190
+ const { pinned, scrollToBottom } = useScrollContainerContext(ScrollDownButton.displayName!);
191
+
192
+ return (
193
+ <div
194
+ role='none'
195
+ className={mx(
196
+ 'absolute bottom-2 right-4 opacity-100 transition-opacity duration-300',
197
+ pinned && 'opacity-0',
198
+ classNames,
199
+ )}
200
+ >
201
+ <IconButton
202
+ variant='primary'
203
+ icon='ph--arrow-down--regular'
204
+ iconOnly
205
+ size={4}
206
+ label='Scroll down'
207
+ onClick={() => scrollToBottom()}
208
+ />
209
+ </div>
210
+ );
211
+ };
212
+
213
+ ScrollDownButton.displayName = 'ScrollContainer.ScrollDownButton';
214
+
215
+ //
216
+ // ScrollContainer
217
+ //
218
+
219
+ export { useScrollContainerContext };
220
+
221
+ export const ScrollContainer = {
222
+ Root,
223
+ Viewport,
224
+ ScrollDownButton,
225
+ };
226
+
227
+ export type {
228
+ RootProps as ScrollContainerRootProps,
229
+ ViewportProps as ScrollContainerViewportProps,
230
+ ScrollDownButtonProps as ScrollContainerScrollDownButtonProps,
231
+ };
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ export * from './ScrollContainer';
@@ -2,14 +2,13 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
7
- import { type StoryObj } from '@storybook/react-vite';
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
8
6
  import React, { useState } from 'react';
9
7
 
10
8
  import { faker } from '@dxos/random';
11
9
 
12
- import { withSurfaceVariantsLayout, withTheme } from '../../testing';
10
+ import { withTheme } from '../../testing';
11
+ import { withLayoutVariants } from '../../testing';
13
12
 
14
13
  import { Select } from './Select';
15
14
 
@@ -42,15 +41,18 @@ const DefaultStory = ({ items = [] }: StoryProps) => {
42
41
  );
43
42
  };
44
43
 
45
- export const Default: StoryObj<StoryProps> = {
44
+ const meta = {
45
+ title: 'ui/react-ui-core/Select',
46
+ render: DefaultStory,
47
+ decorators: [withTheme, withLayoutVariants()],
48
+ } satisfies Meta<typeof DefaultStory>;
49
+
50
+ export default meta;
51
+
52
+ type Story = StoryObj<typeof meta>;
53
+
54
+ export const Default: Story = {
46
55
  args: {
47
56
  items: Array.from({ length: 16 }).map((_, i) => ({ id: `item-${i}`, text: faker.lorem.word() })),
48
57
  },
49
58
  };
50
-
51
- export default {
52
- title: 'ui/react-ui-core/Select',
53
- render: DefaultStory,
54
- decorators: [withSurfaceVariantsLayout(), withTheme],
55
- parameters: { chromatic: { disableSnapshot: false } },
56
- };
@@ -8,7 +8,7 @@ import React, { forwardRef } from 'react';
8
8
  import { useElevationContext, useThemeContext } from '../../hooks';
9
9
  import { useSafeCollisionPadding } from '../../hooks/useSafeCollisionPadding';
10
10
  import { type ThemedClassName } from '../../util';
11
- import { Button, type ButtonProps } from '../Buttons';
11
+ import { Button, type ButtonProps } from '../Button';
12
12
  import { Icon } from '../Icon';
13
13
 
14
14
  type SelectRootProps = SelectPrimitive.SelectProps;
@@ -35,14 +35,13 @@ type SelectTriggerButtonProps = Omit<ButtonProps, 'children'> & Pick<SelectValue
35
35
 
36
36
  const SelectTriggerButton = forwardRef<HTMLButtonElement, SelectTriggerButtonProps>(
37
37
  ({ children, placeholder, ...props }, forwardedRef) => {
38
- const { tx } = useThemeContext();
39
38
  return (
40
39
  <SelectPrimitive.Trigger asChild ref={forwardedRef}>
41
40
  <Button {...props}>
42
41
  <SelectPrimitive.Value placeholder={placeholder}>{children}</SelectPrimitive.Value>
43
- <span className='w-1 flex-1' />
42
+ <span className='is-1 flex-1' />
44
43
  <SelectPrimitive.Icon asChild>
45
- <Icon icon='ph--caret-down--bold' classNames={tx('select.triggerIcon', 'select__trigger__icon', {})} />
44
+ <Icon size={3} icon='ph--caret-down--bold' />
46
45
  </SelectPrimitive.Icon>
47
46
  </Button>
48
47
  </SelectPrimitive.Trigger>
@@ -60,6 +59,7 @@ const SelectContent = forwardRef<HTMLDivElement, SelectContentProps>(
60
59
  return (
61
60
  <SelectPrimitive.Content
62
61
  {...props}
62
+ data-arrow-keys='up down'
63
63
  collisionPadding={safeCollisionPadding}
64
64
  className={tx('select.content', 'select__content', { elevation }, classNames)}
65
65
  position='popper'
@@ -82,7 +82,7 @@ const SelectScrollUpButton = forwardRef<HTMLDivElement, SelectScrollUpButtonProp
82
82
  className={tx('select.scrollButton', 'select__scroll-button--up', {}, classNames)}
83
83
  ref={forwardedRef}
84
84
  >
85
- {children ?? <Icon icon='ph--caret-up--bold' />}
85
+ {children ?? <Icon size={3} icon='ph--caret-up--bold' />}
86
86
  </SelectPrimitive.SelectScrollUpButton>
87
87
  );
88
88
  },
@@ -99,7 +99,7 @@ const SelectScrollDownButton = forwardRef<HTMLDivElement, SelectScrollDownButton
99
99
  className={tx('select.scrollButton', 'select__scroll-button--down', {}, classNames)}
100
100
  ref={forwardedRef}
101
101
  >
102
- {children ?? <Icon icon='ph--caret-down--bold' />}
102
+ {children ?? <Icon size={3} icon='ph--caret-down--bold' />}
103
103
  </SelectPrimitive.SelectScrollDownButton>
104
104
  );
105
105
  },
@@ -108,7 +108,7 @@ const SelectScrollDownButton = forwardRef<HTMLDivElement, SelectScrollDownButton
108
108
  type SelectViewportProps = ThemedClassName<SelectPrimitive.SelectViewportProps>;
109
109
 
110
110
  const SelectViewport = forwardRef<HTMLDivElement, SelectViewportProps>(
111
- ({ classNames, asChild, children, ...props }, forwardedRef) => {
111
+ ({ classNames, children, ...props }, forwardedRef) => {
112
112
  const { tx } = useThemeContext();
113
113
  return (
114
114
  <SelectPrimitive.SelectViewport
@@ -152,12 +152,13 @@ const SelectItemIndicator = forwardRef<HTMLDivElement, SelectItemIndicatorProps>
152
152
 
153
153
  type SelectOptionProps = SelectItemProps;
154
154
 
155
+ // TODO(burdon): Option to show icon on left/right.
155
156
  const SelectOption = forwardRef<HTMLDivElement, SelectItemProps>(({ children, classNames, ...props }, forwardedRef) => {
156
157
  const { tx } = useThemeContext();
157
158
  return (
158
159
  <SelectPrimitive.Item {...props} className={tx('select.item', 'option', {}, classNames)} ref={forwardedRef}>
159
160
  <SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
160
- <span className='grow w-1' />
161
+ <span className='grow is-1' />
161
162
  {/* <SelectPrimitive.ItemIndicator className={tx('select.itemIndicator', 'option__indicator', {})}> */}
162
163
  <Icon icon='ph--check--regular' />
163
164
  {/* </SelectPrimitive.ItemIndicator> */}
@@ -2,20 +2,22 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
7
6
  import React from 'react';
8
7
 
9
8
  import { withTheme } from '../../testing';
10
9
 
11
10
  import { Status } from './Status';
12
11
 
13
- export default {
12
+ const meta = {
14
13
  title: 'ui/react-ui-core/Status',
15
14
  component: Status,
16
15
  decorators: [withTheme],
17
- parameters: { chromatic: { disableSnapshot: false } },
18
- };
16
+ } satisfies Meta<typeof Status>;
17
+
18
+ export default meta;
19
+
20
+ type Story = StoryObj<typeof meta>;
19
21
 
20
22
  export const Normal = (props: any) => {
21
23
  return (
@@ -1,10 +1,11 @@
1
1
  //
2
2
  // Copyright 2022 DXOS.org
3
3
  //
4
+
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
4
6
  import React from 'react';
5
7
 
6
8
  import { hues } from '@dxos/react-ui-theme';
7
- import '@dxos-theme';
8
9
  import { type ChromaticPalette, type MessageValence } from '@dxos/react-ui-types';
9
10
 
10
11
  import { withTheme } from '../../testing';
@@ -13,14 +14,9 @@ import { Tag } from './Tag';
13
14
 
14
15
  const palettes = ['neutral', 'success', 'info', 'warning', 'error', ...hues] as (ChromaticPalette | MessageValence)[];
15
16
 
16
- export default {
17
+ const meta = {
17
18
  title: 'ui/react-ui-core/Tag',
18
19
  component: Tag,
19
- decorators: [withTheme],
20
- parameters: { chromatic: { disableSnapshot: false } },
21
- } as const;
22
-
23
- export const Default = {
24
20
  render: () => (
25
21
  <div role='grid' className='grid grid-cols-5 gap-2 max-is-screen-md'>
26
22
  {palettes.map((palette) => (
@@ -30,4 +26,16 @@ export const Default = {
30
26
  ))}
31
27
  </div>
32
28
  ),
33
- };
29
+ decorators: [withTheme],
30
+ parameters: {
31
+ chromatic: {
32
+ disableSnapshot: false,
33
+ },
34
+ },
35
+ } satisfies Meta<typeof Tag>;
36
+
37
+ export default meta;
38
+
39
+ type Story = StoryObj<typeof meta>;
40
+
41
+ export const Default: Story = {};
@@ -2,12 +2,11 @@
2
2
  // Copyright 2022 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
7
6
  import React, { type ReactNode, useState } from 'react';
8
7
 
9
8
  import { withTheme } from '../../testing';
10
- import { Button } from '../Buttons';
9
+ import { Button } from '../Button';
11
10
 
12
11
  import { Toast } from './Toast';
13
12
 
@@ -45,15 +44,18 @@ const DefaultStory = ({ title, description, actionTriggers, openTrigger, closeTr
45
44
  );
46
45
  };
47
46
 
48
- export default {
47
+ const meta = {
49
48
  title: 'ui/react-ui-core/Toast',
50
- component: Toast,
49
+ component: Toast as any,
51
50
  render: DefaultStory,
52
51
  decorators: [withTheme],
53
- parameters: { chromatic: { disableSnapshot: false } },
54
- };
52
+ } satisfies Meta<typeof DefaultStory>;
53
+
54
+ export default meta;
55
+
56
+ type Story = StoryObj<typeof meta>;
55
57
 
56
- export const Default = {
58
+ export const Default: Story = {
57
59
  args: {
58
60
  openTrigger: 'Open toast',
59
61
  title: 'This is a toast',
@@ -2,12 +2,11 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
6
-
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
7
6
  import React from 'react';
8
7
 
9
8
  import { withTheme } from '../../testing';
10
- import { Toggle } from '../Buttons';
9
+ import { Toggle } from '../Button';
11
10
  import { Icon } from '../Icon';
12
11
  import { Select } from '../Select';
13
12
 
@@ -30,6 +29,7 @@ const DefaultStory = (props: StorybookToolbarProps) => {
30
29
  <Select.Option value={'b'}>B</Select.Option>
31
30
  <Select.Option value={'c'}>C</Select.Option>
32
31
  </Select.Viewport>
32
+ <Select.Arrow />
33
33
  </Select.Content>
34
34
  </Select.Portal>
35
35
  </Select.Root>
@@ -63,21 +63,22 @@ const DefaultStory = (props: StorybookToolbarProps) => {
63
63
  </Toolbar.Button>
64
64
  <Toolbar.Separator />
65
65
  <Toolbar.Button>Test</Toolbar.Button>
66
- <Toolbar.Button>
67
- <Icon icon='ph--arrow-clockwise--regular' />
68
- </Toolbar.Button>
66
+ <Toolbar.IconButton icon='ph--arrow-clockwise--regular' label='Refresh' iconOnly />
69
67
  </Toolbar.Root>
70
68
  );
71
69
  };
72
70
 
73
- export default {
71
+ const meta = {
74
72
  title: 'ui/react-ui-core/Toolbar',
75
- component: Toolbar,
73
+ component: Toolbar as any,
76
74
  render: DefaultStory,
77
75
  decorators: [withTheme],
78
- parameters: { chromatic: { disableSnapshot: false } },
79
- };
76
+ } satisfies Meta<typeof DefaultStory>;
77
+
78
+ export default meta;
79
+
80
+ type Story = StoryObj<typeof meta>;
80
81
 
81
- export const Default = {
82
+ export const Default: Story = {
82
83
  args: {},
83
84
  };
@@ -4,7 +4,7 @@
4
4
 
5
5
  import type { ToggleGroupItemProps as ToggleGroupItemPrimitiveProps } from '@radix-ui/react-toggle-group';
6
6
  import * as ToolbarPrimitive from '@radix-ui/react-toolbar';
7
- import React, { forwardRef } from 'react';
7
+ import React, { Fragment, forwardRef } from 'react';
8
8
 
9
9
  import { useThemeContext } from '../../hooks';
10
10
  import { type ThemedClassName } from '../../util';
@@ -18,22 +18,34 @@ import {
18
18
  Toggle,
19
19
  type ToggleGroupItemProps,
20
20
  type ToggleProps,
21
- } from '../Buttons';
21
+ } from '../Button';
22
22
  import { Link, type LinkProps } from '../Link';
23
23
  import { Separator, type SeparatorProps } from '../Separator';
24
24
 
25
- type ToolbarRootProps = ThemedClassName<ToolbarPrimitive.ToolbarProps> & { layoutManaged?: boolean };
25
+ type ToolbarRootProps = ThemedClassName<
26
+ ToolbarPrimitive.ToolbarProps & {
27
+ textBlockWidth?: boolean;
28
+ layoutManaged?: boolean;
29
+ disabled?: boolean;
30
+ }
31
+ >;
26
32
 
27
33
  const ToolbarRoot = forwardRef<HTMLDivElement, ToolbarRootProps>(
28
- ({ classNames, children, layoutManaged, ...props }, forwardedRef) => {
34
+ ({ classNames, children, layoutManaged, textBlockWidth: textBlockWidthParam, disabled, ...props }, forwardedRef) => {
29
35
  const { tx } = useThemeContext();
36
+ const InnerRoot = textBlockWidthParam ? 'div' : Fragment;
37
+ const innerRootProps = textBlockWidthParam
38
+ ? { role: 'none', className: tx('toolbar.inner', 'toolbar', { layoutManaged }, classNames) }
39
+ : {};
40
+
30
41
  return (
31
42
  <ToolbarPrimitive.Root
32
43
  {...props}
33
- className={tx('toolbar.root', 'toolbar', { layoutManaged }, classNames)}
44
+ data-arrow-keys={props.orientation === 'vertical' ? 'up down' : 'left right'}
45
+ className={tx('toolbar.root', 'toolbar', { layoutManaged, disabled }, classNames)}
34
46
  ref={forwardedRef}
35
47
  >
36
- {children}
48
+ <InnerRoot {...innerRootProps}>{children}</InnerRoot>
37
49
  </ToolbarPrimitive.Root>
38
50
  );
39
51
  },
@@ -2,13 +2,13 @@
2
2
  // Copyright 2022 DXOS.org
3
3
  //
4
4
 
5
- import '@dxos-theme';
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
6
6
  import React from 'react';
7
7
 
8
8
  import { faker } from '@dxos/random';
9
9
 
10
10
  import { withTheme } from '../../testing';
11
- import { Button } from '../Buttons';
11
+ import { Button } from '../Button';
12
12
 
13
13
  import { Tooltip } from './Tooltip';
14
14
 
@@ -29,15 +29,18 @@ const DefaultStory = ({ tooltips, defaultOpen }: StoryProps) => (
29
29
  </Tooltip.Provider>
30
30
  );
31
31
 
32
- export default {
32
+ const meta = {
33
33
  title: 'ui/react-ui-core/Tooltip',
34
- component: Tooltip,
34
+ component: Tooltip as any,
35
35
  render: DefaultStory,
36
36
  decorators: [withTheme],
37
- parameters: { chromatic: { disableSnapshot: false } },
38
- };
37
+ } satisfies Meta<typeof DefaultStory>;
38
+
39
+ export default meta;
40
+
41
+ type Story = StoryObj<typeof meta>;
39
42
 
40
- export const Default = {
43
+ export const Default: Story = {
41
44
  args: {
42
45
  tooltips: [
43
46
  {
@@ -51,9 +54,9 @@ export const Default = {
51
54
  },
52
55
  };
53
56
 
54
- export const DefaultOpen = {
57
+ export const DefaultOpen: Story = {
55
58
  args: {
56
- defaultOption: true,
59
+ defaultOpen: true,
57
60
  tooltips: [
58
61
  {
59
62
  label: 'Tooltip trigger',
@@ -66,9 +69,9 @@ export const DefaultOpen = {
66
69
  },
67
70
  };
68
71
 
69
- export const StressTest = {
72
+ export const StressTest: Story = {
70
73
  args: {
71
- defaultOption: true,
74
+ defaultOpen: true,
72
75
  tooltips: faker.helpers.multiple(
73
76
  () => ({
74
77
  label: faker.lorem.words(2),
@@ -25,6 +25,7 @@ import React, {
25
25
  type FC,
26
26
  type MutableRefObject,
27
27
  type ReactNode,
28
+ type RefObject,
28
29
  type SyntheticEvent,
29
30
  forwardRef,
30
31
  useCallback,
@@ -218,7 +219,7 @@ const TooltipProvider: FC<TooltipProviderProps> = (props: TooltipScopedProps<Too
218
219
  {content}
219
220
  <TooltipArrow className={tx('tooltip.arrow', 'tooltip__arrow')} />
220
221
  </TooltipContent>
221
- <TooltipVirtualTrigger virtualRef={triggerRef} />
222
+ <TooltipVirtualTrigger virtualRef={triggerRef as RefObject<HTMLButtonElement>} />
222
223
  {children}
223
224
  </TooltipContextProvider>
224
225
  </PopperPrimitive.Root>
@@ -5,19 +5,20 @@
5
5
  export * from './AnchoredOverflow';
6
6
  export * from './Avatars';
7
7
  export * from './Breadcrumb';
8
- export * from './Buttons';
8
+ export * from './Button';
9
9
  export * from './Clipboard';
10
- export * from './Dialogs';
10
+ export * from './Dialog';
11
11
  export * from './Icon';
12
12
  export * from './Input';
13
13
  export * from './Link';
14
- export * from './Lists';
14
+ export * from './List';
15
15
  export * from './Main';
16
16
  export * from './Menus';
17
17
  export * from './Message';
18
18
  export * from './Popover';
19
19
  export * from './Status';
20
20
  export * from './ScrollArea';
21
+ export * from './ScrollContainer';
21
22
  export * from './Select';
22
23
  export * from './Separator';
23
24
  export * from './Tag';