@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,233 +1,282 @@
1
- /**
2
- * createTreeItem - Provides accessibility for a tree item.
3
- * Based on @react-aria/tree/useTreeItem.
4
- */
5
-
6
- import { createMemo, createSignal, type Accessor } from 'solid-js';
7
- import type { JSX } from 'solid-js';
8
- import type { TreeState, TreeCollection } from '@proyecto-viviana/solid-stately';
9
- import type { AriaTreeItemProps, TreeItemAria } from './types';
10
- import { getTreeData } from './createTree';
11
-
12
- /**
13
- * Creates accessibility props for a tree item.
14
- */
15
- export function createTreeItem<T extends object, C extends TreeCollection<T> = TreeCollection<T>>(
16
- props: Accessor<AriaTreeItemProps<T>>,
17
- state: Accessor<TreeState<T, C>>,
18
- _ref: Accessor<HTMLDivElement | null>
19
- ): TreeItemAria {
20
- const [isPressed, setIsPressed] = createSignal(false);
21
-
22
- const isSelected = createMemo(() => {
23
- const s = state();
24
- const p = props();
25
- return s.isSelected(p.node.key);
26
- });
27
-
28
- const isDisabled = createMemo(() => {
29
- const s = state();
30
- const p = props();
31
- return s.isDisabled(p.node.key);
32
- });
33
-
34
- const isFocused = createMemo(() => {
35
- const s = state();
36
- const p = props();
37
- return s.focusedKey === p.node.key;
38
- });
39
-
40
- const isExpanded = createMemo(() => {
41
- const s = state();
42
- const p = props();
43
- return s.isExpanded(p.node.key);
44
- });
45
-
46
- const isExpandable = createMemo(() => {
47
- const p = props();
48
- return p.node.isExpandable ?? false;
49
- });
50
-
51
- const level = createMemo(() => {
52
- const p = props();
53
- return p.node.level;
54
- });
55
-
56
- // Handle click/press for selection and actions
57
- const onClick = (e: MouseEvent) => {
58
- const s = state();
59
- const p = props();
60
-
61
- if (isDisabled()) return;
62
-
63
- // Get tree metadata for actions
64
- const treeData = getTreeData(s);
65
- const onAction = treeData?.actions.onAction;
66
-
67
- // Handle selection
68
- if (s.selectionMode !== 'none') {
69
- if (e.shiftKey && s.selectionMode === 'multiple') {
70
- s.extendSelection(p.node.key);
71
- } else if (e.ctrlKey || e.metaKey) {
72
- s.toggleSelection(p.node.key);
73
- } else {
74
- // Replace selection or toggle if already selected
75
- if (isSelected() && s.selectedKeys !== 'all') {
76
- const selectedKeys = s.selectedKeys as Set<unknown>;
77
- if (selectedKeys.size === 1) {
78
- // Single selection, trigger action
79
- if (onAction) {
80
- onAction(p.node.key);
81
- }
82
- if (p.onAction) {
83
- p.onAction();
84
- }
85
- } else {
86
- s.replaceSelection(p.node.key);
87
- }
88
- } else {
89
- s.replaceSelection(p.node.key);
90
- }
91
- }
92
- } else {
93
- // No selection mode, just trigger action
94
- if (onAction) {
95
- onAction(p.node.key);
96
- }
97
- if (p.onAction) {
98
- p.onAction();
99
- }
100
- }
101
- };
102
-
103
- const onKeyDown = (e: KeyboardEvent) => {
104
- const s = state();
105
- const p = props();
106
-
107
- if (isDisabled()) return;
108
-
109
- if (e.key === 'Enter') {
110
- // Get tree metadata for actions
111
- const treeData = getTreeData(s);
112
- const onAction = treeData?.actions.onAction;
113
-
114
- if (onAction || p.onAction) {
115
- e.preventDefault();
116
-
117
- if (onAction) {
118
- onAction(p.node.key);
119
- }
120
-
121
- if (p.onAction) {
122
- p.onAction();
123
- }
124
- }
125
- } else if (e.key === ' ') {
126
- // Space toggles selection
127
- if (s.selectionMode !== 'none') {
128
- e.preventDefault();
129
- s.toggleSelection(p.node.key);
130
- }
131
- }
132
- };
133
-
134
- const onFocus = () => {
135
- const s = state();
136
- const p = props();
137
- s.setFocusedKey(p.node.key);
138
- };
139
-
140
- const onPointerDown = () => {
141
- setIsPressed(true);
142
- };
143
-
144
- const onPointerUp = () => {
145
- setIsPressed(false);
146
- };
147
-
148
- const rowProps = createMemo(() => {
149
- const s = state();
150
- const p = props();
151
- const node = p.node;
152
-
153
- const baseProps: Record<string, unknown> = {
154
- role: 'row',
155
- 'aria-selected': s.selectionMode !== 'none' ? isSelected() : undefined,
156
- 'aria-disabled': isDisabled() || undefined,
157
- 'aria-expanded': isExpandable() ? isExpanded() : undefined,
158
- 'aria-level': node.level + 1, // 1-based for ARIA
159
- tabIndex: isFocused() ? 0 : -1,
160
- onClick,
161
- onKeyDown,
162
- onFocus,
163
- onPointerDown,
164
- onPointerUp,
165
- };
166
-
167
- // Add aria-rowindex for virtualized trees
168
- if (p.isVirtualized && node.rowIndex != null) {
169
- baseProps['aria-rowindex'] = node.rowIndex + 1; // 1-based
170
- }
171
-
172
- return baseProps as JSX.HTMLAttributes<HTMLDivElement>;
173
- });
174
-
175
- const gridCellProps = createMemo(() => {
176
- return {
177
- role: 'gridcell',
178
- } as JSX.HTMLAttributes<HTMLDivElement>;
179
- });
180
-
181
- // Expand button handler
182
- const onExpandClick = (e: MouseEvent) => {
183
- e.stopPropagation(); // Don't trigger row click
184
- const s = state();
185
- const p = props();
186
-
187
- if (isDisabled()) return;
188
-
189
- s.toggleKey(p.node.key);
190
- };
191
-
192
- const expandButtonProps = createMemo(() => {
193
- const baseProps: Record<string, unknown> = {
194
- type: 'button',
195
- 'aria-label': isExpanded() ? 'Collapse' : 'Expand',
196
- onClick: onExpandClick,
197
- tabIndex: -1, // Not in tab order, use arrow keys
198
- 'aria-hidden': !isExpandable() ? true : undefined,
199
- };
200
-
201
- return baseProps as JSX.ButtonHTMLAttributes<HTMLButtonElement>;
202
- });
203
-
204
- return {
205
- get rowProps() {
206
- return rowProps();
207
- },
208
- get gridCellProps() {
209
- return gridCellProps();
210
- },
211
- get expandButtonProps() {
212
- return expandButtonProps();
213
- },
214
- get isSelected() {
215
- return isSelected();
216
- },
217
- get isDisabled() {
218
- return isDisabled();
219
- },
220
- get isPressed() {
221
- return isPressed();
222
- },
223
- get isExpanded() {
224
- return isExpanded();
225
- },
226
- get isExpandable() {
227
- return isExpandable();
228
- },
229
- get level() {
230
- return level();
231
- },
232
- };
233
- }
1
+ /**
2
+ * createTreeItem - Provides accessibility for a tree item.
3
+ * Based on @react-aria/tree/useTreeItem.
4
+ */
5
+
6
+ import { createMemo, createSignal, 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 } from '@proyecto-viviana/solid-stately';
10
+ import type { AriaTreeItemProps, TreeItemAria } from './types';
11
+ import { getTreeData } from './createTree';
12
+
13
+ /**
14
+ * Creates accessibility props for a tree item.
15
+ */
16
+ export function createTreeItem<T extends object, C extends TreeCollection<T> = TreeCollection<T>>(
17
+ props: Accessor<AriaTreeItemProps<T>>,
18
+ state: Accessor<TreeState<T, C>>,
19
+ _ref: Accessor<HTMLDivElement | null>
20
+ ): TreeItemAria {
21
+ const [isPressed, setIsPressed] = createSignal(false);
22
+ const rowId = createId();
23
+ const expandButtonId = createId();
24
+
25
+ const isSelected = createMemo(() => {
26
+ const s = state();
27
+ const p = props();
28
+ return s.isSelected(p.node.key);
29
+ });
30
+
31
+ const isDisabled = createMemo(() => {
32
+ const s = state();
33
+ const p = props();
34
+ return s.isDisabled(p.node.key);
35
+ });
36
+
37
+ const isFocused = createMemo(() => {
38
+ const s = state();
39
+ const p = props();
40
+ return s.focusedKey === p.node.key;
41
+ });
42
+
43
+ const isExpanded = createMemo(() => {
44
+ const s = state();
45
+ const p = props();
46
+ return s.isExpanded(p.node.key);
47
+ });
48
+
49
+ const isExpandable = createMemo(() => {
50
+ const p = props();
51
+ return p.node.isExpandable ?? false;
52
+ });
53
+
54
+ const level = createMemo(() => {
55
+ const p = props();
56
+ return p.node.level;
57
+ });
58
+
59
+ // Handle click/press for selection and actions
60
+ const onClick = (e: MouseEvent) => {
61
+ const s = state();
62
+ const p = props();
63
+
64
+ if (isDisabled()) return;
65
+
66
+ // Get tree metadata for actions
67
+ const treeData = getTreeData(s);
68
+ const onAction = treeData?.actions.onAction;
69
+
70
+ // Handle selection
71
+ if (s.selectionMode !== 'none') {
72
+ if (e.shiftKey && s.selectionMode === 'multiple') {
73
+ s.extendSelection(p.node.key);
74
+ } else if (e.ctrlKey || e.metaKey) {
75
+ s.toggleSelection(p.node.key);
76
+ } else {
77
+ // Replace selection or toggle if already selected
78
+ if (isSelected() && s.selectedKeys !== 'all') {
79
+ const selectedKeys = s.selectedKeys as Set<unknown>;
80
+ if (selectedKeys.size === 1) {
81
+ // Single selection, trigger action
82
+ if (onAction) {
83
+ onAction(p.node.key);
84
+ }
85
+ if (p.onAction) {
86
+ p.onAction();
87
+ }
88
+ } else {
89
+ s.replaceSelection(p.node.key);
90
+ }
91
+ } else {
92
+ s.replaceSelection(p.node.key);
93
+ }
94
+ }
95
+ } else {
96
+ // No selection mode, just trigger action
97
+ if (onAction) {
98
+ onAction(p.node.key);
99
+ }
100
+ if (p.onAction) {
101
+ p.onAction();
102
+ }
103
+ }
104
+ };
105
+
106
+ const onKeyDown = (e: KeyboardEvent) => {
107
+ const s = state();
108
+ const p = props();
109
+
110
+ if (isDisabled()) return;
111
+
112
+ if (e.key === 'Enter') {
113
+ // Get tree metadata for actions
114
+ const treeData = getTreeData(s);
115
+ const onAction = treeData?.actions.onAction;
116
+
117
+ if (onAction || p.onAction) {
118
+ e.preventDefault();
119
+
120
+ if (onAction) {
121
+ onAction(p.node.key);
122
+ }
123
+
124
+ if (p.onAction) {
125
+ p.onAction();
126
+ }
127
+ }
128
+ } else if (e.key === ' ' || e.key === 'Space' || e.key === 'Spacebar') {
129
+ // Space toggles selection
130
+ if (s.selectionMode !== 'none') {
131
+ e.preventDefault();
132
+ s.toggleSelection(p.node.key);
133
+ }
134
+ }
135
+ };
136
+
137
+ const onFocus = () => {
138
+ const s = state();
139
+ const p = props();
140
+ s.setFocusedKey(p.node.key);
141
+ };
142
+
143
+ const onPointerDown = () => {
144
+ setIsPressed(true);
145
+ };
146
+
147
+ const onPointerUp = () => {
148
+ setIsPressed(false);
149
+ };
150
+
151
+ // Compute sibling position (aria-posinset/aria-setsize)
152
+ const siblingInfo = createMemo(() => {
153
+ const s = state();
154
+ const p = props();
155
+ const node = p.node;
156
+ const parentKey = node.parentKey;
157
+
158
+ if (parentKey != null) {
159
+ const parentNode = s.collection.getItem(parentKey);
160
+ if (parentNode) {
161
+ return {
162
+ posinset: node.index + 1, // 1-based
163
+ setsize: parentNode.childNodes.length,
164
+ };
165
+ }
166
+ }
167
+
168
+ // Root-level: count root nodes
169
+ const rootNodes = s.collection.rows.filter((n) => n.level === 0);
170
+ const rootIndex = rootNodes.findIndex((n) => n.key === node.key);
171
+ return {
172
+ posinset: rootIndex >= 0 ? rootIndex + 1 : node.index + 1,
173
+ setsize: rootNodes.length,
174
+ };
175
+ });
176
+
177
+ const rowProps = createMemo(() => {
178
+ const s = state();
179
+ const p = props();
180
+ const node = p.node;
181
+ const { posinset, setsize } = siblingInfo();
182
+
183
+ // Use textValue for aria-label (if available), or explicit textValue prop
184
+ const textValue = p.textValue ?? node.textValue;
185
+
186
+ const baseProps: Record<string, unknown> = {
187
+ role: 'row',
188
+ id: rowId,
189
+ 'aria-label': textValue || undefined,
190
+ 'aria-selected': s.selectionMode !== 'none' ? isSelected() : undefined,
191
+ 'aria-disabled': isDisabled() || undefined,
192
+ 'aria-expanded': isExpandable() ? isExpanded() : undefined,
193
+ 'aria-level': node.level + 1, // 1-based for ARIA
194
+ 'aria-posinset': posinset,
195
+ 'aria-setsize': setsize,
196
+ tabIndex: isFocused() ? 0 : -1,
197
+ onClick,
198
+ onKeyDown,
199
+ onFocus,
200
+ onPointerDown,
201
+ onPointerUp,
202
+ };
203
+
204
+ // Add aria-rowindex for virtualized trees
205
+ if (p.isVirtualized && node.rowIndex != null) {
206
+ baseProps['aria-rowindex'] = node.rowIndex + 1; // 1-based
207
+ }
208
+
209
+ return baseProps as JSX.HTMLAttributes<HTMLDivElement>;
210
+ });
211
+
212
+ const gridCellProps = createMemo(() => {
213
+ return {
214
+ role: 'gridcell',
215
+ } as JSX.HTMLAttributes<HTMLDivElement>;
216
+ });
217
+
218
+ // Expand button handler
219
+ const onExpandClick = (e: MouseEvent) => {
220
+ e.stopPropagation(); // Don't trigger row click
221
+ const s = state();
222
+ const p = props();
223
+
224
+ if (isDisabled()) return;
225
+
226
+ s.toggleKey(p.node.key);
227
+ };
228
+
229
+ const stopPointerPropagation = (e: Event) => {
230
+ // Prevent row pointer handlers from flipping pressed state and re-rendering
231
+ // before the button click handler can run.
232
+ e.stopPropagation();
233
+ };
234
+
235
+ const expandButtonProps = createMemo(() => {
236
+ const baseProps: Record<string, unknown> = {
237
+ type: 'button',
238
+ id: expandButtonId,
239
+ 'aria-label': isExpanded() ? 'Collapse' : 'Expand',
240
+ 'aria-labelledby': isExpandable() ? `${expandButtonId} ${rowId}` : undefined,
241
+ onClick: onExpandClick,
242
+ onPointerDown: stopPointerPropagation,
243
+ onPointerUp: stopPointerPropagation,
244
+ onMouseDown: stopPointerPropagation,
245
+ onMouseUp: stopPointerPropagation,
246
+ tabIndex: -1, // Not in tab order, use arrow keys
247
+ 'aria-hidden': !isExpandable() ? true : undefined,
248
+ };
249
+
250
+ return baseProps as JSX.ButtonHTMLAttributes<HTMLButtonElement>;
251
+ });
252
+
253
+ return {
254
+ get rowProps() {
255
+ return rowProps();
256
+ },
257
+ get gridCellProps() {
258
+ return gridCellProps();
259
+ },
260
+ get expandButtonProps() {
261
+ return expandButtonProps();
262
+ },
263
+ get isSelected() {
264
+ return isSelected();
265
+ },
266
+ get isDisabled() {
267
+ return isDisabled();
268
+ },
269
+ get isPressed() {
270
+ return isPressed();
271
+ },
272
+ get isExpanded() {
273
+ return isExpanded();
274
+ },
275
+ get isExpandable() {
276
+ return isExpandable();
277
+ },
278
+ get level() {
279
+ return level();
280
+ },
281
+ };
282
+ }
@@ -1,68 +1,68 @@
1
- /**
2
- * createTreeSelectionCheckbox - Provides accessibility for a tree item's selection checkbox.
3
- * Based on @react-aria/gridlist/useGridListSelectionCheckbox.
4
- */
5
-
6
- import { createMemo, type Accessor } from 'solid-js';
7
- import type { JSX } from 'solid-js';
8
- import type { TreeState, TreeCollection } from '@proyecto-viviana/solid-stately';
9
- import type { AriaTreeSelectionCheckboxProps, TreeSelectionCheckboxAria } from './types';
10
-
11
- /**
12
- * Creates accessibility props for a tree selection checkbox.
13
- */
14
- export function createTreeSelectionCheckbox<T extends object, C extends TreeCollection<T> = TreeCollection<T>>(
15
- props: Accessor<AriaTreeSelectionCheckboxProps>,
16
- state: Accessor<TreeState<T, C>>
17
- ): TreeSelectionCheckboxAria {
18
- const isSelected = createMemo(() => {
19
- const s = state();
20
- const p = props();
21
- return s.isSelected(p.key);
22
- });
23
-
24
- const isDisabled = createMemo(() => {
25
- const s = state();
26
- const p = props();
27
- return s.isDisabled(p.key);
28
- });
29
-
30
- const onChange = (e: Event) => {
31
- const s = state();
32
- const p = props();
33
- const target = e.target as HTMLInputElement;
34
-
35
- if (isDisabled()) return;
36
-
37
- if (target.checked) {
38
- s.toggleSelection(p.key);
39
- } else {
40
- s.toggleSelection(p.key);
41
- }
42
- };
43
-
44
- const onClick = (e: MouseEvent) => {
45
- // Stop propagation to prevent row click from also firing
46
- e.stopPropagation();
47
- };
48
-
49
- const checkboxProps = createMemo(() => {
50
- const baseProps: Record<string, unknown> = {
51
- type: 'checkbox',
52
- 'aria-label': 'Select',
53
- checked: isSelected(),
54
- disabled: isDisabled(),
55
- onChange,
56
- onClick,
57
- tabIndex: -1, // Use arrow keys to navigate, not tab
58
- };
59
-
60
- return baseProps as JSX.InputHTMLAttributes<HTMLInputElement>;
61
- });
62
-
63
- return {
64
- get checkboxProps() {
65
- return checkboxProps();
66
- },
67
- };
68
- }
1
+ /**
2
+ * createTreeSelectionCheckbox - Provides accessibility for a tree item's selection checkbox.
3
+ * Based on @react-aria/gridlist/useGridListSelectionCheckbox.
4
+ */
5
+
6
+ import { createMemo, type Accessor } from 'solid-js';
7
+ import type { JSX } from 'solid-js';
8
+ import type { TreeState, TreeCollection } from '@proyecto-viviana/solid-stately';
9
+ import type { AriaTreeSelectionCheckboxProps, TreeSelectionCheckboxAria } from './types';
10
+
11
+ /**
12
+ * Creates accessibility props for a tree selection checkbox.
13
+ */
14
+ export function createTreeSelectionCheckbox<T extends object, C extends TreeCollection<T> = TreeCollection<T>>(
15
+ props: Accessor<AriaTreeSelectionCheckboxProps>,
16
+ state: Accessor<TreeState<T, C>>
17
+ ): TreeSelectionCheckboxAria {
18
+ const isSelected = createMemo(() => {
19
+ const s = state();
20
+ const p = props();
21
+ return s.isSelected(p.key);
22
+ });
23
+
24
+ const isDisabled = createMemo(() => {
25
+ const s = state();
26
+ const p = props();
27
+ return s.isDisabled(p.key);
28
+ });
29
+
30
+ const onChange = (e: Event) => {
31
+ const s = state();
32
+ const p = props();
33
+ const target = e.target as HTMLInputElement;
34
+
35
+ if (isDisabled()) return;
36
+
37
+ if (target.checked) {
38
+ s.toggleSelection(p.key);
39
+ } else {
40
+ s.toggleSelection(p.key);
41
+ }
42
+ };
43
+
44
+ const onClick = (e: MouseEvent) => {
45
+ // Stop propagation to prevent row click from also firing
46
+ e.stopPropagation();
47
+ };
48
+
49
+ const checkboxProps = createMemo(() => {
50
+ const baseProps: Record<string, unknown> = {
51
+ type: 'checkbox',
52
+ 'aria-label': 'Select',
53
+ checked: isSelected(),
54
+ disabled: isDisabled(),
55
+ onChange,
56
+ onClick,
57
+ tabIndex: -1, // Use arrow keys to navigate, not tab
58
+ };
59
+
60
+ return baseProps as JSX.InputHTMLAttributes<HTMLInputElement>;
61
+ });
62
+
63
+ return {
64
+ get checkboxProps() {
65
+ return checkboxProps();
66
+ },
67
+ };
68
+ }