@proyecto-viviana/solidaria 0.2.2 → 0.2.3

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 (210) hide show
  1. package/dist/autocomplete/createAutocomplete.d.ts +2 -2
  2. package/dist/autocomplete/createAutocomplete.d.ts.map +1 -1
  3. package/dist/index.js +233 -234
  4. package/dist/index.js.map +2 -2
  5. package/dist/index.ssr.js +233 -234
  6. package/dist/index.ssr.js.map +2 -2
  7. package/dist/interactions/PressEvent.d.ts +13 -10
  8. package/dist/interactions/PressEvent.d.ts.map +1 -1
  9. package/dist/interactions/createPress.d.ts.map +1 -1
  10. package/dist/interactions/index.d.ts +1 -1
  11. package/dist/interactions/index.d.ts.map +1 -1
  12. package/dist/select/createHiddenSelect.d.ts.map +1 -1
  13. package/dist/toolbar/createToolbar.d.ts.map +1 -1
  14. package/dist/tooltip/createTooltipTrigger.d.ts.map +1 -1
  15. package/package.json +9 -7
  16. package/src/autocomplete/createAutocomplete.ts +341 -0
  17. package/src/autocomplete/index.ts +9 -0
  18. package/src/breadcrumbs/createBreadcrumbs.ts +196 -0
  19. package/src/breadcrumbs/index.ts +8 -0
  20. package/src/button/createButton.ts +142 -0
  21. package/src/button/createToggleButton.ts +101 -0
  22. package/src/button/index.ts +4 -0
  23. package/src/button/types.ts +78 -0
  24. package/src/calendar/createCalendar.ts +138 -0
  25. package/src/calendar/createCalendarCell.ts +187 -0
  26. package/src/calendar/createCalendarGrid.ts +140 -0
  27. package/src/calendar/createRangeCalendar.ts +136 -0
  28. package/src/calendar/createRangeCalendarCell.ts +186 -0
  29. package/src/calendar/index.ts +34 -0
  30. package/src/checkbox/createCheckbox.ts +135 -0
  31. package/src/checkbox/createCheckboxGroup.ts +137 -0
  32. package/src/checkbox/createCheckboxGroupItem.ts +117 -0
  33. package/src/checkbox/createCheckboxGroupState.ts +193 -0
  34. package/src/checkbox/index.ts +13 -0
  35. package/src/color/createColorArea.ts +314 -0
  36. package/src/color/createColorField.ts +137 -0
  37. package/src/color/createColorSlider.ts +197 -0
  38. package/src/color/createColorSwatch.ts +40 -0
  39. package/src/color/createColorWheel.ts +208 -0
  40. package/src/color/index.ts +24 -0
  41. package/src/color/types.ts +116 -0
  42. package/src/combobox/createComboBox.ts +647 -0
  43. package/src/combobox/index.ts +6 -0
  44. package/src/combobox/intl/en-US.json +7 -0
  45. package/src/combobox/intl/es-ES.json +7 -0
  46. package/src/combobox/intl/index.ts +23 -0
  47. package/src/datepicker/createDateField.ts +154 -0
  48. package/src/datepicker/createDatePicker.ts +206 -0
  49. package/src/datepicker/createDateSegment.ts +229 -0
  50. package/src/datepicker/createTimeField.ts +154 -0
  51. package/src/datepicker/index.ts +28 -0
  52. package/src/dialog/createDialog.ts +120 -0
  53. package/src/dialog/index.ts +2 -0
  54. package/src/dialog/types.ts +19 -0
  55. package/src/disclosure/createDisclosure.ts +131 -0
  56. package/src/disclosure/createDisclosureGroup.ts +62 -0
  57. package/src/disclosure/index.ts +11 -0
  58. package/src/dnd/createDrag.ts +209 -0
  59. package/src/dnd/createDraggableCollection.ts +63 -0
  60. package/src/dnd/createDraggableItem.ts +243 -0
  61. package/src/dnd/createDrop.ts +321 -0
  62. package/src/dnd/createDroppableCollection.ts +293 -0
  63. package/src/dnd/createDroppableItem.ts +213 -0
  64. package/src/dnd/index.ts +47 -0
  65. package/src/dnd/types.ts +89 -0
  66. package/src/dnd/utils.ts +294 -0
  67. package/src/focus/FocusScope.tsx +408 -0
  68. package/src/focus/createAutoFocus.ts +321 -0
  69. package/src/focus/createFocusRestore.ts +313 -0
  70. package/src/focus/createVirtualFocus.ts +396 -0
  71. package/src/focus/index.ts +35 -0
  72. package/src/form/createFormReset.ts +51 -0
  73. package/src/form/createFormValidation.ts +224 -0
  74. package/src/form/index.ts +11 -0
  75. package/src/grid/GridKeyboardDelegate.ts +429 -0
  76. package/src/grid/createGrid.ts +261 -0
  77. package/src/grid/createGridCell.ts +182 -0
  78. package/src/grid/createGridRow.ts +153 -0
  79. package/src/grid/index.ts +18 -0
  80. package/src/grid/types.ts +133 -0
  81. package/src/gridlist/createGridList.ts +185 -0
  82. package/src/gridlist/createGridListItem.ts +180 -0
  83. package/src/gridlist/createGridListSelectionCheckbox.ts +59 -0
  84. package/src/gridlist/index.ts +16 -0
  85. package/src/gridlist/types.ts +81 -0
  86. package/src/i18n/NumberFormatter.ts +266 -0
  87. package/src/i18n/createCollator.ts +79 -0
  88. package/src/i18n/createDateFormatter.ts +83 -0
  89. package/src/i18n/createFilter.ts +131 -0
  90. package/src/i18n/createNumberFormatter.ts +52 -0
  91. package/src/i18n/createStringFormatter.ts +87 -0
  92. package/src/i18n/index.ts +40 -0
  93. package/src/i18n/locale.tsx +188 -0
  94. package/src/i18n/utils.ts +99 -0
  95. package/src/index.ts +670 -0
  96. package/src/interactions/FocusableProvider.tsx +44 -0
  97. package/src/interactions/PressEvent.ts +126 -0
  98. package/src/interactions/createFocus.ts +163 -0
  99. package/src/interactions/createFocusRing.ts +89 -0
  100. package/src/interactions/createFocusWithin.ts +206 -0
  101. package/src/interactions/createFocusable.ts +168 -0
  102. package/src/interactions/createHover.ts +254 -0
  103. package/src/interactions/createInteractionModality.ts +424 -0
  104. package/src/interactions/createKeyboard.ts +82 -0
  105. package/src/interactions/createLongPress.ts +174 -0
  106. package/src/interactions/createMove.ts +289 -0
  107. package/src/interactions/createPress.ts +834 -0
  108. package/src/interactions/index.ts +78 -0
  109. package/src/label/createField.ts +145 -0
  110. package/src/label/createLabel.ts +117 -0
  111. package/src/label/createLabels.ts +50 -0
  112. package/src/label/index.ts +19 -0
  113. package/src/landmark/createLandmark.ts +377 -0
  114. package/src/landmark/index.ts +8 -0
  115. package/src/link/createLink.ts +182 -0
  116. package/src/link/index.ts +1 -0
  117. package/src/listbox/createListBox.ts +269 -0
  118. package/src/listbox/createOption.ts +151 -0
  119. package/src/listbox/index.ts +12 -0
  120. package/src/live-announcer/announce.ts +322 -0
  121. package/src/live-announcer/index.ts +9 -0
  122. package/src/menu/createMenu.ts +396 -0
  123. package/src/menu/createMenuItem.ts +149 -0
  124. package/src/menu/createMenuTrigger.ts +88 -0
  125. package/src/menu/index.ts +18 -0
  126. package/src/meter/createMeter.ts +75 -0
  127. package/src/meter/index.ts +1 -0
  128. package/src/numberfield/createNumberField.ts +268 -0
  129. package/src/numberfield/index.ts +5 -0
  130. package/src/overlays/ariaHideOutside.ts +219 -0
  131. package/src/overlays/createInteractOutside.ts +149 -0
  132. package/src/overlays/createModal.tsx +202 -0
  133. package/src/overlays/createOverlay.ts +155 -0
  134. package/src/overlays/createOverlayTrigger.ts +85 -0
  135. package/src/overlays/createPreventScroll.ts +266 -0
  136. package/src/overlays/index.ts +44 -0
  137. package/src/popover/calculatePosition.ts +766 -0
  138. package/src/popover/createOverlayPosition.ts +356 -0
  139. package/src/popover/createPopover.ts +170 -0
  140. package/src/popover/index.ts +24 -0
  141. package/src/progress/createProgressBar.ts +128 -0
  142. package/src/progress/index.ts +5 -0
  143. package/src/radio/createRadio.ts +287 -0
  144. package/src/radio/createRadioGroup.ts +189 -0
  145. package/src/radio/createRadioGroupState.ts +201 -0
  146. package/src/radio/index.ts +23 -0
  147. package/src/searchfield/createSearchField.ts +186 -0
  148. package/src/searchfield/index.ts +2 -0
  149. package/src/select/createHiddenSelect.tsx +236 -0
  150. package/src/select/createSelect.ts +395 -0
  151. package/src/select/index.ts +14 -0
  152. package/src/selection/createTypeSelect.ts +201 -0
  153. package/src/selection/index.ts +6 -0
  154. package/src/separator/createSeparator.ts +82 -0
  155. package/src/separator/index.ts +6 -0
  156. package/src/slider/createSlider.ts +349 -0
  157. package/src/slider/index.ts +2 -0
  158. package/src/ssr/index.tsx +370 -0
  159. package/src/switch/createSwitch.ts +70 -0
  160. package/src/switch/index.ts +1 -0
  161. package/src/table/createTable.ts +526 -0
  162. package/src/table/createTableCell.ts +147 -0
  163. package/src/table/createTableColumnHeader.ts +115 -0
  164. package/src/table/createTableHeaderRow.ts +40 -0
  165. package/src/table/createTableRow.ts +155 -0
  166. package/src/table/createTableRowGroup.ts +32 -0
  167. package/src/table/createTableSelectAllCheckbox.ts +73 -0
  168. package/src/table/createTableSelectionCheckbox.ts +59 -0
  169. package/src/table/index.ts +30 -0
  170. package/src/table/types.ts +165 -0
  171. package/src/tabs/createTabs.ts +472 -0
  172. package/src/tabs/index.ts +14 -0
  173. package/src/tag/createTag.ts +194 -0
  174. package/src/tag/createTagGroup.ts +154 -0
  175. package/src/tag/index.ts +12 -0
  176. package/src/textfield/createTextField.ts +198 -0
  177. package/src/textfield/index.ts +5 -0
  178. package/src/toast/createToast.ts +118 -0
  179. package/src/toast/createToastRegion.ts +100 -0
  180. package/src/toast/index.ts +11 -0
  181. package/src/toggle/createToggle.ts +223 -0
  182. package/src/toggle/createToggleState.ts +94 -0
  183. package/src/toggle/index.ts +7 -0
  184. package/src/toolbar/createToolbar.ts +369 -0
  185. package/src/toolbar/index.ts +6 -0
  186. package/src/tooltip/createTooltip.ts +79 -0
  187. package/src/tooltip/createTooltipTrigger.ts +222 -0
  188. package/src/tooltip/index.ts +6 -0
  189. package/src/tree/createTree.ts +246 -0
  190. package/src/tree/createTreeItem.ts +233 -0
  191. package/src/tree/createTreeSelectionCheckbox.ts +68 -0
  192. package/src/tree/index.ts +16 -0
  193. package/src/tree/types.ts +87 -0
  194. package/src/utils/createDescription.ts +137 -0
  195. package/src/utils/dom.ts +327 -0
  196. package/src/utils/env.ts +54 -0
  197. package/src/utils/events.ts +106 -0
  198. package/src/utils/filterDOMProps.ts +116 -0
  199. package/src/utils/focus.ts +151 -0
  200. package/src/utils/geometry.ts +115 -0
  201. package/src/utils/globalListeners.ts +142 -0
  202. package/src/utils/index.ts +80 -0
  203. package/src/utils/mergeProps.ts +52 -0
  204. package/src/utils/platform.ts +52 -0
  205. package/src/utils/reactivity.ts +36 -0
  206. package/src/utils/textSelection.ts +114 -0
  207. package/src/visually-hidden/createVisuallyHidden.ts +124 -0
  208. package/src/visually-hidden/index.ts +6 -0
  209. package/dist/index.jsx +0 -15845
  210. package/dist/index.jsx.map +0 -7
@@ -0,0 +1,396 @@
1
+ /**
2
+ * Provides the behavior and accessibility implementation for a menu component.
3
+ * A menu displays a list of actions or options that a user can choose.
4
+ * Based on @react-aria/menu useMenu.
5
+ */
6
+
7
+ import { createEffect, onCleanup, type JSX, type Accessor } from 'solid-js';
8
+ import { createFocusWithin } from '../interactions/createFocusWithin';
9
+ import { createLabel } from '../label/createLabel';
10
+ import { createTypeSelect } from '../selection/createTypeSelect';
11
+ import { filterDOMProps } from '../utils/filterDOMProps';
12
+ import { mergeProps } from '../utils/mergeProps';
13
+ import { createId } from '../ssr';
14
+ import { access, type MaybeAccessor } from '../utils/reactivity';
15
+ import { isDevEnv } from '../utils/env';
16
+ import type { MenuState, Key, Collection } from '@proyecto-viviana/solid-stately';
17
+
18
+ /**
19
+ * Default number of items to skip for page up/down when DOM measurement is not available.
20
+ */
21
+ const DEFAULT_PAGE_SIZE = 10;
22
+
23
+ /**
24
+ * Find the next non-disabled key in a collection.
25
+ */
26
+ function findNextNonDisabledKey<T>(
27
+ collection: Collection<T>,
28
+ currentKey: Key | null,
29
+ direction: 'next' | 'prev',
30
+ isDisabled: (key: Key) => boolean,
31
+ wrap: boolean
32
+ ): Key | null {
33
+ const getNextKey = direction === 'next'
34
+ ? (key: Key) => collection.getKeyAfter(key)
35
+ : (key: Key) => collection.getKeyBefore(key);
36
+
37
+ const getFirstKey = direction === 'next'
38
+ ? () => collection.getFirstKey()
39
+ : () => collection.getLastKey();
40
+
41
+ let nextKey = currentKey != null ? getNextKey(currentKey) : getFirstKey();
42
+
43
+ // Skip disabled keys
44
+ while (nextKey != null && isDisabled(nextKey)) {
45
+ nextKey = getNextKey(nextKey);
46
+ }
47
+
48
+ // If we've reached the end and wrapping is enabled
49
+ if (nextKey == null && wrap) {
50
+ nextKey = getFirstKey();
51
+ // Skip disabled keys from the start
52
+ while (nextKey != null && isDisabled(nextKey)) {
53
+ nextKey = getNextKey(nextKey);
54
+ }
55
+ }
56
+
57
+ return nextKey;
58
+ }
59
+
60
+ export interface AriaMenuProps {
61
+ /** An ID for the menu. */
62
+ id?: string;
63
+ /** Whether the menu is disabled. */
64
+ isDisabled?: boolean;
65
+ /** The label for the menu. */
66
+ label?: JSX.Element;
67
+ /** An accessible label for the menu when no visible label is provided. */
68
+ 'aria-label'?: string;
69
+ /** The ID of an element that labels the menu. */
70
+ 'aria-labelledby'?: string;
71
+ /** The ID of an element that describes the menu. */
72
+ 'aria-describedby'?: string;
73
+ /** Handler called when focus moves into the menu. */
74
+ onFocus?: (e: FocusEvent) => void;
75
+ /** Handler called when focus moves out of the menu. */
76
+ onBlur?: (e: FocusEvent) => void;
77
+ /** Handler called when the focus state changes. */
78
+ onFocusChange?: (isFocused: boolean) => void;
79
+ /** Handler called when an item is activated (pressed). */
80
+ onAction?: (key: Key) => void;
81
+ /** Handler called when the menu should close. */
82
+ onClose?: () => void;
83
+ /** Whether focus should automatically wrap around. */
84
+ shouldFocusWrap?: boolean;
85
+ /** Whether to auto-focus the first item when the menu opens. */
86
+ autoFocus?: boolean | 'first' | 'last';
87
+ /** Whether type-to-select is disabled. @default false */
88
+ disallowTypeAhead?: boolean;
89
+ }
90
+
91
+ export interface MenuAria {
92
+ /** Props for the menu element. */
93
+ menuProps: JSX.HTMLAttributes<HTMLElement>;
94
+ /** Props for the menu's label element (if any). */
95
+ labelProps: JSX.HTMLAttributes<HTMLElement>;
96
+ }
97
+
98
+ // Shared data between menu and menu items
99
+ const menuData = new WeakMap<object, MenuData>();
100
+
101
+ interface MenuData {
102
+ id: string;
103
+ onAction?: (key: Key) => void;
104
+ onClose?: () => void;
105
+ }
106
+
107
+ export function getMenuData(state: MenuState): MenuData | undefined {
108
+ return menuData.get(state);
109
+ }
110
+
111
+ /**
112
+ * Provides the behavior and accessibility implementation for a menu component.
113
+ * A menu displays a list of actions or options that a user can choose.
114
+ */
115
+ export function createMenu<T>(
116
+ props: MaybeAccessor<AriaMenuProps>,
117
+ state: MenuState<T>,
118
+ ref?: Accessor<HTMLElement | null>
119
+ ): MenuAria {
120
+ const getProps = () => access(props);
121
+ const id = createId(getProps().id);
122
+
123
+ // Development-time warning for missing accessibility labels
124
+ if (isDevEnv()) {
125
+ const p = getProps();
126
+ if (!p.label && !p['aria-label'] && !p['aria-labelledby']) {
127
+ console.warn(
128
+ '[solidaria] A Menu requires an aria-label or aria-labelledby attribute for accessibility.'
129
+ );
130
+ }
131
+ }
132
+
133
+ // Filter DOM props
134
+ const domProps = () => filterDOMProps(getProps() as unknown as Record<string, unknown>, { labelable: true });
135
+
136
+ // Share data with child menu items
137
+ createEffect(() => {
138
+ const p = getProps();
139
+ menuData.set(state, {
140
+ id,
141
+ onAction: p.onAction,
142
+ onClose: p.onClose,
143
+ });
144
+
145
+ onCleanup(() => {
146
+ menuData.delete(state);
147
+ });
148
+ });
149
+
150
+ // Handle focus within
151
+ const { focusWithinProps } = createFocusWithin({
152
+ onFocusWithin: (e) => getProps().onFocus?.(e),
153
+ onBlurWithin: (e) => getProps().onBlur?.(e),
154
+ onFocusWithinChange: (isFocused) => {
155
+ getProps().onFocusChange?.(isFocused);
156
+ state.setFocused(isFocused);
157
+ },
158
+ });
159
+
160
+ // Label handling
161
+ const { labelProps, fieldProps } = createLabel({
162
+ get id() {
163
+ return id;
164
+ },
165
+ get label() {
166
+ return getProps().label;
167
+ },
168
+ get 'aria-label'() {
169
+ return getProps()['aria-label'];
170
+ },
171
+ get 'aria-labelledby'() {
172
+ return getProps()['aria-labelledby'];
173
+ },
174
+ labelElementType: 'span',
175
+ });
176
+
177
+ // Type-to-select
178
+ const { typeSelectProps } = createTypeSelect({
179
+ collection: () => state.collection(),
180
+ focusedKey: () => state.focusedKey(),
181
+ onFocusedKeyChange: (key) => state.setFocusedKey(key),
182
+ isKeyDisabled: (key) => state.isDisabled(key),
183
+ get isDisabled() {
184
+ return getProps().disallowTypeAhead ?? false;
185
+ },
186
+ });
187
+
188
+ // Keyboard navigation
189
+ const onKeyDown: JSX.EventHandler<HTMLElement, KeyboardEvent> = (e) => {
190
+ if (getProps().isDisabled) return;
191
+
192
+ const collection = state.collection();
193
+ const p = getProps();
194
+ const wrap = p.shouldFocusWrap ?? false;
195
+
196
+ // Use state.isDisabled which properly checks the disabledKeys accessor
197
+ const isDisabled = (key: Key) => state.isDisabled(key);
198
+
199
+ switch (e.key) {
200
+ case 'ArrowDown': {
201
+ e.preventDefault();
202
+ const currentKey = state.focusedKey();
203
+ const nextKey = findNextNonDisabledKey(collection, currentKey, 'next', isDisabled, wrap);
204
+ if (nextKey != null) {
205
+ state.setFocusedKey(nextKey);
206
+ }
207
+ break;
208
+ }
209
+ case 'ArrowUp': {
210
+ e.preventDefault();
211
+ const currentKey = state.focusedKey();
212
+ const prevKey = findNextNonDisabledKey(collection, currentKey, 'prev', isDisabled, wrap);
213
+ if (prevKey != null) {
214
+ state.setFocusedKey(prevKey);
215
+ }
216
+ break;
217
+ }
218
+ case 'Home': {
219
+ e.preventDefault();
220
+ // Find first non-disabled key
221
+ let firstKey = collection.getFirstKey();
222
+ while (firstKey != null && isDisabled(firstKey)) {
223
+ firstKey = collection.getKeyAfter(firstKey);
224
+ }
225
+ if (firstKey != null) {
226
+ state.setFocusedKey(firstKey);
227
+ }
228
+ break;
229
+ }
230
+ case 'End': {
231
+ e.preventDefault();
232
+ // Find last non-disabled key
233
+ let lastKey = collection.getLastKey();
234
+ while (lastKey != null && isDisabled(lastKey)) {
235
+ lastKey = collection.getKeyBefore(lastKey);
236
+ }
237
+ if (lastKey != null) {
238
+ state.setFocusedKey(lastKey);
239
+ }
240
+ break;
241
+ }
242
+ case ' ':
243
+ case 'Enter': {
244
+ e.preventDefault();
245
+ const focusedKey = state.focusedKey();
246
+ // Don't activate disabled items
247
+ if (focusedKey != null && !isDisabled(focusedKey)) {
248
+ p.onAction?.(focusedKey);
249
+ p.onClose?.();
250
+ }
251
+ break;
252
+ }
253
+ case 'Escape': {
254
+ e.preventDefault();
255
+ p.onClose?.();
256
+ break;
257
+ }
258
+ case 'PageDown': {
259
+ e.preventDefault();
260
+ const currentKey = state.focusedKey();
261
+ const el = ref?.();
262
+
263
+ if (el) {
264
+ // Use DOM measurements to calculate how many items fit in a page
265
+ const visibleHeight = el.clientHeight;
266
+ let traveled = 0;
267
+ let targetKey = currentKey;
268
+
269
+ while (targetKey != null && traveled < visibleHeight) {
270
+ const nextKey = collection.getKeyAfter(targetKey);
271
+ if (nextKey == null) break;
272
+
273
+ // Try to measure the item height
274
+ const itemElement = el.querySelector(`[data-key="${targetKey}"]`);
275
+ traveled += itemElement?.clientHeight ?? 32;
276
+
277
+ // Skip disabled items
278
+ if (!isDisabled(nextKey)) {
279
+ targetKey = nextKey;
280
+ } else {
281
+ // Skip over disabled items without counting them
282
+ const afterDisabled = findNextNonDisabledKey(collection, nextKey, 'next', isDisabled, false);
283
+ if (afterDisabled != null) {
284
+ targetKey = afterDisabled;
285
+ } else {
286
+ break;
287
+ }
288
+ }
289
+ }
290
+
291
+ if (targetKey != null && targetKey !== currentKey) {
292
+ state.setFocusedKey(targetKey);
293
+ }
294
+ } else {
295
+ // Fallback: move by DEFAULT_PAGE_SIZE items
296
+ let count = DEFAULT_PAGE_SIZE;
297
+ let targetKey = currentKey;
298
+
299
+ while (count > 0 && targetKey != null) {
300
+ const nextKey = findNextNonDisabledKey(collection, targetKey, 'next', isDisabled, false);
301
+ if (nextKey == null) break;
302
+ targetKey = nextKey;
303
+ count--;
304
+ }
305
+
306
+ if (targetKey != null) {
307
+ state.setFocusedKey(targetKey);
308
+ }
309
+ }
310
+ break;
311
+ }
312
+ case 'PageUp': {
313
+ e.preventDefault();
314
+ const currentKey = state.focusedKey();
315
+ const el = ref?.();
316
+
317
+ if (el) {
318
+ // Use DOM measurements to calculate how many items fit in a page
319
+ const visibleHeight = el.clientHeight;
320
+ let traveled = 0;
321
+ let targetKey = currentKey;
322
+
323
+ while (targetKey != null && traveled < visibleHeight) {
324
+ const prevKey = collection.getKeyBefore(targetKey);
325
+ if (prevKey == null) break;
326
+
327
+ // Try to measure the item height
328
+ const itemElement = el.querySelector(`[data-key="${targetKey}"]`);
329
+ traveled += itemElement?.clientHeight ?? 32;
330
+
331
+ // Skip disabled items
332
+ if (!isDisabled(prevKey)) {
333
+ targetKey = prevKey;
334
+ } else {
335
+ // Skip over disabled items without counting them
336
+ const beforeDisabled = findNextNonDisabledKey(collection, prevKey, 'prev', isDisabled, false);
337
+ if (beforeDisabled != null) {
338
+ targetKey = beforeDisabled;
339
+ } else {
340
+ break;
341
+ }
342
+ }
343
+ }
344
+
345
+ if (targetKey != null && targetKey !== currentKey) {
346
+ state.setFocusedKey(targetKey);
347
+ }
348
+ } else {
349
+ // Fallback: move by DEFAULT_PAGE_SIZE items
350
+ let count = DEFAULT_PAGE_SIZE;
351
+ let targetKey = currentKey;
352
+
353
+ while (count > 0 && targetKey != null) {
354
+ const prevKey = findNextNonDisabledKey(collection, targetKey, 'prev', isDisabled, false);
355
+ if (prevKey == null) break;
356
+ targetKey = prevKey;
357
+ count--;
358
+ }
359
+
360
+ if (targetKey != null) {
361
+ state.setFocusedKey(targetKey);
362
+ }
363
+ }
364
+ break;
365
+ }
366
+ }
367
+ };
368
+
369
+ return {
370
+ get labelProps() {
371
+ return labelProps as JSX.HTMLAttributes<HTMLElement>;
372
+ },
373
+ get menuProps() {
374
+ const p = getProps();
375
+
376
+ const baseProps = mergeProps(
377
+ domProps(),
378
+ focusWithinProps as Record<string, unknown>,
379
+ fieldProps as Record<string, unknown>,
380
+ {
381
+ role: 'menu',
382
+ tabIndex: p.isDisabled ? undefined : 0,
383
+ 'aria-disabled': p.isDisabled || undefined,
384
+ onKeyDown,
385
+ } as Record<string, unknown>
386
+ );
387
+
388
+ // Add type-select props if enabled
389
+ if (!p.disallowTypeAhead) {
390
+ return mergeProps(baseProps, typeSelectProps as Record<string, unknown>) as JSX.HTMLAttributes<HTMLElement>;
391
+ }
392
+
393
+ return baseProps as JSX.HTMLAttributes<HTMLElement>;
394
+ },
395
+ };
396
+ }
@@ -0,0 +1,149 @@
1
+ /**
2
+ * Provides the behavior and accessibility implementation for a menu item.
3
+ * Based on @react-aria/menu useMenuItem.
4
+ */
5
+
6
+ import { type JSX, type Accessor } from 'solid-js';
7
+ import { createPress } from '../interactions/createPress';
8
+ import { createHover } from '../interactions/createHover';
9
+ import { createFocusRing } from '../interactions/createFocusRing';
10
+ import { mergeProps } from '../utils/mergeProps';
11
+ import { access, type MaybeAccessor } from '../utils/reactivity';
12
+ import { getMenuData } from './createMenu';
13
+ import type { MenuState, Key } from '@proyecto-viviana/solid-stately';
14
+
15
+ export interface AriaMenuItemProps {
16
+ /** The unique key for the menu item. */
17
+ key: Key;
18
+ /** Whether the menu item is disabled. */
19
+ isDisabled?: boolean;
20
+ /** An accessible label for the menu item. */
21
+ 'aria-label'?: string;
22
+ /** Handler called when the menu item is selected. */
23
+ onAction?: () => void;
24
+ /** Whether to close the menu when this item is selected. */
25
+ closeOnSelect?: boolean;
26
+ }
27
+
28
+ export interface MenuItemAria {
29
+ /** Props for the menu item element. */
30
+ menuItemProps: JSX.HTMLAttributes<HTMLElement>;
31
+ /** Props for the label text inside the menu item. */
32
+ labelProps: JSX.HTMLAttributes<HTMLElement>;
33
+ /** Props for the description text inside the menu item. */
34
+ descriptionProps: JSX.HTMLAttributes<HTMLElement>;
35
+ /** Props for the keyboard shortcut inside the menu item. */
36
+ keyboardShortcutProps: JSX.HTMLAttributes<HTMLElement>;
37
+ /** Whether the menu item is currently focused. */
38
+ isFocused: Accessor<boolean>;
39
+ /** Whether the menu item is keyboard focused. */
40
+ isFocusVisible: Accessor<boolean>;
41
+ /** Whether the menu item is currently pressed. */
42
+ isPressed: Accessor<boolean>;
43
+ /** Whether the menu item is disabled. */
44
+ isDisabled: Accessor<boolean>;
45
+ }
46
+
47
+ /**
48
+ * Provides the behavior and accessibility implementation for a menu item.
49
+ */
50
+ export function createMenuItem<T>(
51
+ props: MaybeAccessor<AriaMenuItemProps>,
52
+ state: MenuState<T>,
53
+ _ref?: () => HTMLElement | null
54
+ ): MenuItemAria {
55
+ const getProps = () => access(props);
56
+
57
+ // Get shared data from menu
58
+ const getData = () => getMenuData(state);
59
+
60
+ // Computed states
61
+ const isDisabled: Accessor<boolean> = () => {
62
+ return getProps().isDisabled ?? state.isDisabled(getProps().key);
63
+ };
64
+
65
+ const isFocused: Accessor<boolean> = () => {
66
+ return state.focusedKey() === getProps().key;
67
+ };
68
+
69
+ // Handle press
70
+ const { pressProps, isPressed } = createPress({
71
+ get isDisabled() {
72
+ return isDisabled();
73
+ },
74
+ onPress() {
75
+ const p = getProps();
76
+ const key = p.key;
77
+ const data = getData();
78
+
79
+ // Call item-specific onAction
80
+ p.onAction?.();
81
+
82
+ // Call menu-level onAction
83
+ data?.onAction?.(key);
84
+
85
+ // Close menu if closeOnSelect is not explicitly false
86
+ if (p.closeOnSelect !== false) {
87
+ data?.onClose?.();
88
+ }
89
+ },
90
+ });
91
+
92
+ // Handle hover
93
+ const { hoverProps } = createHover({
94
+ get isDisabled() {
95
+ return isDisabled();
96
+ },
97
+ onHoverStart() {
98
+ state.setFocusedKey(getProps().key);
99
+ },
100
+ });
101
+
102
+ // Handle focus ring
103
+ const { isFocusVisible, focusProps } = createFocusRing();
104
+
105
+ // Generate unique IDs for label and description
106
+ const labelId = `${getProps().key}-label`;
107
+ const descriptionId = `${getProps().key}-desc`;
108
+ const keyboardId = `${getProps().key}-kbd`;
109
+
110
+ return {
111
+ get menuItemProps() {
112
+ const key = getProps().key;
113
+ const ariaLabel = getProps()['aria-label'];
114
+
115
+ return mergeProps(
116
+ pressProps as Record<string, unknown>,
117
+ hoverProps as Record<string, unknown>,
118
+ focusProps as Record<string, unknown>,
119
+ {
120
+ role: 'menuitem',
121
+ id: String(key),
122
+ 'aria-disabled': isDisabled() || undefined,
123
+ 'aria-label': ariaLabel,
124
+ 'aria-labelledby': !ariaLabel ? labelId : undefined,
125
+ 'aria-describedby': descriptionId,
126
+ tabIndex: isFocused() ? 0 : -1,
127
+ 'data-focused': isFocused() || undefined,
128
+ 'data-focus-visible': isFocusVisible() || undefined,
129
+ 'data-pressed': isPressed() || undefined,
130
+ 'data-disabled': isDisabled() || undefined,
131
+ } as Record<string, unknown>
132
+ ) as JSX.HTMLAttributes<HTMLElement>;
133
+ },
134
+ labelProps: {
135
+ id: labelId,
136
+ },
137
+ descriptionProps: {
138
+ id: descriptionId,
139
+ },
140
+ keyboardShortcutProps: {
141
+ id: keyboardId,
142
+ 'aria-hidden': true,
143
+ },
144
+ isFocused,
145
+ isFocusVisible: () => isFocused() && isFocusVisible(),
146
+ isPressed,
147
+ isDisabled,
148
+ };
149
+ }
@@ -0,0 +1,88 @@
1
+ /**
2
+ * Provides the behavior and accessibility implementation for a menu trigger.
3
+ * Based on @react-aria/menu useMenuTrigger.
4
+ */
5
+
6
+ import { type JSX } from 'solid-js';
7
+ import { createId } from '../ssr';
8
+ import { access, type MaybeAccessor } from '../utils/reactivity';
9
+ import type { OverlayTriggerState } from '@proyecto-viviana/solid-stately';
10
+
11
+ export interface AriaMenuTriggerProps {
12
+ /** The type of menu that the menu trigger opens. */
13
+ type?: 'menu' | 'listbox';
14
+ /** Whether the menu trigger is disabled. */
15
+ isDisabled?: boolean;
16
+ /** An ID for the menu. */
17
+ id?: string;
18
+ }
19
+
20
+ export interface MenuTriggerAria {
21
+ /** Props for the menu trigger button. */
22
+ menuTriggerProps: JSX.HTMLAttributes<HTMLElement> & {
23
+ onPress: () => void;
24
+ onKeyDown: (e: KeyboardEvent) => void;
25
+ };
26
+ /** Props for the menu element. */
27
+ menuProps: JSX.HTMLAttributes<HTMLElement>;
28
+ }
29
+
30
+ /**
31
+ * Provides the behavior and accessibility implementation for a menu trigger.
32
+ */
33
+ export function createMenuTrigger(
34
+ props: MaybeAccessor<AriaMenuTriggerProps>,
35
+ state: OverlayTriggerState
36
+ ): MenuTriggerAria {
37
+ const getProps = () => access(props);
38
+ const menuId = createId(getProps().id);
39
+
40
+ const onPress = () => {
41
+ if (!getProps().isDisabled) {
42
+ state.toggle();
43
+ }
44
+ };
45
+
46
+ const onKeyDown = (e: KeyboardEvent) => {
47
+ if (getProps().isDisabled) return;
48
+
49
+ switch (e.key) {
50
+ case 'Enter':
51
+ case ' ':
52
+ case 'ArrowDown': {
53
+ e.preventDefault();
54
+ if (!state.isOpen()) {
55
+ state.open();
56
+ }
57
+ break;
58
+ }
59
+ case 'ArrowUp': {
60
+ e.preventDefault();
61
+ if (!state.isOpen()) {
62
+ state.open();
63
+ }
64
+ break;
65
+ }
66
+ }
67
+ };
68
+
69
+ return {
70
+ get menuTriggerProps() {
71
+ const p = getProps();
72
+ const type = p.type ?? 'menu';
73
+ const isOpen = state.isOpen();
74
+
75
+ return {
76
+ 'aria-haspopup': type,
77
+ 'aria-expanded': isOpen,
78
+ 'aria-controls': isOpen ? menuId : undefined,
79
+ 'aria-disabled': p.isDisabled || undefined,
80
+ onPress,
81
+ onKeyDown,
82
+ };
83
+ },
84
+ menuProps: {
85
+ id: menuId,
86
+ },
87
+ };
88
+ }
@@ -0,0 +1,18 @@
1
+ export {
2
+ createMenu,
3
+ getMenuData,
4
+ type AriaMenuProps,
5
+ type MenuAria,
6
+ } from './createMenu';
7
+
8
+ export {
9
+ createMenuItem,
10
+ type AriaMenuItemProps,
11
+ type MenuItemAria,
12
+ } from './createMenuItem';
13
+
14
+ export {
15
+ createMenuTrigger,
16
+ type AriaMenuTriggerProps,
17
+ type MenuTriggerAria,
18
+ } from './createMenuTrigger';