@proyecto-viviana/solidaria 0.2.2 → 0.2.4
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,269 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Provides the behavior and accessibility implementation for a listbox component.
|
|
3
|
+
* A listbox displays a list of options and allows a user to select one or more of them.
|
|
4
|
+
* Based on @react-aria/listbox useListBox.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { createEffect, onCleanup, type JSX } 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 { ListState, Key } from '@proyecto-viviana/solid-stately';
|
|
17
|
+
|
|
18
|
+
export interface AriaListBoxProps {
|
|
19
|
+
/** An ID for the listbox. */
|
|
20
|
+
id?: string;
|
|
21
|
+
/** Whether the listbox is disabled. */
|
|
22
|
+
isDisabled?: boolean;
|
|
23
|
+
/** The label for the listbox. */
|
|
24
|
+
label?: JSX.Element;
|
|
25
|
+
/** An accessible label for the listbox when no visible label is provided. */
|
|
26
|
+
'aria-label'?: string;
|
|
27
|
+
/** The ID of an element that labels the listbox. */
|
|
28
|
+
'aria-labelledby'?: string;
|
|
29
|
+
/** The ID of an element that describes the listbox. */
|
|
30
|
+
'aria-describedby'?: string;
|
|
31
|
+
/** Handler called when focus moves into the listbox. */
|
|
32
|
+
onFocus?: (e: FocusEvent) => void;
|
|
33
|
+
/** Handler called when focus moves out of the listbox. */
|
|
34
|
+
onBlur?: (e: FocusEvent) => void;
|
|
35
|
+
/** Handler called when the focus state changes. */
|
|
36
|
+
onFocusChange?: (isFocused: boolean) => void;
|
|
37
|
+
/** Handler called when an item is activated (pressed). */
|
|
38
|
+
onAction?: (key: Key) => void;
|
|
39
|
+
/** Whether focus should automatically wrap around. */
|
|
40
|
+
shouldFocusWrap?: boolean;
|
|
41
|
+
/** Whether selection should occur on press up. */
|
|
42
|
+
shouldSelectOnPressUp?: boolean;
|
|
43
|
+
/** Whether to focus items on hover. */
|
|
44
|
+
shouldFocusOnHover?: boolean;
|
|
45
|
+
/** Whether type-to-select is disabled. @default false */
|
|
46
|
+
disallowTypeAhead?: boolean;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
export interface ListBoxAria {
|
|
50
|
+
/** Props for the listbox element. */
|
|
51
|
+
listBoxProps: JSX.HTMLAttributes<HTMLElement>;
|
|
52
|
+
/** Props for the listbox's label element (if any). */
|
|
53
|
+
labelProps: JSX.HTMLAttributes<HTMLElement>;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Shared data between listbox and options
|
|
57
|
+
const listBoxData = new WeakMap<object, ListBoxData>();
|
|
58
|
+
|
|
59
|
+
interface ListBoxData {
|
|
60
|
+
id: string;
|
|
61
|
+
onAction?: (key: Key) => void;
|
|
62
|
+
shouldSelectOnPressUp?: boolean;
|
|
63
|
+
shouldFocusOnHover?: boolean;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
export function getListBoxData(state: ListState): ListBoxData | undefined {
|
|
67
|
+
return listBoxData.get(state);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Provides the behavior and accessibility implementation for a listbox component.
|
|
72
|
+
* A listbox displays a list of options and allows a user to select one or more of them.
|
|
73
|
+
*/
|
|
74
|
+
export function createListBox<T>(
|
|
75
|
+
props: MaybeAccessor<AriaListBoxProps>,
|
|
76
|
+
state: ListState<T>,
|
|
77
|
+
_ref?: () => HTMLElement | null
|
|
78
|
+
): ListBoxAria {
|
|
79
|
+
const getProps = () => access(props);
|
|
80
|
+
const id = createId(getProps().id);
|
|
81
|
+
|
|
82
|
+
// Development-time warning for missing accessibility labels
|
|
83
|
+
if (isDevEnv()) {
|
|
84
|
+
const p = getProps();
|
|
85
|
+
if (!p.label && !p['aria-label'] && !p['aria-labelledby']) {
|
|
86
|
+
console.warn(
|
|
87
|
+
'[solidaria] A ListBox requires an aria-label or aria-labelledby attribute for accessibility.'
|
|
88
|
+
);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Filter DOM props
|
|
93
|
+
const domProps = () => filterDOMProps(getProps() as unknown as Record<string, unknown>, { labelable: true });
|
|
94
|
+
|
|
95
|
+
// Share data with child options
|
|
96
|
+
createEffect(() => {
|
|
97
|
+
const p = getProps();
|
|
98
|
+
listBoxData.set(state, {
|
|
99
|
+
id,
|
|
100
|
+
onAction: p.onAction,
|
|
101
|
+
shouldSelectOnPressUp: p.shouldSelectOnPressUp,
|
|
102
|
+
shouldFocusOnHover: p.shouldFocusOnHover,
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
onCleanup(() => {
|
|
106
|
+
listBoxData.delete(state);
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
// Handle focus within
|
|
111
|
+
const { focusWithinProps } = createFocusWithin({
|
|
112
|
+
onFocusWithin: (e) => getProps().onFocus?.(e),
|
|
113
|
+
onBlurWithin: (e) => getProps().onBlur?.(e),
|
|
114
|
+
onFocusWithinChange: (isFocused) => {
|
|
115
|
+
getProps().onFocusChange?.(isFocused);
|
|
116
|
+
state.setFocused(isFocused);
|
|
117
|
+
},
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
// Label handling
|
|
121
|
+
const { labelProps, fieldProps } = createLabel({
|
|
122
|
+
get id() {
|
|
123
|
+
return id;
|
|
124
|
+
},
|
|
125
|
+
get label() {
|
|
126
|
+
return getProps().label;
|
|
127
|
+
},
|
|
128
|
+
get 'aria-label'() {
|
|
129
|
+
return getProps()['aria-label'];
|
|
130
|
+
},
|
|
131
|
+
get 'aria-labelledby'() {
|
|
132
|
+
return getProps()['aria-labelledby'];
|
|
133
|
+
},
|
|
134
|
+
labelElementType: 'span',
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
// Type-to-select
|
|
138
|
+
const { typeSelectProps } = createTypeSelect({
|
|
139
|
+
collection: () => state.collection(),
|
|
140
|
+
focusedKey: () => state.focusedKey(),
|
|
141
|
+
onFocusedKeyChange: (key) => state.setFocusedKey(key),
|
|
142
|
+
isKeyDisabled: (key) => state.isDisabled(key),
|
|
143
|
+
get isDisabled() {
|
|
144
|
+
return getProps().disallowTypeAhead ?? false;
|
|
145
|
+
},
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
// Keyboard navigation
|
|
149
|
+
const onKeyDown: JSX.EventHandler<HTMLElement, KeyboardEvent> = (e) => {
|
|
150
|
+
if (getProps().isDisabled) return;
|
|
151
|
+
|
|
152
|
+
const collection = state.collection();
|
|
153
|
+
|
|
154
|
+
switch (e.key) {
|
|
155
|
+
case 'ArrowDown': {
|
|
156
|
+
e.preventDefault();
|
|
157
|
+
const currentKey = state.focusedKey();
|
|
158
|
+
const nextKey = currentKey ? collection.getKeyAfter(currentKey) : collection.getFirstKey();
|
|
159
|
+
if (nextKey) {
|
|
160
|
+
state.setFocusedKey(nextKey);
|
|
161
|
+
if (!e.shiftKey && state.selectionMode() === 'single') {
|
|
162
|
+
state.replaceSelection(nextKey);
|
|
163
|
+
} else if (e.shiftKey && state.selectionMode() === 'multiple') {
|
|
164
|
+
state.extendSelection(nextKey, collection);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
break;
|
|
168
|
+
}
|
|
169
|
+
case 'ArrowUp': {
|
|
170
|
+
e.preventDefault();
|
|
171
|
+
const currentKey = state.focusedKey();
|
|
172
|
+
const prevKey = currentKey ? collection.getKeyBefore(currentKey) : collection.getLastKey();
|
|
173
|
+
if (prevKey) {
|
|
174
|
+
state.setFocusedKey(prevKey);
|
|
175
|
+
if (!e.shiftKey && state.selectionMode() === 'single') {
|
|
176
|
+
state.replaceSelection(prevKey);
|
|
177
|
+
} else if (e.shiftKey && state.selectionMode() === 'multiple') {
|
|
178
|
+
state.extendSelection(prevKey, collection);
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
break;
|
|
182
|
+
}
|
|
183
|
+
case 'Home': {
|
|
184
|
+
e.preventDefault();
|
|
185
|
+
const firstKey = collection.getFirstKey();
|
|
186
|
+
if (firstKey) {
|
|
187
|
+
state.setFocusedKey(firstKey);
|
|
188
|
+
if (e.ctrlKey && e.shiftKey && state.selectionMode() === 'multiple') {
|
|
189
|
+
// Select from current to first
|
|
190
|
+
state.extendSelection(firstKey, collection);
|
|
191
|
+
} else if (!e.shiftKey && state.selectionMode() === 'single') {
|
|
192
|
+
state.replaceSelection(firstKey);
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
break;
|
|
196
|
+
}
|
|
197
|
+
case 'End': {
|
|
198
|
+
e.preventDefault();
|
|
199
|
+
const lastKey = collection.getLastKey();
|
|
200
|
+
if (lastKey) {
|
|
201
|
+
state.setFocusedKey(lastKey);
|
|
202
|
+
if (e.ctrlKey && e.shiftKey && state.selectionMode() === 'multiple') {
|
|
203
|
+
// Select from current to last
|
|
204
|
+
state.extendSelection(lastKey, collection);
|
|
205
|
+
} else if (!e.shiftKey && state.selectionMode() === 'single') {
|
|
206
|
+
state.replaceSelection(lastKey);
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
break;
|
|
210
|
+
}
|
|
211
|
+
case ' ':
|
|
212
|
+
case 'Enter': {
|
|
213
|
+
e.preventDefault();
|
|
214
|
+
const focusedKey = state.focusedKey();
|
|
215
|
+
if (focusedKey != null) {
|
|
216
|
+
if (state.selectionMode() !== 'none') {
|
|
217
|
+
state.toggleSelection(focusedKey);
|
|
218
|
+
}
|
|
219
|
+
getProps().onAction?.(focusedKey);
|
|
220
|
+
}
|
|
221
|
+
break;
|
|
222
|
+
}
|
|
223
|
+
case 'a': {
|
|
224
|
+
if (e.ctrlKey && state.selectionMode() === 'multiple') {
|
|
225
|
+
e.preventDefault();
|
|
226
|
+
state.selectAll();
|
|
227
|
+
}
|
|
228
|
+
break;
|
|
229
|
+
}
|
|
230
|
+
case 'Escape': {
|
|
231
|
+
e.preventDefault();
|
|
232
|
+
if (!state.disallowEmptySelection()) {
|
|
233
|
+
state.clearSelection();
|
|
234
|
+
}
|
|
235
|
+
break;
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
};
|
|
239
|
+
|
|
240
|
+
return {
|
|
241
|
+
get labelProps() {
|
|
242
|
+
return labelProps as JSX.HTMLAttributes<HTMLElement>;
|
|
243
|
+
},
|
|
244
|
+
get listBoxProps() {
|
|
245
|
+
const p = getProps();
|
|
246
|
+
const selectionMode = state.selectionMode();
|
|
247
|
+
|
|
248
|
+
const baseProps = mergeProps(
|
|
249
|
+
domProps(),
|
|
250
|
+
focusWithinProps as Record<string, unknown>,
|
|
251
|
+
fieldProps as Record<string, unknown>,
|
|
252
|
+
{
|
|
253
|
+
role: 'listbox',
|
|
254
|
+
tabIndex: p.isDisabled ? undefined : 0,
|
|
255
|
+
'aria-disabled': p.isDisabled || undefined,
|
|
256
|
+
'aria-multiselectable': selectionMode === 'multiple' ? true : undefined,
|
|
257
|
+
onKeyDown,
|
|
258
|
+
}
|
|
259
|
+
);
|
|
260
|
+
|
|
261
|
+
// Add type-select props if enabled
|
|
262
|
+
if (!p.disallowTypeAhead) {
|
|
263
|
+
return mergeProps(baseProps, typeSelectProps as Record<string, unknown>) as JSX.HTMLAttributes<HTMLElement>;
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
return baseProps as JSX.HTMLAttributes<HTMLElement>;
|
|
267
|
+
},
|
|
268
|
+
};
|
|
269
|
+
}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Provides the behavior and accessibility implementation for an option in a listbox.
|
|
3
|
+
* Based on @react-aria/listbox useOption.
|
|
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 { getListBoxData } from './createListBox';
|
|
13
|
+
import type { ListState, Key } from '@proyecto-viviana/solid-stately';
|
|
14
|
+
|
|
15
|
+
export interface AriaOptionProps {
|
|
16
|
+
/** The unique key for the option. */
|
|
17
|
+
key: Key;
|
|
18
|
+
/** Whether the option is disabled. */
|
|
19
|
+
isDisabled?: boolean;
|
|
20
|
+
/** An accessible label for the option. */
|
|
21
|
+
'aria-label'?: string;
|
|
22
|
+
/** Whether selection should occur on press up. */
|
|
23
|
+
shouldSelectOnPressUp?: boolean;
|
|
24
|
+
/** Whether to focus the option on hover. */
|
|
25
|
+
shouldFocusOnHover?: boolean;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export interface OptionAria {
|
|
29
|
+
/** Props for the option element. */
|
|
30
|
+
optionProps: JSX.HTMLAttributes<HTMLElement>;
|
|
31
|
+
/** Props for the label text inside the option. */
|
|
32
|
+
labelProps: JSX.HTMLAttributes<HTMLElement>;
|
|
33
|
+
/** Props for the description text inside the option. */
|
|
34
|
+
descriptionProps: JSX.HTMLAttributes<HTMLElement>;
|
|
35
|
+
/** Whether the option is currently selected. */
|
|
36
|
+
isSelected: Accessor<boolean>;
|
|
37
|
+
/** Whether the option is currently focused. */
|
|
38
|
+
isFocused: Accessor<boolean>;
|
|
39
|
+
/** Whether the option is keyboard focused. */
|
|
40
|
+
isFocusVisible: Accessor<boolean>;
|
|
41
|
+
/** Whether the option is currently pressed. */
|
|
42
|
+
isPressed: Accessor<boolean>;
|
|
43
|
+
/** Whether the option is disabled. */
|
|
44
|
+
isDisabled: Accessor<boolean>;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Provides the behavior and accessibility implementation for an option in a listbox.
|
|
49
|
+
*/
|
|
50
|
+
export function createOption<T>(
|
|
51
|
+
props: MaybeAccessor<AriaOptionProps>,
|
|
52
|
+
state: ListState<T>,
|
|
53
|
+
_ref?: () => HTMLElement | null
|
|
54
|
+
): OptionAria {
|
|
55
|
+
const getProps = () => access(props);
|
|
56
|
+
|
|
57
|
+
// Get shared data from listbox
|
|
58
|
+
const getData = () => getListBoxData(state);
|
|
59
|
+
|
|
60
|
+
// Computed states
|
|
61
|
+
const isDisabled: Accessor<boolean> = () => {
|
|
62
|
+
return getProps().isDisabled ?? state.isDisabled(getProps().key);
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
const isSelected: Accessor<boolean> = () => {
|
|
66
|
+
return state.isSelected(getProps().key);
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
const isFocused: Accessor<boolean> = () => {
|
|
70
|
+
return state.focusedKey() === getProps().key;
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
// Handle press
|
|
74
|
+
const { pressProps, isPressed } = createPress({
|
|
75
|
+
get isDisabled() {
|
|
76
|
+
return isDisabled();
|
|
77
|
+
},
|
|
78
|
+
onPress() {
|
|
79
|
+
const key = getProps().key;
|
|
80
|
+
const data = getData();
|
|
81
|
+
|
|
82
|
+
if (state.selectionMode() !== 'none') {
|
|
83
|
+
state.select(key);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Call onAction from listbox
|
|
87
|
+
data?.onAction?.(key);
|
|
88
|
+
},
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
// Handle hover
|
|
92
|
+
const { hoverProps, isHovered } = createHover({
|
|
93
|
+
get isDisabled() {
|
|
94
|
+
return isDisabled();
|
|
95
|
+
},
|
|
96
|
+
onHoverStart() {
|
|
97
|
+
const shouldFocus = getProps().shouldFocusOnHover ?? getData()?.shouldFocusOnHover;
|
|
98
|
+
if (shouldFocus) {
|
|
99
|
+
state.setFocusedKey(getProps().key);
|
|
100
|
+
}
|
|
101
|
+
},
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
// Handle focus ring
|
|
105
|
+
const { isFocusVisible, focusProps } = createFocusRing();
|
|
106
|
+
|
|
107
|
+
// Generate unique IDs for label and description
|
|
108
|
+
const labelId = `${getProps().key}-label`;
|
|
109
|
+
const descriptionId = `${getProps().key}-desc`;
|
|
110
|
+
|
|
111
|
+
return {
|
|
112
|
+
get optionProps() {
|
|
113
|
+
const key = getProps().key;
|
|
114
|
+
const selectionMode = state.selectionMode();
|
|
115
|
+
const ariaLabel = getProps()['aria-label'];
|
|
116
|
+
|
|
117
|
+
return mergeProps(
|
|
118
|
+
pressProps as Record<string, unknown>,
|
|
119
|
+
hoverProps as Record<string, unknown>,
|
|
120
|
+
focusProps as Record<string, unknown>,
|
|
121
|
+
{
|
|
122
|
+
role: 'option',
|
|
123
|
+
id: String(key),
|
|
124
|
+
'aria-selected': selectionMode !== 'none' ? isSelected() : undefined,
|
|
125
|
+
'aria-disabled': isDisabled() || undefined,
|
|
126
|
+
'aria-label': ariaLabel,
|
|
127
|
+
'aria-labelledby': !ariaLabel ? labelId : undefined,
|
|
128
|
+
'aria-describedby': descriptionId,
|
|
129
|
+
tabIndex: isFocused() ? 0 : -1,
|
|
130
|
+
'data-selected': isSelected() || undefined,
|
|
131
|
+
'data-focused': isFocused() || undefined,
|
|
132
|
+
'data-focus-visible': isFocusVisible() || undefined,
|
|
133
|
+
'data-pressed': isPressed() || undefined,
|
|
134
|
+
'data-disabled': isDisabled() || undefined,
|
|
135
|
+
'data-hovered': isHovered() || undefined,
|
|
136
|
+
} as Record<string, unknown>
|
|
137
|
+
) as JSX.HTMLAttributes<HTMLElement>;
|
|
138
|
+
},
|
|
139
|
+
labelProps: {
|
|
140
|
+
id: labelId,
|
|
141
|
+
},
|
|
142
|
+
descriptionProps: {
|
|
143
|
+
id: descriptionId,
|
|
144
|
+
},
|
|
145
|
+
isSelected,
|
|
146
|
+
isFocused,
|
|
147
|
+
isFocusVisible: () => isFocused() && isFocusVisible(),
|
|
148
|
+
isPressed,
|
|
149
|
+
isDisabled,
|
|
150
|
+
};
|
|
151
|
+
}
|