@dxos/react-ui 0.8.4-main.69d29f4 → 0.8.4-main.6fa680abb7

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 (271) hide show
  1. package/dist/lib/browser/{chunk-CEKVHJ27.mjs → chunk-2FKSMWNY.mjs} +117 -117
  2. package/dist/lib/browser/chunk-2FKSMWNY.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +2804 -1957
  4. package/dist/lib/browser/index.mjs.map +4 -4
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/testing/index.mjs +56 -32
  7. package/dist/lib/browser/testing/index.mjs.map +4 -4
  8. package/dist/lib/node-esm/{chunk-2NHEX4AD.mjs → chunk-ZNBLTSHI.mjs} +117 -117
  9. package/dist/lib/node-esm/chunk-ZNBLTSHI.mjs.map +7 -0
  10. package/dist/lib/node-esm/index.mjs +2804 -1957
  11. package/dist/lib/node-esm/index.mjs.map +4 -4
  12. package/dist/lib/node-esm/meta.json +1 -1
  13. package/dist/lib/node-esm/testing/index.mjs +56 -32
  14. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  15. package/dist/types/src/components/Avatars/Avatar.d.ts.map +1 -1
  16. package/dist/types/src/components/Avatars/Avatar.stories.d.ts.map +1 -1
  17. package/dist/types/src/components/Breadcrumb/Breadcrumb.d.ts.map +1 -1
  18. package/dist/types/src/components/Button/Button.d.ts.map +1 -1
  19. package/dist/types/src/components/Card/Card.d.ts +107 -0
  20. package/dist/types/src/components/Card/Card.d.ts.map +1 -0
  21. package/dist/types/src/components/Card/Card.stories.d.ts +21 -0
  22. package/dist/types/src/components/Card/Card.stories.d.ts.map +1 -0
  23. package/dist/types/src/components/Card/index.d.ts +2 -0
  24. package/dist/types/src/components/Card/index.d.ts.map +1 -0
  25. package/dist/types/src/components/Dialog/AlertDialog.d.ts +12 -3
  26. package/dist/types/src/components/Dialog/AlertDialog.d.ts.map +1 -1
  27. package/dist/types/src/components/Dialog/AlertDialog.stories.d.ts.map +1 -1
  28. package/dist/types/src/components/Dialog/Dialog.d.ts +11 -4
  29. package/dist/types/src/components/Dialog/Dialog.d.ts.map +1 -1
  30. package/dist/types/src/components/Dialog/Dialog.stories.d.ts +3 -6
  31. package/dist/types/src/components/Dialog/Dialog.stories.d.ts.map +1 -1
  32. package/dist/types/src/components/ErrorFallback/ErrorFallback.d.ts +11 -0
  33. package/dist/types/src/components/ErrorFallback/ErrorFallback.d.ts.map +1 -0
  34. package/dist/types/src/components/ErrorFallback/ErrorFallback.stories.d.ts +7 -0
  35. package/dist/types/src/components/ErrorFallback/ErrorFallback.stories.d.ts.map +1 -0
  36. package/dist/types/src/components/ErrorFallback/ErrorStack.d.ts +8 -0
  37. package/dist/types/src/components/ErrorFallback/ErrorStack.d.ts.map +1 -0
  38. package/dist/types/src/components/ErrorFallback/ThrowError.d.ts +9 -0
  39. package/dist/types/src/components/ErrorFallback/ThrowError.d.ts.map +1 -0
  40. package/dist/types/src/components/ErrorFallback/index.d.ts +5 -0
  41. package/dist/types/src/components/ErrorFallback/index.d.ts.map +1 -0
  42. package/dist/types/src/components/Image/Image.d.ts +14 -0
  43. package/dist/types/src/components/Image/Image.d.ts.map +1 -0
  44. package/dist/types/src/components/Image/Image.stories.d.ts +33 -0
  45. package/dist/types/src/components/Image/Image.stories.d.ts.map +1 -0
  46. package/dist/types/src/components/Image/index.d.ts +2 -0
  47. package/dist/types/src/components/Image/index.d.ts.map +1 -0
  48. package/dist/types/src/components/Input/Input.d.ts +2 -5
  49. package/dist/types/src/components/Input/Input.d.ts.map +1 -1
  50. package/dist/types/src/components/Input/Input.stories.d.ts +7 -7
  51. package/dist/types/src/components/Input/Input.stories.d.ts.map +1 -1
  52. package/dist/types/src/components/List/List.d.ts.map +1 -1
  53. package/dist/types/src/components/List/List.stories.d.ts.map +1 -1
  54. package/dist/types/src/components/List/Treegrid.d.ts.map +1 -1
  55. package/dist/types/src/components/Main/Main.d.ts +9 -10
  56. package/dist/types/src/components/Main/Main.d.ts.map +1 -1
  57. package/dist/types/src/components/Main/Main.stories.d.ts +0 -3
  58. package/dist/types/src/components/Main/Main.stories.d.ts.map +1 -1
  59. package/dist/types/src/components/Menu/ContextMenu.d.ts.map +1 -1
  60. package/dist/types/src/components/Message/Message.d.ts.map +1 -1
  61. package/dist/types/src/components/Message/Message.stories.d.ts +2 -3
  62. package/dist/types/src/components/Message/Message.stories.d.ts.map +1 -1
  63. package/dist/types/src/components/Popover/Popover.d.ts +1 -0
  64. package/dist/types/src/components/Popover/Popover.d.ts.map +1 -1
  65. package/dist/types/src/components/ScrollArea/ScrollArea.d.ts +21 -26
  66. package/dist/types/src/components/ScrollArea/ScrollArea.d.ts.map +1 -1
  67. package/dist/types/src/components/ScrollArea/ScrollArea.stories.d.ts +41 -9
  68. package/dist/types/src/components/ScrollArea/ScrollArea.stories.d.ts.map +1 -1
  69. package/dist/types/src/components/ScrollContainer/ScrollContainer.d.ts.map +1 -1
  70. package/dist/types/src/components/ScrollContainer/ScrollContainer.stories.d.ts +6 -1
  71. package/dist/types/src/components/ScrollContainer/ScrollContainer.stories.d.ts.map +1 -1
  72. package/dist/types/src/components/Select/Select.d.ts.map +1 -1
  73. package/dist/types/src/components/Skeleton/Skeleton.d.ts +12 -0
  74. package/dist/types/src/components/Skeleton/Skeleton.d.ts.map +1 -0
  75. package/dist/types/src/components/Skeleton/Skeleton.stories.d.ts +17 -0
  76. package/dist/types/src/components/Skeleton/Skeleton.stories.d.ts.map +1 -0
  77. package/dist/types/src/components/Skeleton/index.d.ts +2 -0
  78. package/dist/types/src/components/Skeleton/index.d.ts.map +1 -0
  79. package/dist/types/src/components/Splitter/Splitter.d.ts +32 -0
  80. package/dist/types/src/components/Splitter/Splitter.d.ts.map +1 -0
  81. package/dist/types/src/components/Splitter/Splitter.stories.d.ts +7 -0
  82. package/dist/types/src/components/Splitter/Splitter.stories.d.ts.map +1 -0
  83. package/dist/types/src/components/Splitter/index.d.ts +2 -0
  84. package/dist/types/src/components/Splitter/index.d.ts.map +1 -0
  85. package/dist/types/src/components/Status/Status.stories.d.ts +4 -2
  86. package/dist/types/src/components/Status/Status.stories.d.ts.map +1 -1
  87. package/dist/types/src/components/Tag/Tag.d.ts.map +1 -1
  88. package/dist/types/src/components/Tag/Tag.stories.d.ts +0 -5
  89. package/dist/types/src/components/Tag/Tag.stories.d.ts.map +1 -1
  90. package/dist/types/src/components/ThemeProvider/ThemeProvider.d.ts +1 -0
  91. package/dist/types/src/components/ThemeProvider/ThemeProvider.d.ts.map +1 -1
  92. package/dist/types/src/components/ThemeProvider/ThemeProvider.stories.d.ts +12 -0
  93. package/dist/types/src/components/ThemeProvider/ThemeProvider.stories.d.ts.map +1 -0
  94. package/dist/types/src/components/Toast/Toast.d.ts.map +1 -1
  95. package/dist/types/src/components/Toolbar/Toolbar.d.ts +33 -7
  96. package/dist/types/src/components/Toolbar/Toolbar.d.ts.map +1 -1
  97. package/dist/types/src/components/Tooltip/Tooltip.d.ts.map +1 -1
  98. package/dist/types/src/components/Tooltip/Tooltip.stories.d.ts.map +1 -1
  99. package/dist/types/src/components/index.d.ts +8 -4
  100. package/dist/types/src/components/index.d.ts.map +1 -1
  101. package/dist/types/src/exemplars/generics.stories.d.ts +18 -0
  102. package/dist/types/src/exemplars/generics.stories.d.ts.map +1 -0
  103. package/dist/types/src/exemplars/slot.stories.d.ts +14 -0
  104. package/dist/types/src/exemplars/slot.stories.d.ts.map +1 -0
  105. package/dist/types/src/exemplars/tabster.stories.d.ts +8 -0
  106. package/dist/types/src/exemplars/tabster.stories.d.ts.map +1 -0
  107. package/dist/types/src/exemplars/virtualizer.stories.d.ts +11 -0
  108. package/dist/types/src/exemplars/virtualizer.stories.d.ts.map +1 -0
  109. package/dist/types/src/index.d.ts +1 -0
  110. package/dist/types/src/index.d.ts.map +1 -1
  111. package/dist/types/src/playground/Controls.stories.d.ts.map +1 -1
  112. package/dist/types/src/primitives/Column/Column.d.ts +26 -0
  113. package/dist/types/src/primitives/Column/Column.d.ts.map +1 -0
  114. package/dist/types/src/primitives/Column/Column.stories.d.ts +6 -0
  115. package/dist/types/src/primitives/Column/Column.stories.d.ts.map +1 -0
  116. package/dist/types/src/primitives/Column/index.d.ts +2 -0
  117. package/dist/types/src/primitives/Column/index.d.ts.map +1 -0
  118. package/dist/types/src/primitives/Container/Container.d.ts +8 -0
  119. package/dist/types/src/primitives/Container/Container.d.ts.map +1 -0
  120. package/dist/types/src/primitives/Container/Container.stories.d.ts +6 -0
  121. package/dist/types/src/primitives/Container/Container.stories.d.ts.map +1 -0
  122. package/dist/types/src/primitives/Container/index.d.ts +2 -0
  123. package/dist/types/src/primitives/Container/index.d.ts.map +1 -0
  124. package/dist/types/src/primitives/Flex/Flex.d.ts +13 -0
  125. package/dist/types/src/primitives/Flex/Flex.d.ts.map +1 -0
  126. package/dist/types/src/primitives/Flex/Flex.stories.d.ts +8 -0
  127. package/dist/types/src/primitives/Flex/Flex.stories.d.ts.map +1 -0
  128. package/dist/types/src/primitives/Flex/index.d.ts +2 -0
  129. package/dist/types/src/primitives/Flex/index.d.ts.map +1 -0
  130. package/dist/types/src/primitives/Grid/Grid.d.ts +15 -0
  131. package/dist/types/src/primitives/Grid/Grid.d.ts.map +1 -0
  132. package/dist/types/src/primitives/Grid/Grid.stories.d.ts +8 -0
  133. package/dist/types/src/primitives/Grid/Grid.stories.d.ts.map +1 -0
  134. package/dist/types/src/primitives/Grid/index.d.ts +2 -0
  135. package/dist/types/src/primitives/Grid/index.d.ts.map +1 -0
  136. package/dist/types/src/primitives/Panel/Panel.d.ts +26 -0
  137. package/dist/types/src/primitives/Panel/Panel.d.ts.map +1 -0
  138. package/dist/types/src/primitives/Panel/Panel.stories.d.ts +6 -0
  139. package/dist/types/src/primitives/Panel/Panel.stories.d.ts.map +1 -0
  140. package/dist/types/src/primitives/Panel/index.d.ts +2 -0
  141. package/dist/types/src/primitives/Panel/index.d.ts.map +1 -0
  142. package/dist/types/src/primitives/index.d.ts +6 -0
  143. package/dist/types/src/primitives/index.d.ts.map +1 -0
  144. package/dist/types/src/testing/Loading.d.ts +9 -0
  145. package/dist/types/src/testing/Loading.d.ts.map +1 -0
  146. package/dist/types/src/testing/decorators/withLayout.d.ts +1 -1
  147. package/dist/types/src/testing/decorators/withLayout.d.ts.map +1 -1
  148. package/dist/types/src/testing/decorators/withLayoutVariants.d.ts.map +1 -1
  149. package/dist/types/src/testing/decorators/withTheme.d.ts +3 -2
  150. package/dist/types/src/testing/decorators/withTheme.d.ts.map +1 -1
  151. package/dist/types/src/testing/index.d.ts +1 -0
  152. package/dist/types/src/testing/index.d.ts.map +1 -1
  153. package/dist/types/src/translations.d.ts +11 -0
  154. package/dist/types/src/translations.d.ts.map +1 -0
  155. package/dist/types/tsconfig.tsbuildinfo +1 -1
  156. package/package.json +25 -21
  157. package/src/components/Avatars/Avatar.stories.tsx +5 -6
  158. package/src/components/Avatars/Avatar.tsx +5 -12
  159. package/src/components/Avatars/AvatarGroup.stories.tsx +2 -2
  160. package/src/components/Breadcrumb/Breadcrumb.stories.tsx +3 -3
  161. package/src/components/Breadcrumb/Breadcrumb.tsx +11 -37
  162. package/src/components/Button/Button.stories.tsx +3 -3
  163. package/src/components/Button/Button.tsx +6 -12
  164. package/src/components/Button/IconButton.stories.tsx +4 -4
  165. package/src/components/Button/IconButton.tsx +1 -1
  166. package/src/components/Button/Toggle.stories.tsx +2 -2
  167. package/src/components/Button/ToggleGroup.stories.tsx +2 -2
  168. package/src/components/Card/Card.stories.tsx +151 -0
  169. package/src/components/Card/Card.tsx +347 -0
  170. package/src/components/Card/index.ts +5 -0
  171. package/src/components/Clipboard/CopyButton.tsx +3 -3
  172. package/src/components/Dialog/AlertDialog.stories.tsx +15 -15
  173. package/src/components/Dialog/AlertDialog.tsx +116 -16
  174. package/src/components/Dialog/Dialog.stories.tsx +40 -15
  175. package/src/components/Dialog/Dialog.tsx +75 -45
  176. package/src/components/ErrorFallback/ErrorFallback.stories.tsx +50 -0
  177. package/src/components/ErrorFallback/ErrorFallback.tsx +70 -0
  178. package/src/components/ErrorFallback/ErrorStack.tsx +80 -0
  179. package/src/components/ErrorFallback/ThrowError.tsx +37 -0
  180. package/src/components/ErrorFallback/index.ts +9 -0
  181. package/src/components/Icon/Icon.stories.tsx +2 -2
  182. package/src/components/Icon/Icon.tsx +1 -1
  183. package/src/components/Image/Image.stories.tsx +86 -0
  184. package/src/components/Image/Image.tsx +223 -0
  185. package/src/components/Image/index.ts +5 -0
  186. package/src/components/Input/Input.stories.tsx +20 -39
  187. package/src/components/Input/Input.tsx +20 -65
  188. package/src/components/Link/Link.stories.tsx +2 -2
  189. package/src/components/Link/Link.tsx +2 -2
  190. package/src/components/List/List.stories.tsx +15 -22
  191. package/src/components/List/List.tsx +11 -16
  192. package/src/components/List/ListDropIndicator.tsx +7 -7
  193. package/src/components/List/Tree.stories.tsx +4 -4
  194. package/src/components/List/TreeDropIndicator.tsx +6 -6
  195. package/src/components/List/Treegrid.stories.tsx +3 -3
  196. package/src/components/List/Treegrid.tsx +10 -15
  197. package/src/components/Main/Main.stories.tsx +41 -23
  198. package/src/components/Main/Main.tsx +138 -81
  199. package/src/components/Menu/ContextMenu.stories.tsx +2 -2
  200. package/src/components/Menu/ContextMenu.tsx +9 -33
  201. package/src/components/Menu/DropdownMenu.stories.tsx +2 -2
  202. package/src/components/Menu/DropdownMenu.tsx +10 -10
  203. package/src/components/Message/Message.stories.tsx +25 -10
  204. package/src/components/Message/Message.tsx +17 -29
  205. package/src/components/Popover/Popover.stories.tsx +4 -4
  206. package/src/components/Popover/Popover.tsx +23 -20
  207. package/src/components/ScrollArea/ScrollArea.stories.tsx +152 -76
  208. package/src/components/ScrollArea/ScrollArea.tsx +72 -116
  209. package/src/components/ScrollArea/index.ts +1 -1
  210. package/src/components/ScrollContainer/ScrollContainer.stories.tsx +41 -22
  211. package/src/components/ScrollContainer/ScrollContainer.tsx +18 -13
  212. package/src/components/Select/Select.stories.tsx +2 -2
  213. package/src/components/Select/Select.tsx +11 -27
  214. package/src/components/Separator/Separator.tsx +1 -1
  215. package/src/components/Skeleton/Skeleton.stories.tsx +52 -0
  216. package/src/components/Skeleton/Skeleton.tsx +26 -0
  217. package/src/components/Skeleton/index.ts +5 -0
  218. package/src/components/Splitter/Splitter.stories.tsx +83 -0
  219. package/src/components/Splitter/Splitter.tsx +126 -0
  220. package/src/components/Splitter/index.ts +5 -0
  221. package/src/components/Status/Status.stories.tsx +21 -17
  222. package/src/components/Status/Status.tsx +2 -2
  223. package/src/components/Tag/Tag.stories.tsx +4 -9
  224. package/src/components/Tag/Tag.tsx +2 -7
  225. package/src/components/ThemeProvider/ThemeProvider.stories.tsx +32 -0
  226. package/src/components/ThemeProvider/ThemeProvider.tsx +4 -3
  227. package/src/components/Toast/Toast.stories.tsx +2 -2
  228. package/src/components/Toast/Toast.tsx +10 -14
  229. package/src/components/Toolbar/Toolbar.stories.tsx +2 -2
  230. package/src/components/Toolbar/Toolbar.tsx +174 -12
  231. package/src/components/Tooltip/Tooltip.stories.tsx +15 -13
  232. package/src/components/Tooltip/Tooltip.tsx +3 -2
  233. package/src/components/index.ts +9 -5
  234. package/src/exemplars/generics.stories.tsx +49 -0
  235. package/src/exemplars/slot.stories.tsx +107 -0
  236. package/src/exemplars/tabster.stories.tsx +127 -0
  237. package/src/exemplars/virtualizer.stories.tsx +137 -0
  238. package/src/index.ts +1 -0
  239. package/src/playground/Controls.stories.tsx +3 -10
  240. package/src/playground/Custom.stories.tsx +10 -10
  241. package/src/playground/Typography.stories.tsx +3 -3
  242. package/src/primitives/Column/Column.stories.tsx +78 -0
  243. package/src/primitives/Column/Column.tsx +134 -0
  244. package/src/primitives/Column/index.ts +5 -0
  245. package/src/primitives/Container/Container.stories.tsx +30 -0
  246. package/src/primitives/Container/Container.tsx +22 -0
  247. package/src/primitives/Container/index.ts +5 -0
  248. package/src/primitives/Flex/Flex.stories.tsx +58 -0
  249. package/src/primitives/Flex/Flex.tsx +29 -0
  250. package/src/primitives/Flex/index.ts +5 -0
  251. package/src/primitives/Grid/Grid.stories.tsx +57 -0
  252. package/src/primitives/Grid/Grid.tsx +35 -0
  253. package/src/primitives/Grid/index.ts +5 -0
  254. package/src/primitives/Panel/Panel.stories.tsx +67 -0
  255. package/src/primitives/Panel/Panel.tsx +119 -0
  256. package/src/primitives/Panel/index.ts +5 -0
  257. package/src/primitives/index.ts +9 -0
  258. package/src/testing/Loading.tsx +26 -0
  259. package/src/testing/decorators/withLayout.tsx +21 -7
  260. package/src/testing/decorators/withLayoutVariants.tsx +18 -21
  261. package/src/testing/decorators/withTheme.tsx +19 -17
  262. package/src/testing/index.ts +2 -0
  263. package/src/translations.ts +19 -0
  264. package/dist/lib/browser/chunk-CEKVHJ27.mjs.map +0 -7
  265. package/dist/lib/node-esm/chunk-2NHEX4AD.mjs.map +0 -7
  266. package/dist/types/src/components/AnchoredOverflow/AnchoredOverflow.d.ts +0 -15
  267. package/dist/types/src/components/AnchoredOverflow/AnchoredOverflow.d.ts.map +0 -1
  268. package/dist/types/src/components/AnchoredOverflow/index.d.ts +0 -2
  269. package/dist/types/src/components/AnchoredOverflow/index.d.ts.map +0 -1
  270. package/src/components/AnchoredOverflow/AnchoredOverflow.tsx +0 -59
  271. package/src/components/AnchoredOverflow/index.ts +0 -5
@@ -0,0 +1,57 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
6
+ import React from 'react';
7
+
8
+ import { type ChromaticPalette } from '@dxos/ui-types';
9
+
10
+ import { withLayout, withTheme } from '../../testing';
11
+
12
+ import { Grid } from './Grid';
13
+
14
+ const Cell = ({ label, hue }: { label: string; hue: ChromaticPalette }) => (
15
+ <div data-hue={hue} className='dx-panel p-2 text-sm font-mono border rounded-sm'>
16
+ {label}
17
+ </div>
18
+ );
19
+
20
+ const ColsStory = () => (
21
+ <Grid cols={3} classNames='gap-2 p-2'>
22
+ <Cell label='Row 1' hue='red' />
23
+ <Cell label='Row 2' hue='green' />
24
+ <Cell label='Row 3' hue='blue' />
25
+ </Grid>
26
+ );
27
+
28
+ const RowsStory = () => (
29
+ <Grid rows={3} classNames='gap-2 p-2'>
30
+ <Cell label='Row 1' hue='red' />
31
+ <Cell label='Row 2' hue='green' />
32
+ <Cell label='Row 3' hue='blue' />
33
+ </Grid>
34
+ );
35
+
36
+ const MixedStory = () => (
37
+ <Grid cols={2} rows={2} classNames='gap-2 p-2'>
38
+ <Cell label='A' hue='red' />
39
+ <Cell label='B' hue='green' />
40
+ <Cell label='C' hue='blue' />
41
+ <Cell label='D' hue='yellow' />
42
+ </Grid>
43
+ );
44
+
45
+ const meta: Meta = {
46
+ title: 'ui/react-ui-core/primitives/Grid',
47
+ decorators: [withTheme(), withLayout({ layout: 'column' })],
48
+ parameters: { layout: 'fullscreen' },
49
+ };
50
+
51
+ export default meta;
52
+
53
+ type Story = StoryObj<typeof meta>;
54
+
55
+ export const Cols: Story = { render: ColsStory };
56
+ export const Rows: Story = { render: RowsStory };
57
+ export const Mixed: Story = { render: MixedStory };
@@ -0,0 +1,35 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import React, { forwardRef } from 'react';
6
+
7
+ import { composableProps, mx } from '@dxos/ui-theme';
8
+ import { type ComposableProps } from '@dxos/ui-types';
9
+
10
+ export type GridProps = ComposableProps<HTMLDivElement> & {
11
+ cols?: number;
12
+ rows?: number;
13
+ grow?: boolean;
14
+ };
15
+
16
+ export const Grid = forwardRef<HTMLDivElement, GridProps>(
17
+ ({ children, style, role, cols, rows, grow = true, ...props }, forwardedRef) => {
18
+ const { className, ...rest } = composableProps(props);
19
+ return (
20
+ <div
21
+ ref={forwardedRef}
22
+ {...rest}
23
+ role={role ?? 'none'}
24
+ className={mx('grid overflow-hidden', grow && 'dx-container', className)}
25
+ style={{
26
+ gridTemplateColumns: cols ? `repeat(${cols}, 1fr)` : undefined,
27
+ gridTemplateRows: rows ? `repeat(${rows}, 1fr)` : undefined,
28
+ ...style,
29
+ }}
30
+ >
31
+ {children}
32
+ </div>
33
+ );
34
+ },
35
+ );
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ export * from './Grid';
@@ -0,0 +1,67 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { type Meta, type StoryObj } from '@storybook/react-vite';
6
+ import React from 'react';
7
+
8
+ import { Input, ScrollArea, Toolbar } from '../../components';
9
+ import { withLayout, withTheme } from '../../testing';
10
+
11
+ import { Panel } from './Panel';
12
+
13
+ const List = () => {
14
+ return (
15
+ <ScrollArea.Root margin role='list'>
16
+ <ScrollArea.Viewport>
17
+ {Array.from({ length: 100 }).map((_, i) => (
18
+ <div key={i} role='listitem' className='p-1 hover:bg-hover-surface'>
19
+ Item {i}
20
+ </div>
21
+ ))}
22
+ </ScrollArea.Viewport>
23
+ </ScrollArea.Root>
24
+ );
25
+ };
26
+
27
+ const DefaultStory = () => {
28
+ return (
29
+ <Panel.Root className='dx-article'>
30
+ <Panel.Toolbar asChild>
31
+ <Toolbar.Root classNames='gap-2'>
32
+ <Toolbar.IconButton icon='ph--plus--regular' variant='primary' label='Add' />
33
+ <Input.Root>
34
+ <Input.TextInput placeholder='Search' />
35
+ </Input.Root>
36
+ <Toolbar.IconButton icon='ph--dots-three-vertical--regular' iconOnly label='Menu' />
37
+ </Toolbar.Root>
38
+ </Panel.Toolbar>
39
+
40
+ <Panel.Content asChild>
41
+ <List />
42
+ </Panel.Content>
43
+
44
+ <Panel.Statusbar asChild>
45
+ <Toolbar.Root classNames='justify-between'>
46
+ <Toolbar.IconButton variant='ghost' icon='ph--house--regular' iconOnly label='Add' size={4} />
47
+ <Toolbar.IconButton variant='ghost' icon='ph--alarm--regular' iconOnly label='Status' size={4} />
48
+ </Toolbar.Root>
49
+ </Panel.Statusbar>
50
+ </Panel.Root>
51
+ );
52
+ };
53
+
54
+ const meta: Meta = {
55
+ title: 'ui/react-ui-core/primitives/Panel',
56
+ render: DefaultStory,
57
+ decorators: [withTheme(), withLayout({ layout: 'fullscreen' })],
58
+ parameters: {
59
+ layout: 'fullscreen',
60
+ },
61
+ };
62
+
63
+ export default meta;
64
+
65
+ type Story = StoryObj<typeof meta>;
66
+
67
+ export const Default: Story = {};
@@ -0,0 +1,119 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { Primitive } from '@radix-ui/react-primitive';
6
+ import { Slot } from '@radix-ui/react-slot';
7
+ import React, { forwardRef } from 'react';
8
+
9
+ import { composableProps } from '@dxos/ui-theme';
10
+ import { type SlottableProps } from '@dxos/ui-types';
11
+
12
+ import { useThemeContext } from '../../hooks';
13
+
14
+ //
15
+ // Root
16
+ //
17
+
18
+ const GRID_TEMPLATE_ROWS = 'auto 1fr auto';
19
+ const GRID_TEMPLATE_AREAS = '"toolbar" "content" "statusbar"';
20
+
21
+ type RootProps = SlottableProps<HTMLDivElement>;
22
+
23
+ const Root = forwardRef<HTMLDivElement, RootProps>(({ children, asChild, role, style, ...props }, forwardedRef) => {
24
+ const { className, ...rest } = composableProps(props);
25
+ const Comp = asChild ? Slot : Primitive.div;
26
+ const { tx } = useThemeContext();
27
+ return (
28
+ <Comp
29
+ {...rest}
30
+ role={role ?? 'none'}
31
+ style={{
32
+ gridTemplateRows: GRID_TEMPLATE_ROWS,
33
+ gridTemplateAreas: GRID_TEMPLATE_AREAS,
34
+ ...style,
35
+ }}
36
+ className={tx('panel.root', {}, className)}
37
+ ref={forwardedRef}
38
+ >
39
+ {children}
40
+ </Comp>
41
+ );
42
+ });
43
+
44
+ Root.displayName = 'Panel.Root';
45
+
46
+ //
47
+ // Toolbar
48
+ //
49
+
50
+ type ToolbarProps = SlottableProps<HTMLDivElement>;
51
+
52
+ const Toolbar = forwardRef<HTMLDivElement, ToolbarProps>(({ children, asChild, ...props }, forwardedRef) => {
53
+ const { className, ...rest } = composableProps(props);
54
+ const Comp = asChild ? Slot : Primitive.div;
55
+ const { tx } = useThemeContext();
56
+ return (
57
+ <Comp {...rest} data-slot='toolbar' className={tx('panel.toolbar', {}, className)} ref={forwardedRef}>
58
+ {children}
59
+ </Comp>
60
+ );
61
+ });
62
+
63
+ Toolbar.displayName = 'Panel.Toolbar';
64
+
65
+ //
66
+ // Content
67
+ //
68
+
69
+ type ContentProps = SlottableProps<HTMLDivElement>;
70
+
71
+ const Content = forwardRef<HTMLDivElement, ContentProps>(({ children, asChild, ...props }, forwardedRef) => {
72
+ const { className, ...rest } = composableProps(props);
73
+ const Comp = asChild ? Slot : Primitive.div;
74
+ const { tx } = useThemeContext();
75
+ return (
76
+ <Comp {...rest} data-slot='content' className={tx('panel.content', {}, className)} ref={forwardedRef}>
77
+ {children}
78
+ </Comp>
79
+ );
80
+ });
81
+
82
+ Content.displayName = 'Panel.Content';
83
+
84
+ //
85
+ // Statusbar
86
+ //
87
+
88
+ type StatusbarProps = SlottableProps<HTMLDivElement>;
89
+
90
+ const Statusbar = forwardRef<HTMLDivElement, StatusbarProps>(({ children, asChild, ...props }, forwardedRef) => {
91
+ const { className, ...rest } = composableProps(props);
92
+ const Comp = asChild ? Slot : Primitive.div;
93
+ const { tx } = useThemeContext();
94
+ return (
95
+ <Comp {...rest} data-slot='statusbar' className={tx('panel.statusbar', {}, className)} ref={forwardedRef}>
96
+ {children}
97
+ </Comp>
98
+ );
99
+ });
100
+
101
+ Statusbar.displayName = 'Panel.Statusbar';
102
+
103
+ //
104
+ // Panel
105
+ //
106
+
107
+ export const Panel = {
108
+ Root,
109
+ Toolbar,
110
+ Content,
111
+ Statusbar,
112
+ };
113
+
114
+ export type {
115
+ RootProps as PanelRootProps,
116
+ ToolbarProps as PanelToolbarProps,
117
+ ContentProps as PanelContentProps,
118
+ StatusbarProps as PanelStatusbarProps,
119
+ };
@@ -0,0 +1,5 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ export * from './Panel';
@@ -0,0 +1,9 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ export * from './Column';
6
+ export * from './Container';
7
+ export * from './Flex';
8
+ export * from './Grid';
9
+ export * from './Panel';
@@ -0,0 +1,26 @@
1
+ //
2
+ // Copyright 2023 DXOS.org
3
+ //
4
+
5
+ import { mx } from '@dxos/ui-theme';
6
+ import { safeStringify } from '@dxos/util';
7
+ import React, { useEffect, useState } from 'react';
8
+
9
+ export type LoadingProps = { data?: any };
10
+
11
+ /**
12
+ * Storybook loading component.
13
+ */
14
+ export const Loading = ({ data }: LoadingProps) => {
15
+ const [visible, setVisible] = useState(false);
16
+ useEffect(() => {
17
+ const t = setTimeout(() => setVisible(true), 500);
18
+ return () => clearTimeout(t);
19
+ }, []);
20
+ return (
21
+ <div className={mx('flex flex-col opacity-0 transition delay-1000 duration-1000', visible && 'opacity-100')}>
22
+ <h2 className='uppercase'>Loading</h2>
23
+ <pre>{safeStringify(data, undefined, 2)}</pre>
24
+ </div>
25
+ );
26
+ };
@@ -10,7 +10,7 @@ import { mx } from '@dxos/ui-theme';
10
10
 
11
11
  export type ContainerProps = ThemedClassName<PropsWithChildren>;
12
12
 
13
- export type ContainerType = 'fullscreen' | 'column';
13
+ export type ContainerType = 'default' | 'fullscreen' | 'centered' | 'column';
14
14
 
15
15
  export type WithLayoutProps =
16
16
  | FC<ContainerProps>
@@ -36,7 +36,7 @@ export const withLayout =
36
36
  </Container>
37
37
  );
38
38
  } else {
39
- const { layout = 'fullscreen', classNames, scroll } = props;
39
+ const { layout = 'default', classNames, scroll } = props;
40
40
  const Container = layouts[layout] ?? layouts.fullscreen;
41
41
  return (
42
42
  <Container classNames={mx(classNames, scroll ? 'overflow-y-auto' : 'overflow-hidden')}>
@@ -47,15 +47,29 @@ export const withLayout =
47
47
  };
48
48
 
49
49
  const layouts: Record<ContainerType, FC<ContainerProps>> = {
50
- fullscreen: ({ children, classNames }: ContainerProps) => (
51
- <div role='none' className={mx('fixed inset-0 flex overflow-hidden bg-deckSurface', classNames)}>
50
+ default: ({ classNames, children }: ContainerProps) => (
51
+ <div role='none' className={mx('p-4', classNames)}>
52
52
  {children}
53
53
  </div>
54
54
  ),
55
55
 
56
- column: ({ children, classNames }: ContainerProps) => (
57
- <div role='none' className='fixed inset-0 flex justify-center overflow-hidden bg-deckSurface'>
58
- <div role='none' className={mx('flex flex-col is-[40rem] bg-baseSurface', classNames)}>
56
+ fullscreen: ({ classNames, children }: ContainerProps) => (
57
+ <div role='none' className={mx('fixed inset-0 flex overflow-hidden bg-base-surface', classNames)}>
58
+ {children}
59
+ </div>
60
+ ),
61
+
62
+ centered: ({ classNames, children }: ContainerProps) => (
63
+ <div role='none' className={mx('fixed inset-0 grid overflow-hidden place-items-center bg-base-surface')}>
64
+ <div role='none' className={mx('flex flex-col w-[40rem] bg-group-surface', classNames)}>
65
+ {children}
66
+ </div>
67
+ </div>
68
+ ),
69
+
70
+ column: ({ classNames, children }: ContainerProps) => (
71
+ <div role='none' className='fixed inset-0 flex overflow-hidden justify-center bg-base-surface'>
72
+ <div role='none' className={mx('flex flex-col w-[40rem] bg-group-surface', classNames)}>
59
73
  {children}
60
74
  </div>
61
75
  </div>
@@ -14,7 +14,7 @@ type Config = {
14
14
  };
15
15
 
16
16
  const Container = ({ children, elevation, surface }: PropsWithChildren<{ elevation: Elevation; surface?: string }>) => (
17
- <div className={mx('rounded-md border border-separator', surface, surfaceShadow({ elevation }))}>{children}</div>
17
+ <div className={mx('p-4 rounded-sm', surface, surfaceShadow({ elevation }))}>{children}</div>
18
18
  );
19
19
 
20
20
  const Panel = ({
@@ -22,30 +22,27 @@ const Panel = ({
22
22
  elevations,
23
23
  densities,
24
24
  className,
25
- }: { Story: ComponentType } & Config & { className?: string }) => (
26
- <div className={mx('flex flex-col bs-full p-4 gap-4', className)}>
27
- {elevations?.map(({ elevation, surface }) =>
28
- densities?.map((density) => (
29
- <Container key={`${elevation}--${density}`} surface={surface} elevation={elevation}>
30
- <Story />
31
- </Container>
32
- )),
33
- )}
34
- </div>
35
- );
25
+ }: { Story: ComponentType } & Config & { className?: string }) => {
26
+ return (
27
+ <div className={mx('flex flex-col h-full p-8 gap-8', className)}>
28
+ {elevations?.map(({ elevation, surface }) =>
29
+ densities?.map((density) => (
30
+ <Container key={`${elevation}--${density}`} surface={surface} elevation={elevation}>
31
+ <Story />
32
+ </Container>
33
+ )),
34
+ )}
35
+ </div>
36
+ );
37
+ };
36
38
 
37
39
  export const withLayoutVariants = ({
38
40
  elevations = [
39
- { elevation: 'base', surface: 'bg-baseSurface' },
40
- { elevation: 'positioned', surface: 'bg-cardSurface' },
41
- { elevation: 'dialog', surface: 'bg-modalSurface' },
41
+ { elevation: 'dialog', surface: 'bg-modal-surface' },
42
+ { elevation: 'positioned', surface: 'bg-card-surface' },
43
+ { elevation: 'base', surface: 'bg-base-surface' },
42
44
  ],
43
45
  densities = ['coarse'],
44
46
  }: Config = {}): Decorator => {
45
- return (Story) => (
46
- <div className='fixed inset-0 grid grid-cols-2 overflow-y-auto'>
47
- <Panel Story={Story} className='light' elevations={elevations} densities={densities} />
48
- <Panel Story={Story} className='dark' elevations={elevations} densities={densities} />
49
- </div>
50
- );
47
+ return (Story) => <Panel Story={Story} elevations={elevations} densities={densities} />;
51
48
  };
@@ -8,25 +8,27 @@ import React, { memo } from 'react';
8
8
  import { defaultTx } from '@dxos/ui-theme';
9
9
  import { type ThemeMode } from '@dxos/ui-types';
10
10
 
11
- import { ThemeProvider, Tooltip } from '../../components';
11
+ import { type ThemeContextValue, ThemeProvider, Tooltip } from '../../components';
12
12
 
13
13
  /**
14
- * Adds theme decorator (add to preview.ts)
14
+ * Adds theme decorator.
15
15
  */
16
- export const withTheme: Decorator = (Story, context) => {
17
- const {
18
- globals: { theme },
19
- parameters: { translations },
20
- } = context;
16
+ export const withTheme =
17
+ ({ tx = defaultTx, ...props }: Partial<ThemeContextValue> = {}): Decorator =>
18
+ (Story, context) => {
19
+ const {
20
+ globals: { theme },
21
+ parameters: { translations },
22
+ } = context;
21
23
 
22
- // Prevent re-rendering of the story.
23
- const MemoizedStory = memo(Story);
24
+ // Prevent re-rendering of the story.
25
+ const MemoizedStory = memo(Story);
24
26
 
25
- return (
26
- <ThemeProvider tx={defaultTx} themeMode={theme as ThemeMode} resourceExtensions={translations} noCache>
27
- <Tooltip.Provider>
28
- <MemoizedStory />
29
- </Tooltip.Provider>
30
- </ThemeProvider>
31
- );
32
- };
27
+ return (
28
+ <ThemeProvider {...props} tx={tx} themeMode={theme as ThemeMode} resourceExtensions={translations} noCache>
29
+ <Tooltip.Provider>
30
+ <MemoizedStory />
31
+ </Tooltip.Provider>
32
+ </ThemeProvider>
33
+ );
34
+ };
@@ -3,3 +3,5 @@
3
3
  //
4
4
 
5
5
  export * from './decorators';
6
+
7
+ export * from './Loading';
@@ -0,0 +1,19 @@
1
+ //
2
+ // Copyright 2023 DXOS.org
3
+ //
4
+
5
+ import { type Resource } from '@dxos/react-ui';
6
+
7
+ export const translationKey = '@dxos/react-ui';
8
+
9
+ export const translations = [
10
+ {
11
+ 'en-US': {
12
+ [translationKey]: {
13
+ 'toolbar menu label': 'Action menu',
14
+ 'toolbar drag handle label': 'Drag to rearrange',
15
+ 'toolbar close label': 'Close',
16
+ },
17
+ },
18
+ },
19
+ ] as const satisfies Resource[];