@proyecto-viviana/solidaria 0.2.4 → 0.2.8

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 (219) hide show
  1. package/LICENSE +21 -0
  2. package/dist/actiongroup/createActionGroup.d.ts +29 -0
  3. package/dist/actiongroup/createActionGroup.d.ts.map +1 -0
  4. package/dist/actiongroup/index.d.ts +2 -0
  5. package/dist/actiongroup/index.d.ts.map +1 -0
  6. package/dist/autocomplete/createAutocomplete.d.ts +6 -2
  7. package/dist/autocomplete/createAutocomplete.d.ts.map +1 -1
  8. package/dist/breadcrumbs/createBreadcrumbs.d.ts +2 -0
  9. package/dist/breadcrumbs/createBreadcrumbs.d.ts.map +1 -1
  10. package/dist/button/createToggleButtonGroup.d.ts +32 -0
  11. package/dist/button/createToggleButtonGroup.d.ts.map +1 -0
  12. package/dist/button/index.d.ts +2 -0
  13. package/dist/button/index.d.ts.map +1 -1
  14. package/dist/calendar/createCalendarCell.d.ts +2 -0
  15. package/dist/calendar/createCalendarCell.d.ts.map +1 -1
  16. package/dist/calendar/createCalendarGrid.d.ts.map +1 -1
  17. package/dist/calendar/createRangeCalendarCell.d.ts +3 -1
  18. package/dist/calendar/createRangeCalendarCell.d.ts.map +1 -1
  19. package/dist/checkbox/createCheckboxGroup.d.ts +5 -1
  20. package/dist/checkbox/createCheckboxGroup.d.ts.map +1 -1
  21. package/dist/collections/index.d.ts +56 -0
  22. package/dist/collections/index.d.ts.map +1 -0
  23. package/dist/color/createColorArea.d.ts.map +1 -1
  24. package/dist/color/createColorSlider.d.ts.map +1 -1
  25. package/dist/color/createColorWheel.d.ts.map +1 -1
  26. package/dist/combobox/createComboBox.d.ts +6 -0
  27. package/dist/combobox/createComboBox.d.ts.map +1 -1
  28. package/dist/datepicker/createDatePicker.d.ts +6 -0
  29. package/dist/datepicker/createDatePicker.d.ts.map +1 -1
  30. package/dist/datepicker/createDateRangePicker.d.ts +40 -0
  31. package/dist/datepicker/createDateRangePicker.d.ts.map +1 -0
  32. package/dist/datepicker/createDateSegment.d.ts +1 -1
  33. package/dist/datepicker/createDateSegment.d.ts.map +1 -1
  34. package/dist/datepicker/createTimeSegment.d.ts +29 -0
  35. package/dist/datepicker/createTimeSegment.d.ts.map +1 -0
  36. package/dist/datepicker/index.d.ts +2 -0
  37. package/dist/datepicker/index.d.ts.map +1 -1
  38. package/dist/disclosure/createDisclosureGroup.d.ts +2 -1
  39. package/dist/disclosure/createDisclosureGroup.d.ts.map +1 -1
  40. package/dist/dnd/createDrag.d.ts.map +1 -1
  41. package/dist/dnd/createDraggableCollection.d.ts +4 -0
  42. package/dist/dnd/createDraggableCollection.d.ts.map +1 -1
  43. package/dist/dnd/createDraggableItem.d.ts.map +1 -1
  44. package/dist/dnd/createDrop.d.ts.map +1 -1
  45. package/dist/dnd/createDroppableCollection.d.ts +32 -1
  46. package/dist/dnd/createDroppableCollection.d.ts.map +1 -1
  47. package/dist/dnd/createDroppableItem.d.ts.map +1 -1
  48. package/dist/dnd/index.d.ts +1 -1
  49. package/dist/dnd/index.d.ts.map +1 -1
  50. package/dist/grid/createGrid.d.ts.map +1 -1
  51. package/dist/gridlist/createGridList.d.ts.map +1 -1
  52. package/dist/index.d.ts +6 -4
  53. package/dist/index.d.ts.map +1 -1
  54. package/dist/index.js +4659 -3452
  55. package/dist/index.js.map +1 -7
  56. package/dist/index.ssr.js +4659 -3452
  57. package/dist/index.ssr.js.map +1 -7
  58. package/dist/interactions/createFocus.d.ts.map +1 -1
  59. package/dist/interactions/createFocusWithin.d.ts.map +1 -1
  60. package/dist/link/createLink.d.ts +10 -0
  61. package/dist/link/createLink.d.ts.map +1 -1
  62. package/dist/listbox/createListBox.d.ts +1 -0
  63. package/dist/listbox/createListBox.d.ts.map +1 -1
  64. package/dist/listbox/createOption.d.ts.map +1 -1
  65. package/dist/menu/createMenu.d.ts +1 -0
  66. package/dist/menu/createMenu.d.ts.map +1 -1
  67. package/dist/meter/createMeter.d.ts.map +1 -1
  68. package/dist/numberfield/createNumberField.d.ts +18 -0
  69. package/dist/numberfield/createNumberField.d.ts.map +1 -1
  70. package/dist/overlays/createModal.d.ts +16 -0
  71. package/dist/overlays/createModal.d.ts.map +1 -1
  72. package/dist/overlays/createOverlay.d.ts.map +1 -1
  73. package/dist/overlays/index.d.ts +1 -1
  74. package/dist/overlays/index.d.ts.map +1 -1
  75. package/dist/popover/createOverlayPosition.d.ts.map +1 -1
  76. package/dist/popover/createPopover.d.ts.map +1 -1
  77. package/dist/progress/createProgressBar.d.ts.map +1 -1
  78. package/dist/radio/createRadioGroup.d.ts +2 -2
  79. package/dist/radio/createRadioGroup.d.ts.map +1 -1
  80. package/dist/searchfield/createSearchField.d.ts.map +1 -1
  81. package/dist/select/createHiddenSelect.d.ts.map +1 -1
  82. package/dist/select/createSelect.d.ts.map +1 -1
  83. package/dist/slider/createSlider.d.ts.map +1 -1
  84. package/dist/table/createTable.d.ts.map +1 -1
  85. package/dist/tabs/createTabs.d.ts +1 -1
  86. package/dist/tabs/createTabs.d.ts.map +1 -1
  87. package/dist/tag/createTag.d.ts.map +1 -1
  88. package/dist/tag/createTagGroup.d.ts.map +1 -1
  89. package/dist/toast/createToast.d.ts +4 -0
  90. package/dist/toast/createToast.d.ts.map +1 -1
  91. package/dist/toast/createToastRegion.d.ts.map +1 -1
  92. package/dist/toolbar/createToolbar.d.ts.map +1 -1
  93. package/dist/tooltip/createTooltipTrigger.d.ts.map +1 -1
  94. package/dist/tree/createTree.d.ts.map +1 -1
  95. package/dist/tree/createTreeItem.d.ts.map +1 -1
  96. package/dist/tree/types.d.ts +4 -0
  97. package/dist/tree/types.d.ts.map +1 -1
  98. package/dist/utils/env.d.ts +1 -1
  99. package/dist/utils/env.d.ts.map +1 -1
  100. package/dist/utils/platform.d.ts.map +1 -1
  101. package/dist/visually-hidden/createVisuallyHidden.d.ts.map +1 -1
  102. package/package.json +8 -6
  103. package/src/actiongroup/createActionGroup.ts +324 -0
  104. package/src/actiongroup/index.ts +8 -0
  105. package/src/autocomplete/createAutocomplete.ts +32 -9
  106. package/src/breadcrumbs/createBreadcrumbs.ts +10 -15
  107. package/src/button/createButton.ts +1 -1
  108. package/src/button/createToggleButtonGroup.ts +128 -0
  109. package/src/button/index.ts +9 -0
  110. package/src/calendar/createCalendarCell.ts +6 -4
  111. package/src/calendar/createCalendarGrid.ts +27 -18
  112. package/src/calendar/createRangeCalendarCell.ts +26 -9
  113. package/src/checkbox/createCheckboxGroup.ts +21 -4
  114. package/src/collections/index.ts +242 -0
  115. package/src/color/createColorArea.ts +380 -314
  116. package/src/color/createColorField.ts +137 -137
  117. package/src/color/createColorSlider.ts +286 -197
  118. package/src/color/createColorSwatch.ts +40 -40
  119. package/src/color/createColorWheel.ts +218 -208
  120. package/src/color/index.ts +24 -24
  121. package/src/color/types.ts +116 -116
  122. package/src/combobox/createComboBox.ts +670 -647
  123. package/src/combobox/index.ts +6 -6
  124. package/src/datepicker/createDatePicker.ts +54 -16
  125. package/src/datepicker/createDateRangePicker.ts +246 -0
  126. package/src/datepicker/createDateSegment.ts +185 -31
  127. package/src/datepicker/createTimeSegment.ts +370 -0
  128. package/src/datepicker/index.ts +14 -0
  129. package/src/dialog/createDialog.ts +120 -120
  130. package/src/dialog/index.ts +2 -2
  131. package/src/dialog/types.ts +19 -19
  132. package/src/disclosure/createDisclosureGroup.ts +5 -2
  133. package/src/dnd/createDrag.ts +224 -209
  134. package/src/dnd/createDraggableCollection.ts +96 -63
  135. package/src/dnd/createDraggableItem.ts +259 -243
  136. package/src/dnd/createDrop.ts +322 -321
  137. package/src/dnd/createDroppableCollection.ts +682 -293
  138. package/src/dnd/createDroppableItem.ts +215 -213
  139. package/src/dnd/index.ts +55 -47
  140. package/src/dnd/types.ts +89 -89
  141. package/src/dnd/utils.ts +294 -294
  142. package/src/focus/createAutoFocus.ts +321 -321
  143. package/src/focus/createFocusRestore.ts +313 -313
  144. package/src/focus/createVirtualFocus.ts +396 -396
  145. package/src/form/createFormValidation.ts +224 -224
  146. package/src/form/index.ts +11 -11
  147. package/src/grid/createGrid.ts +3 -1
  148. package/src/gridlist/createGridList.ts +16 -0
  149. package/src/gridlist/createGridListItem.ts +1 -1
  150. package/src/i18n/NumberFormatter.ts +266 -266
  151. package/src/i18n/createCollator.ts +79 -79
  152. package/src/i18n/createDateFormatter.ts +83 -83
  153. package/src/i18n/createFilter.ts +131 -131
  154. package/src/i18n/createNumberFormatter.ts +52 -52
  155. package/src/i18n/index.ts +40 -40
  156. package/src/i18n/locale.tsx +188 -188
  157. package/src/i18n/utils.ts +99 -99
  158. package/src/index.ts +51 -0
  159. package/src/interactions/createFocus.ts +6 -5
  160. package/src/interactions/createFocusWithin.ts +6 -5
  161. package/src/interactions/createLongPress.ts +174 -174
  162. package/src/interactions/createMove.ts +289 -289
  163. package/src/interactions/createPress.ts +5 -5
  164. package/src/landmark/createLandmark.ts +377 -377
  165. package/src/landmark/index.ts +8 -8
  166. package/src/link/createLink.ts +23 -8
  167. package/src/listbox/createListBox.ts +308 -269
  168. package/src/listbox/createOption.ts +162 -151
  169. package/src/listbox/index.ts +12 -12
  170. package/src/live-announcer/announce.ts +322 -322
  171. package/src/live-announcer/index.ts +9 -9
  172. package/src/menu/createMenu.ts +405 -396
  173. package/src/menu/createMenuItem.ts +149 -149
  174. package/src/menu/createMenuTrigger.ts +88 -88
  175. package/src/menu/index.ts +18 -18
  176. package/src/meter/createMeter.ts +1 -6
  177. package/src/numberfield/createNumberField.ts +311 -268
  178. package/src/numberfield/index.ts +5 -5
  179. package/src/overlays/ariaHideOutside.ts +219 -219
  180. package/src/overlays/createInteractOutside.ts +149 -149
  181. package/src/overlays/createModal.tsx +238 -202
  182. package/src/overlays/createOverlay.ts +165 -155
  183. package/src/overlays/createOverlayTrigger.ts +85 -85
  184. package/src/overlays/createPreventScroll.ts +266 -266
  185. package/src/overlays/index.ts +48 -44
  186. package/src/popover/calculatePosition.ts +6 -6
  187. package/src/popover/createOverlayPosition.ts +7 -4
  188. package/src/popover/createPopover.ts +21 -7
  189. package/src/progress/createProgressBar.ts +6 -1
  190. package/src/radio/createRadioGroup.ts +88 -14
  191. package/src/searchfield/createSearchField.ts +241 -186
  192. package/src/searchfield/index.ts +2 -2
  193. package/src/select/createHiddenSelect.tsx +263 -236
  194. package/src/select/createSelect.ts +373 -395
  195. package/src/select/index.ts +14 -14
  196. package/src/slider/createSlider.ts +364 -349
  197. package/src/slider/index.ts +2 -2
  198. package/src/ssr/index.tsx +370 -370
  199. package/src/table/createTable.ts +3 -1
  200. package/src/table/createTableColumnHeader.ts +1 -1
  201. package/src/table/createTableRow.ts +1 -1
  202. package/src/tabs/createTabs.ts +80 -51
  203. package/src/tag/createTag.ts +135 -6
  204. package/src/tag/createTagGroup.ts +7 -2
  205. package/src/toast/createToast.ts +8 -2
  206. package/src/toast/createToastRegion.ts +0 -1
  207. package/src/toolbar/createToolbar.ts +75 -1
  208. package/src/tooltip/createTooltip.ts +79 -79
  209. package/src/tooltip/createTooltipTrigger.ts +226 -222
  210. package/src/tooltip/index.ts +6 -6
  211. package/src/tree/createTree.ts +261 -246
  212. package/src/tree/createTreeItem.ts +282 -233
  213. package/src/tree/createTreeSelectionCheckbox.ts +68 -68
  214. package/src/tree/index.ts +16 -16
  215. package/src/tree/types.ts +91 -87
  216. package/src/utils/env.ts +55 -54
  217. package/src/utils/platform.ts +16 -6
  218. package/src/visually-hidden/createVisuallyHidden.ts +139 -124
  219. package/src/visually-hidden/index.ts +6 -6
@@ -1,246 +1,261 @@
1
- /**
2
- * createTree - Provides accessibility for a tree component.
3
- * Based on @react-aria/tree/useTree.
4
- */
5
-
6
- import { createMemo, type Accessor } from 'solid-js';
7
- import type { JSX } from 'solid-js';
8
- import { createId } from '@proyecto-viviana/solid-stately';
9
- import type { TreeState, TreeCollection, Key } from '@proyecto-viviana/solid-stately';
10
- import type { AriaTreeProps, TreeAria } from './types';
11
-
12
- /**
13
- * Metadata stored for a tree instance.
14
- */
15
- interface TreeData {
16
- /** The generated ID for the tree. */
17
- treeId: string;
18
- /** Actions registered for the tree. */
19
- actions: {
20
- onAction?: (key: Key) => void;
21
- };
22
- }
23
-
24
- /**
25
- * WeakMap to store tree data for child components to access.
26
- */
27
- const treeDataMap = new WeakMap<object, TreeData>();
28
-
29
- /**
30
- * Gets the tree data for a given state.
31
- */
32
- export function getTreeData<T extends object, C extends TreeCollection<T>>(
33
- state: TreeState<T, C>
34
- ): TreeData | undefined {
35
- return treeDataMap.get(state);
36
- }
37
-
38
- /**
39
- * Creates accessibility props for a tree.
40
- */
41
- export function createTree<T extends object, C extends TreeCollection<T> = TreeCollection<T>>(
42
- props: Accessor<AriaTreeProps>,
43
- state: Accessor<TreeState<T, C>>,
44
- _ref: Accessor<HTMLDivElement | null>
45
- ): TreeAria {
46
- // Generate a unique ID for the tree
47
- const treeId = props().id ?? createId();
48
-
49
- // Store tree data for child components
50
- const treeData: TreeData = {
51
- treeId,
52
- actions: {
53
- get onAction() {
54
- return props().onAction;
55
- },
56
- },
57
- };
58
-
59
- // Store in WeakMap using the state as key
60
- treeDataMap.set(state(), treeData);
61
-
62
- // Handle keyboard navigation
63
- const onKeyDown = (e: KeyboardEvent) => {
64
- const s = state();
65
- const p = props();
66
- const collection = s.collection;
67
- const focusedKey = s.focusedKey;
68
-
69
- if (p.isDisabled) return;
70
-
71
- switch (e.key) {
72
- case 'ArrowDown': {
73
- e.preventDefault();
74
- if (focusedKey != null) {
75
- const nextKey = collection.getKeyAfter(focusedKey);
76
- if (nextKey != null) {
77
- s.setFocusedKey(nextKey);
78
- }
79
- } else {
80
- const firstKey = collection.getFirstKey();
81
- if (firstKey != null) {
82
- s.setFocusedKey(firstKey);
83
- }
84
- }
85
- break;
86
- }
87
- case 'ArrowUp': {
88
- e.preventDefault();
89
- if (focusedKey != null) {
90
- const prevKey = collection.getKeyBefore(focusedKey);
91
- if (prevKey != null) {
92
- s.setFocusedKey(prevKey);
93
- }
94
- } else {
95
- const lastKey = collection.getLastKey();
96
- if (lastKey != null) {
97
- s.setFocusedKey(lastKey);
98
- }
99
- }
100
- break;
101
- }
102
- case 'ArrowRight': {
103
- e.preventDefault();
104
- if (focusedKey != null) {
105
- const node = collection.getItem(focusedKey);
106
- if (node?.isExpandable) {
107
- if (!s.isExpanded(focusedKey)) {
108
- // Expand the node
109
- s.expandKey(focusedKey);
110
- } else {
111
- // Move to first child
112
- const children = [...collection.getChildren(focusedKey)];
113
- if (children.length > 0) {
114
- s.setFocusedKey(children[0].key);
115
- }
116
- }
117
- }
118
- }
119
- break;
120
- }
121
- case 'ArrowLeft': {
122
- e.preventDefault();
123
- if (focusedKey != null) {
124
- const node = collection.getItem(focusedKey);
125
- if (node?.isExpandable && s.isExpanded(focusedKey)) {
126
- // Collapse the node
127
- s.collapseKey(focusedKey);
128
- } else if (node?.parentKey != null) {
129
- // Move to parent
130
- s.setFocusedKey(node.parentKey);
131
- }
132
- }
133
- break;
134
- }
135
- case 'Home': {
136
- e.preventDefault();
137
- const firstKey = collection.getFirstKey();
138
- if (firstKey != null) {
139
- s.setFocusedKey(firstKey);
140
- }
141
- break;
142
- }
143
- case 'End': {
144
- e.preventDefault();
145
- const lastKey = collection.getLastKey();
146
- if (lastKey != null) {
147
- s.setFocusedKey(lastKey);
148
- }
149
- break;
150
- }
151
- case 'a':
152
- case 'A': {
153
- if ((e.ctrlKey || e.metaKey) && s.selectionMode === 'multiple') {
154
- e.preventDefault();
155
- s.selectAll();
156
- }
157
- break;
158
- }
159
- case 'Escape': {
160
- if (s.selectionMode !== 'none') {
161
- e.preventDefault();
162
- s.clearSelection();
163
- }
164
- break;
165
- }
166
- case '*': {
167
- // Expand all siblings at current level
168
- e.preventDefault();
169
- if (focusedKey != null) {
170
- const node = collection.getItem(focusedKey);
171
- if (node) {
172
- // Find all siblings at the same level
173
- const parentKey = node.parentKey;
174
- let siblings: Key[];
175
- if (parentKey != null) {
176
- siblings = [...collection.getChildren(parentKey)].map((n) => n.key);
177
- } else {
178
- // Root level siblings
179
- siblings = collection.rows
180
- .filter((n) => n.level === 0)
181
- .map((n) => n.key);
182
- }
183
- // Expand all expandable siblings
184
- for (const siblingKey of siblings) {
185
- const sibling = collection.getItem(siblingKey);
186
- if (sibling?.isExpandable && !s.isExpanded(siblingKey)) {
187
- s.expandKey(siblingKey);
188
- }
189
- }
190
- }
191
- }
192
- break;
193
- }
194
- }
195
- };
196
-
197
- const onFocus = () => {
198
- const s = state();
199
- s.setFocused(true);
200
-
201
- // If nothing is focused, focus the first item
202
- if (s.focusedKey == null) {
203
- const firstKey = s.collection.getFirstKey();
204
- if (firstKey != null) {
205
- s.setFocusedKey(firstKey);
206
- }
207
- }
208
- };
209
-
210
- const onBlur = () => {
211
- const s = state();
212
- s.setFocused(false);
213
- };
214
-
215
- const treeProps = createMemo(() => {
216
- const p = props();
217
- const s = state();
218
-
219
- const baseProps: Record<string, unknown> = {
220
- role: 'treegrid',
221
- id: treeId,
222
- 'aria-label': p['aria-label'],
223
- 'aria-labelledby': p['aria-labelledby'],
224
- 'aria-describedby': p['aria-describedby'],
225
- 'aria-multiselectable': s.selectionMode === 'multiple' ? true : undefined,
226
- 'aria-disabled': p.isDisabled || undefined,
227
- tabIndex: p.isDisabled ? undefined : 0,
228
- onKeyDown,
229
- onFocus,
230
- onBlur,
231
- };
232
-
233
- // Add row count for virtualized trees
234
- if (p.isVirtualized) {
235
- baseProps['aria-rowcount'] = s.collection.rowCount;
236
- }
237
-
238
- return baseProps as JSX.HTMLAttributes<HTMLDivElement>;
239
- });
240
-
241
- return {
242
- get treeProps() {
243
- return treeProps();
244
- },
245
- };
246
- }
1
+ /**
2
+ * createTree - Provides accessibility for a tree component.
3
+ * Based on @react-aria/tree/useTree.
4
+ */
5
+
6
+ import { createMemo, type Accessor } from 'solid-js';
7
+ import type { JSX } from 'solid-js';
8
+ import { createId } from '@proyecto-viviana/solid-stately';
9
+ import type { TreeState, TreeCollection, Key } from '@proyecto-viviana/solid-stately';
10
+ import type { AriaTreeProps, TreeAria } from './types';
11
+
12
+ /**
13
+ * Metadata stored for a tree instance.
14
+ */
15
+ interface TreeData {
16
+ /** The generated ID for the tree. */
17
+ treeId: string;
18
+ /** Actions registered for the tree. */
19
+ actions: {
20
+ onAction?: (key: Key) => void;
21
+ };
22
+ }
23
+
24
+ /**
25
+ * WeakMap to store tree data for child components to access.
26
+ */
27
+ const treeDataMap = new WeakMap<object, TreeData>();
28
+
29
+ /**
30
+ * Gets the tree data for a given state.
31
+ */
32
+ export function getTreeData<T extends object, C extends TreeCollection<T>>(
33
+ state: TreeState<T, C>
34
+ ): TreeData | undefined {
35
+ return treeDataMap.get(state);
36
+ }
37
+
38
+ /**
39
+ * Creates accessibility props for a tree.
40
+ */
41
+ export function createTree<T extends object, C extends TreeCollection<T> = TreeCollection<T>>(
42
+ props: Accessor<AriaTreeProps>,
43
+ state: Accessor<TreeState<T, C>>,
44
+ _ref: Accessor<HTMLDivElement | null>
45
+ ): TreeAria {
46
+ // Generate a unique ID for the tree
47
+ const treeId = props().id ?? createId();
48
+
49
+ // Store tree data for child components
50
+ const treeData: TreeData = {
51
+ treeId,
52
+ actions: {
53
+ get onAction() {
54
+ return props().onAction;
55
+ },
56
+ },
57
+ };
58
+
59
+ // Store in WeakMap using the state as key
60
+ treeDataMap.set(state(), treeData);
61
+
62
+ // Handle keyboard navigation
63
+ const onKeyDown = (e: KeyboardEvent) => {
64
+ const s = state();
65
+ const p = props();
66
+ const collection = s.collection;
67
+ const focusedKey = s.focusedKey;
68
+ const direction = p.direction ?? 'ltr';
69
+ // In RTL, ArrowLeft expands and ArrowRight collapses (opposite of LTR)
70
+ const expandKey = direction === 'rtl' ? 'ArrowLeft' : 'ArrowRight';
71
+ const collapseKey = direction === 'rtl' ? 'ArrowRight' : 'ArrowLeft';
72
+
73
+ if (p.isDisabled) return;
74
+
75
+ switch (e.key) {
76
+ case 'ArrowDown': {
77
+ e.preventDefault();
78
+ if (focusedKey != null) {
79
+ const nextKey = collection.getKeyAfter(focusedKey);
80
+ if (nextKey != null) {
81
+ s.setFocusedKey(nextKey);
82
+ }
83
+ } else {
84
+ const firstKey = collection.getFirstKey();
85
+ if (firstKey != null) {
86
+ s.setFocusedKey(firstKey);
87
+ }
88
+ }
89
+ break;
90
+ }
91
+ case 'ArrowUp': {
92
+ e.preventDefault();
93
+ if (focusedKey != null) {
94
+ const prevKey = collection.getKeyBefore(focusedKey);
95
+ if (prevKey != null) {
96
+ s.setFocusedKey(prevKey);
97
+ }
98
+ } else {
99
+ const lastKey = collection.getLastKey();
100
+ if (lastKey != null) {
101
+ s.setFocusedKey(lastKey);
102
+ }
103
+ }
104
+ break;
105
+ }
106
+ case 'ArrowRight':
107
+ case 'ArrowLeft': {
108
+ e.preventDefault();
109
+ if (focusedKey != null) {
110
+ const node = collection.getItem(focusedKey);
111
+ if (e.key === expandKey) {
112
+ // Expand or move to first child
113
+ if (node?.isExpandable) {
114
+ if (!s.isExpanded(focusedKey)) {
115
+ s.expandKey(focusedKey);
116
+ } else {
117
+ const children = [...collection.getChildren(focusedKey)];
118
+ if (children.length > 0) {
119
+ s.setFocusedKey(children[0].key);
120
+ }
121
+ }
122
+ }
123
+ } else if (e.key === collapseKey) {
124
+ // Collapse or move to parent
125
+ if (node?.isExpandable && s.isExpanded(focusedKey)) {
126
+ s.collapseKey(focusedKey);
127
+ } else if (node?.parentKey != null) {
128
+ s.setFocusedKey(node.parentKey);
129
+ }
130
+ }
131
+ }
132
+ break;
133
+ }
134
+ case 'Home': {
135
+ e.preventDefault();
136
+ const firstKey = collection.getFirstKey();
137
+ if (firstKey != null) {
138
+ s.setFocusedKey(firstKey);
139
+ }
140
+ break;
141
+ }
142
+ case 'End': {
143
+ e.preventDefault();
144
+ const lastKey = collection.getLastKey();
145
+ if (lastKey != null) {
146
+ s.setFocusedKey(lastKey);
147
+ }
148
+ break;
149
+ }
150
+ case 'a':
151
+ case 'A': {
152
+ if ((e.ctrlKey || e.metaKey) && s.selectionMode === 'multiple') {
153
+ e.preventDefault();
154
+ s.selectAll();
155
+ }
156
+ break;
157
+ }
158
+ case ' ':
159
+ case 'Space':
160
+ case 'Spacebar': {
161
+ if (focusedKey != null && s.selectionMode !== 'none' && !s.isDisabled(focusedKey)) {
162
+ e.preventDefault();
163
+ s.toggleSelection(focusedKey);
164
+ }
165
+ break;
166
+ }
167
+ case 'Enter': {
168
+ if (focusedKey != null && !s.isDisabled(focusedKey)) {
169
+ e.preventDefault();
170
+ p.onAction?.(focusedKey);
171
+ }
172
+ break;
173
+ }
174
+ case 'Escape': {
175
+ if (s.selectionMode !== 'none') {
176
+ e.preventDefault();
177
+ s.clearSelection();
178
+ }
179
+ break;
180
+ }
181
+ case '*': {
182
+ // Expand all siblings at current level
183
+ e.preventDefault();
184
+ if (focusedKey != null) {
185
+ const node = collection.getItem(focusedKey);
186
+ if (node) {
187
+ // Find all siblings at the same level
188
+ const parentKey = node.parentKey;
189
+ let siblings: Key[];
190
+ if (parentKey != null) {
191
+ siblings = [...collection.getChildren(parentKey)].map((n) => n.key);
192
+ } else {
193
+ // Root level siblings
194
+ siblings = collection.rows
195
+ .filter((n) => n.level === 0)
196
+ .map((n) => n.key);
197
+ }
198
+ // Expand all expandable siblings
199
+ for (const siblingKey of siblings) {
200
+ const sibling = collection.getItem(siblingKey);
201
+ if (sibling?.isExpandable && !s.isExpanded(siblingKey)) {
202
+ s.expandKey(siblingKey);
203
+ }
204
+ }
205
+ }
206
+ }
207
+ break;
208
+ }
209
+ }
210
+ };
211
+
212
+ const onFocus = () => {
213
+ const s = state();
214
+ s.setFocused(true);
215
+
216
+ // If nothing is focused, focus the first item
217
+ if (s.focusedKey == null) {
218
+ const firstKey = s.collection.getFirstKey();
219
+ if (firstKey != null) {
220
+ s.setFocusedKey(firstKey);
221
+ }
222
+ }
223
+ };
224
+
225
+ const onBlur = () => {
226
+ const s = state();
227
+ s.setFocused(false);
228
+ };
229
+
230
+ const treeProps = createMemo(() => {
231
+ const p = props();
232
+ const s = state();
233
+
234
+ const baseProps: Record<string, unknown> = {
235
+ role: 'treegrid',
236
+ id: treeId,
237
+ 'aria-label': p['aria-label'],
238
+ 'aria-labelledby': p['aria-labelledby'],
239
+ 'aria-describedby': p['aria-describedby'],
240
+ 'aria-multiselectable': s.selectionMode === 'multiple' ? true : undefined,
241
+ 'aria-disabled': p.isDisabled || undefined,
242
+ tabIndex: p.isDisabled ? undefined : 0,
243
+ onKeyDown,
244
+ onFocus,
245
+ onBlur,
246
+ };
247
+
248
+ // Add row count for virtualized trees
249
+ if (p.isVirtualized) {
250
+ baseProps['aria-rowcount'] = s.collection.rowCount;
251
+ }
252
+
253
+ return baseProps as JSX.HTMLAttributes<HTMLDivElement>;
254
+ });
255
+
256
+ return {
257
+ get treeProps() {
258
+ return treeProps();
259
+ },
260
+ };
261
+ }