@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.
- package/dist/autocomplete/createAutocomplete.d.ts +2 -2
- package/dist/autocomplete/createAutocomplete.d.ts.map +1 -1
- package/dist/index.js +233 -234
- package/dist/index.js.map +2 -2
- package/dist/index.ssr.js +233 -234
- package/dist/index.ssr.js.map +2 -2
- package/dist/interactions/PressEvent.d.ts +13 -10
- package/dist/interactions/PressEvent.d.ts.map +1 -1
- package/dist/interactions/createPress.d.ts.map +1 -1
- package/dist/interactions/index.d.ts +1 -1
- package/dist/interactions/index.d.ts.map +1 -1
- package/dist/select/createHiddenSelect.d.ts.map +1 -1
- package/dist/toolbar/createToolbar.d.ts.map +1 -1
- package/dist/tooltip/createTooltipTrigger.d.ts.map +1 -1
- package/package.json +9 -7
- package/src/autocomplete/createAutocomplete.ts +341 -0
- package/src/autocomplete/index.ts +9 -0
- package/src/breadcrumbs/createBreadcrumbs.ts +196 -0
- package/src/breadcrumbs/index.ts +8 -0
- package/src/button/createButton.ts +142 -0
- package/src/button/createToggleButton.ts +101 -0
- package/src/button/index.ts +4 -0
- package/src/button/types.ts +78 -0
- package/src/calendar/createCalendar.ts +138 -0
- package/src/calendar/createCalendarCell.ts +187 -0
- package/src/calendar/createCalendarGrid.ts +140 -0
- package/src/calendar/createRangeCalendar.ts +136 -0
- package/src/calendar/createRangeCalendarCell.ts +186 -0
- package/src/calendar/index.ts +34 -0
- package/src/checkbox/createCheckbox.ts +135 -0
- package/src/checkbox/createCheckboxGroup.ts +137 -0
- package/src/checkbox/createCheckboxGroupItem.ts +117 -0
- package/src/checkbox/createCheckboxGroupState.ts +193 -0
- package/src/checkbox/index.ts +13 -0
- package/src/color/createColorArea.ts +314 -0
- package/src/color/createColorField.ts +137 -0
- package/src/color/createColorSlider.ts +197 -0
- package/src/color/createColorSwatch.ts +40 -0
- package/src/color/createColorWheel.ts +208 -0
- package/src/color/index.ts +24 -0
- package/src/color/types.ts +116 -0
- package/src/combobox/createComboBox.ts +647 -0
- package/src/combobox/index.ts +6 -0
- package/src/combobox/intl/en-US.json +7 -0
- package/src/combobox/intl/es-ES.json +7 -0
- package/src/combobox/intl/index.ts +23 -0
- package/src/datepicker/createDateField.ts +154 -0
- package/src/datepicker/createDatePicker.ts +206 -0
- package/src/datepicker/createDateSegment.ts +229 -0
- package/src/datepicker/createTimeField.ts +154 -0
- package/src/datepicker/index.ts +28 -0
- package/src/dialog/createDialog.ts +120 -0
- package/src/dialog/index.ts +2 -0
- package/src/dialog/types.ts +19 -0
- package/src/disclosure/createDisclosure.ts +131 -0
- package/src/disclosure/createDisclosureGroup.ts +62 -0
- package/src/disclosure/index.ts +11 -0
- package/src/dnd/createDrag.ts +209 -0
- package/src/dnd/createDraggableCollection.ts +63 -0
- package/src/dnd/createDraggableItem.ts +243 -0
- package/src/dnd/createDrop.ts +321 -0
- package/src/dnd/createDroppableCollection.ts +293 -0
- package/src/dnd/createDroppableItem.ts +213 -0
- package/src/dnd/index.ts +47 -0
- package/src/dnd/types.ts +89 -0
- package/src/dnd/utils.ts +294 -0
- package/src/focus/FocusScope.tsx +408 -0
- package/src/focus/createAutoFocus.ts +321 -0
- package/src/focus/createFocusRestore.ts +313 -0
- package/src/focus/createVirtualFocus.ts +396 -0
- package/src/focus/index.ts +35 -0
- package/src/form/createFormReset.ts +51 -0
- package/src/form/createFormValidation.ts +224 -0
- package/src/form/index.ts +11 -0
- package/src/grid/GridKeyboardDelegate.ts +429 -0
- package/src/grid/createGrid.ts +261 -0
- package/src/grid/createGridCell.ts +182 -0
- package/src/grid/createGridRow.ts +153 -0
- package/src/grid/index.ts +18 -0
- package/src/grid/types.ts +133 -0
- package/src/gridlist/createGridList.ts +185 -0
- package/src/gridlist/createGridListItem.ts +180 -0
- package/src/gridlist/createGridListSelectionCheckbox.ts +59 -0
- package/src/gridlist/index.ts +16 -0
- package/src/gridlist/types.ts +81 -0
- package/src/i18n/NumberFormatter.ts +266 -0
- package/src/i18n/createCollator.ts +79 -0
- package/src/i18n/createDateFormatter.ts +83 -0
- package/src/i18n/createFilter.ts +131 -0
- package/src/i18n/createNumberFormatter.ts +52 -0
- package/src/i18n/createStringFormatter.ts +87 -0
- package/src/i18n/index.ts +40 -0
- package/src/i18n/locale.tsx +188 -0
- package/src/i18n/utils.ts +99 -0
- package/src/index.ts +670 -0
- package/src/interactions/FocusableProvider.tsx +44 -0
- package/src/interactions/PressEvent.ts +126 -0
- package/src/interactions/createFocus.ts +163 -0
- package/src/interactions/createFocusRing.ts +89 -0
- package/src/interactions/createFocusWithin.ts +206 -0
- package/src/interactions/createFocusable.ts +168 -0
- package/src/interactions/createHover.ts +254 -0
- package/src/interactions/createInteractionModality.ts +424 -0
- package/src/interactions/createKeyboard.ts +82 -0
- package/src/interactions/createLongPress.ts +174 -0
- package/src/interactions/createMove.ts +289 -0
- package/src/interactions/createPress.ts +834 -0
- package/src/interactions/index.ts +78 -0
- package/src/label/createField.ts +145 -0
- package/src/label/createLabel.ts +117 -0
- package/src/label/createLabels.ts +50 -0
- package/src/label/index.ts +19 -0
- package/src/landmark/createLandmark.ts +377 -0
- package/src/landmark/index.ts +8 -0
- package/src/link/createLink.ts +182 -0
- package/src/link/index.ts +1 -0
- package/src/listbox/createListBox.ts +269 -0
- package/src/listbox/createOption.ts +151 -0
- package/src/listbox/index.ts +12 -0
- package/src/live-announcer/announce.ts +322 -0
- package/src/live-announcer/index.ts +9 -0
- package/src/menu/createMenu.ts +396 -0
- package/src/menu/createMenuItem.ts +149 -0
- package/src/menu/createMenuTrigger.ts +88 -0
- package/src/menu/index.ts +18 -0
- package/src/meter/createMeter.ts +75 -0
- package/src/meter/index.ts +1 -0
- package/src/numberfield/createNumberField.ts +268 -0
- package/src/numberfield/index.ts +5 -0
- package/src/overlays/ariaHideOutside.ts +219 -0
- package/src/overlays/createInteractOutside.ts +149 -0
- package/src/overlays/createModal.tsx +202 -0
- package/src/overlays/createOverlay.ts +155 -0
- package/src/overlays/createOverlayTrigger.ts +85 -0
- package/src/overlays/createPreventScroll.ts +266 -0
- package/src/overlays/index.ts +44 -0
- package/src/popover/calculatePosition.ts +766 -0
- package/src/popover/createOverlayPosition.ts +356 -0
- package/src/popover/createPopover.ts +170 -0
- package/src/popover/index.ts +24 -0
- package/src/progress/createProgressBar.ts +128 -0
- package/src/progress/index.ts +5 -0
- package/src/radio/createRadio.ts +287 -0
- package/src/radio/createRadioGroup.ts +189 -0
- package/src/radio/createRadioGroupState.ts +201 -0
- package/src/radio/index.ts +23 -0
- package/src/searchfield/createSearchField.ts +186 -0
- package/src/searchfield/index.ts +2 -0
- package/src/select/createHiddenSelect.tsx +236 -0
- package/src/select/createSelect.ts +395 -0
- package/src/select/index.ts +14 -0
- package/src/selection/createTypeSelect.ts +201 -0
- package/src/selection/index.ts +6 -0
- package/src/separator/createSeparator.ts +82 -0
- package/src/separator/index.ts +6 -0
- package/src/slider/createSlider.ts +349 -0
- package/src/slider/index.ts +2 -0
- package/src/ssr/index.tsx +370 -0
- package/src/switch/createSwitch.ts +70 -0
- package/src/switch/index.ts +1 -0
- package/src/table/createTable.ts +526 -0
- package/src/table/createTableCell.ts +147 -0
- package/src/table/createTableColumnHeader.ts +115 -0
- package/src/table/createTableHeaderRow.ts +40 -0
- package/src/table/createTableRow.ts +155 -0
- package/src/table/createTableRowGroup.ts +32 -0
- package/src/table/createTableSelectAllCheckbox.ts +73 -0
- package/src/table/createTableSelectionCheckbox.ts +59 -0
- package/src/table/index.ts +30 -0
- package/src/table/types.ts +165 -0
- package/src/tabs/createTabs.ts +472 -0
- package/src/tabs/index.ts +14 -0
- package/src/tag/createTag.ts +194 -0
- package/src/tag/createTagGroup.ts +154 -0
- package/src/tag/index.ts +12 -0
- package/src/textfield/createTextField.ts +198 -0
- package/src/textfield/index.ts +5 -0
- package/src/toast/createToast.ts +118 -0
- package/src/toast/createToastRegion.ts +100 -0
- package/src/toast/index.ts +11 -0
- package/src/toggle/createToggle.ts +223 -0
- package/src/toggle/createToggleState.ts +94 -0
- package/src/toggle/index.ts +7 -0
- package/src/toolbar/createToolbar.ts +369 -0
- package/src/toolbar/index.ts +6 -0
- package/src/tooltip/createTooltip.ts +79 -0
- package/src/tooltip/createTooltipTrigger.ts +222 -0
- package/src/tooltip/index.ts +6 -0
- package/src/tree/createTree.ts +246 -0
- package/src/tree/createTreeItem.ts +233 -0
- package/src/tree/createTreeSelectionCheckbox.ts +68 -0
- package/src/tree/index.ts +16 -0
- package/src/tree/types.ts +87 -0
- package/src/utils/createDescription.ts +137 -0
- package/src/utils/dom.ts +327 -0
- package/src/utils/env.ts +54 -0
- package/src/utils/events.ts +106 -0
- package/src/utils/filterDOMProps.ts +116 -0
- package/src/utils/focus.ts +151 -0
- package/src/utils/geometry.ts +115 -0
- package/src/utils/globalListeners.ts +142 -0
- package/src/utils/index.ts +80 -0
- package/src/utils/mergeProps.ts +52 -0
- package/src/utils/platform.ts +52 -0
- package/src/utils/reactivity.ts +36 -0
- package/src/utils/textSelection.ts +114 -0
- package/src/visually-hidden/createVisuallyHidden.ts +124 -0
- package/src/visually-hidden/index.ts +6 -0
- package/dist/index.jsx +0 -15845
- 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';
|