@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.
- package/LICENSE +21 -0
- package/dist/actiongroup/createActionGroup.d.ts +29 -0
- package/dist/actiongroup/createActionGroup.d.ts.map +1 -0
- package/dist/actiongroup/index.d.ts +2 -0
- package/dist/actiongroup/index.d.ts.map +1 -0
- package/dist/autocomplete/createAutocomplete.d.ts +6 -2
- package/dist/autocomplete/createAutocomplete.d.ts.map +1 -1
- package/dist/breadcrumbs/createBreadcrumbs.d.ts +2 -0
- package/dist/breadcrumbs/createBreadcrumbs.d.ts.map +1 -1
- package/dist/button/createToggleButtonGroup.d.ts +32 -0
- package/dist/button/createToggleButtonGroup.d.ts.map +1 -0
- package/dist/button/index.d.ts +2 -0
- package/dist/button/index.d.ts.map +1 -1
- package/dist/calendar/createCalendarCell.d.ts +2 -0
- package/dist/calendar/createCalendarCell.d.ts.map +1 -1
- package/dist/calendar/createCalendarGrid.d.ts.map +1 -1
- package/dist/calendar/createRangeCalendarCell.d.ts +3 -1
- package/dist/calendar/createRangeCalendarCell.d.ts.map +1 -1
- package/dist/checkbox/createCheckboxGroup.d.ts +5 -1
- package/dist/checkbox/createCheckboxGroup.d.ts.map +1 -1
- package/dist/collections/index.d.ts +56 -0
- package/dist/collections/index.d.ts.map +1 -0
- package/dist/color/createColorArea.d.ts.map +1 -1
- package/dist/color/createColorSlider.d.ts.map +1 -1
- package/dist/color/createColorWheel.d.ts.map +1 -1
- package/dist/combobox/createComboBox.d.ts +6 -0
- package/dist/combobox/createComboBox.d.ts.map +1 -1
- package/dist/datepicker/createDatePicker.d.ts +6 -0
- package/dist/datepicker/createDatePicker.d.ts.map +1 -1
- package/dist/datepicker/createDateRangePicker.d.ts +40 -0
- package/dist/datepicker/createDateRangePicker.d.ts.map +1 -0
- package/dist/datepicker/createDateSegment.d.ts +1 -1
- package/dist/datepicker/createDateSegment.d.ts.map +1 -1
- package/dist/datepicker/createTimeSegment.d.ts +29 -0
- package/dist/datepicker/createTimeSegment.d.ts.map +1 -0
- package/dist/datepicker/index.d.ts +2 -0
- package/dist/datepicker/index.d.ts.map +1 -1
- package/dist/disclosure/createDisclosureGroup.d.ts +2 -1
- package/dist/disclosure/createDisclosureGroup.d.ts.map +1 -1
- package/dist/dnd/createDrag.d.ts.map +1 -1
- package/dist/dnd/createDraggableCollection.d.ts +4 -0
- package/dist/dnd/createDraggableCollection.d.ts.map +1 -1
- package/dist/dnd/createDraggableItem.d.ts.map +1 -1
- package/dist/dnd/createDrop.d.ts.map +1 -1
- package/dist/dnd/createDroppableCollection.d.ts +32 -1
- package/dist/dnd/createDroppableCollection.d.ts.map +1 -1
- package/dist/dnd/createDroppableItem.d.ts.map +1 -1
- package/dist/dnd/index.d.ts +1 -1
- package/dist/dnd/index.d.ts.map +1 -1
- package/dist/grid/createGrid.d.ts.map +1 -1
- package/dist/gridlist/createGridList.d.ts.map +1 -1
- package/dist/index.d.ts +6 -4
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +4659 -3452
- package/dist/index.js.map +1 -7
- package/dist/index.ssr.js +4659 -3452
- package/dist/index.ssr.js.map +1 -7
- package/dist/interactions/createFocus.d.ts.map +1 -1
- package/dist/interactions/createFocusWithin.d.ts.map +1 -1
- package/dist/link/createLink.d.ts +10 -0
- package/dist/link/createLink.d.ts.map +1 -1
- package/dist/listbox/createListBox.d.ts +1 -0
- package/dist/listbox/createListBox.d.ts.map +1 -1
- package/dist/listbox/createOption.d.ts.map +1 -1
- package/dist/menu/createMenu.d.ts +1 -0
- package/dist/menu/createMenu.d.ts.map +1 -1
- package/dist/meter/createMeter.d.ts.map +1 -1
- package/dist/numberfield/createNumberField.d.ts +18 -0
- package/dist/numberfield/createNumberField.d.ts.map +1 -1
- package/dist/overlays/createModal.d.ts +16 -0
- package/dist/overlays/createModal.d.ts.map +1 -1
- package/dist/overlays/createOverlay.d.ts.map +1 -1
- package/dist/overlays/index.d.ts +1 -1
- package/dist/overlays/index.d.ts.map +1 -1
- package/dist/popover/createOverlayPosition.d.ts.map +1 -1
- package/dist/popover/createPopover.d.ts.map +1 -1
- package/dist/progress/createProgressBar.d.ts.map +1 -1
- package/dist/radio/createRadioGroup.d.ts +2 -2
- package/dist/radio/createRadioGroup.d.ts.map +1 -1
- package/dist/searchfield/createSearchField.d.ts.map +1 -1
- package/dist/select/createHiddenSelect.d.ts.map +1 -1
- package/dist/select/createSelect.d.ts.map +1 -1
- package/dist/slider/createSlider.d.ts.map +1 -1
- package/dist/table/createTable.d.ts.map +1 -1
- package/dist/tabs/createTabs.d.ts +1 -1
- package/dist/tabs/createTabs.d.ts.map +1 -1
- package/dist/tag/createTag.d.ts.map +1 -1
- package/dist/tag/createTagGroup.d.ts.map +1 -1
- package/dist/toast/createToast.d.ts +4 -0
- package/dist/toast/createToast.d.ts.map +1 -1
- package/dist/toast/createToastRegion.d.ts.map +1 -1
- package/dist/toolbar/createToolbar.d.ts.map +1 -1
- package/dist/tooltip/createTooltipTrigger.d.ts.map +1 -1
- package/dist/tree/createTree.d.ts.map +1 -1
- package/dist/tree/createTreeItem.d.ts.map +1 -1
- package/dist/tree/types.d.ts +4 -0
- package/dist/tree/types.d.ts.map +1 -1
- package/dist/utils/env.d.ts +1 -1
- package/dist/utils/env.d.ts.map +1 -1
- package/dist/utils/platform.d.ts.map +1 -1
- package/dist/visually-hidden/createVisuallyHidden.d.ts.map +1 -1
- package/package.json +8 -6
- package/src/actiongroup/createActionGroup.ts +324 -0
- package/src/actiongroup/index.ts +8 -0
- package/src/autocomplete/createAutocomplete.ts +32 -9
- package/src/breadcrumbs/createBreadcrumbs.ts +10 -15
- package/src/button/createButton.ts +1 -1
- package/src/button/createToggleButtonGroup.ts +128 -0
- package/src/button/index.ts +9 -0
- package/src/calendar/createCalendarCell.ts +6 -4
- package/src/calendar/createCalendarGrid.ts +27 -18
- package/src/calendar/createRangeCalendarCell.ts +26 -9
- package/src/checkbox/createCheckboxGroup.ts +21 -4
- package/src/collections/index.ts +242 -0
- package/src/color/createColorArea.ts +380 -314
- package/src/color/createColorField.ts +137 -137
- package/src/color/createColorSlider.ts +286 -197
- package/src/color/createColorSwatch.ts +40 -40
- package/src/color/createColorWheel.ts +218 -208
- package/src/color/index.ts +24 -24
- package/src/color/types.ts +116 -116
- package/src/combobox/createComboBox.ts +670 -647
- package/src/combobox/index.ts +6 -6
- package/src/datepicker/createDatePicker.ts +54 -16
- package/src/datepicker/createDateRangePicker.ts +246 -0
- package/src/datepicker/createDateSegment.ts +185 -31
- package/src/datepicker/createTimeSegment.ts +370 -0
- package/src/datepicker/index.ts +14 -0
- package/src/dialog/createDialog.ts +120 -120
- package/src/dialog/index.ts +2 -2
- package/src/dialog/types.ts +19 -19
- package/src/disclosure/createDisclosureGroup.ts +5 -2
- package/src/dnd/createDrag.ts +224 -209
- package/src/dnd/createDraggableCollection.ts +96 -63
- package/src/dnd/createDraggableItem.ts +259 -243
- package/src/dnd/createDrop.ts +322 -321
- package/src/dnd/createDroppableCollection.ts +682 -293
- package/src/dnd/createDroppableItem.ts +215 -213
- package/src/dnd/index.ts +55 -47
- package/src/dnd/types.ts +89 -89
- package/src/dnd/utils.ts +294 -294
- package/src/focus/createAutoFocus.ts +321 -321
- package/src/focus/createFocusRestore.ts +313 -313
- package/src/focus/createVirtualFocus.ts +396 -396
- package/src/form/createFormValidation.ts +224 -224
- package/src/form/index.ts +11 -11
- package/src/grid/createGrid.ts +3 -1
- package/src/gridlist/createGridList.ts +16 -0
- package/src/gridlist/createGridListItem.ts +1 -1
- package/src/i18n/NumberFormatter.ts +266 -266
- package/src/i18n/createCollator.ts +79 -79
- package/src/i18n/createDateFormatter.ts +83 -83
- package/src/i18n/createFilter.ts +131 -131
- package/src/i18n/createNumberFormatter.ts +52 -52
- package/src/i18n/index.ts +40 -40
- package/src/i18n/locale.tsx +188 -188
- package/src/i18n/utils.ts +99 -99
- package/src/index.ts +51 -0
- package/src/interactions/createFocus.ts +6 -5
- package/src/interactions/createFocusWithin.ts +6 -5
- package/src/interactions/createLongPress.ts +174 -174
- package/src/interactions/createMove.ts +289 -289
- package/src/interactions/createPress.ts +5 -5
- package/src/landmark/createLandmark.ts +377 -377
- package/src/landmark/index.ts +8 -8
- package/src/link/createLink.ts +23 -8
- package/src/listbox/createListBox.ts +308 -269
- package/src/listbox/createOption.ts +162 -151
- package/src/listbox/index.ts +12 -12
- package/src/live-announcer/announce.ts +322 -322
- package/src/live-announcer/index.ts +9 -9
- package/src/menu/createMenu.ts +405 -396
- package/src/menu/createMenuItem.ts +149 -149
- package/src/menu/createMenuTrigger.ts +88 -88
- package/src/menu/index.ts +18 -18
- package/src/meter/createMeter.ts +1 -6
- package/src/numberfield/createNumberField.ts +311 -268
- package/src/numberfield/index.ts +5 -5
- package/src/overlays/ariaHideOutside.ts +219 -219
- package/src/overlays/createInteractOutside.ts +149 -149
- package/src/overlays/createModal.tsx +238 -202
- package/src/overlays/createOverlay.ts +165 -155
- package/src/overlays/createOverlayTrigger.ts +85 -85
- package/src/overlays/createPreventScroll.ts +266 -266
- package/src/overlays/index.ts +48 -44
- package/src/popover/calculatePosition.ts +6 -6
- package/src/popover/createOverlayPosition.ts +7 -4
- package/src/popover/createPopover.ts +21 -7
- package/src/progress/createProgressBar.ts +6 -1
- package/src/radio/createRadioGroup.ts +88 -14
- package/src/searchfield/createSearchField.ts +241 -186
- package/src/searchfield/index.ts +2 -2
- package/src/select/createHiddenSelect.tsx +263 -236
- package/src/select/createSelect.ts +373 -395
- package/src/select/index.ts +14 -14
- package/src/slider/createSlider.ts +364 -349
- package/src/slider/index.ts +2 -2
- package/src/ssr/index.tsx +370 -370
- package/src/table/createTable.ts +3 -1
- package/src/table/createTableColumnHeader.ts +1 -1
- package/src/table/createTableRow.ts +1 -1
- package/src/tabs/createTabs.ts +80 -51
- package/src/tag/createTag.ts +135 -6
- package/src/tag/createTagGroup.ts +7 -2
- package/src/toast/createToast.ts +8 -2
- package/src/toast/createToastRegion.ts +0 -1
- package/src/toolbar/createToolbar.ts +75 -1
- package/src/tooltip/createTooltip.ts +79 -79
- package/src/tooltip/createTooltipTrigger.ts +226 -222
- package/src/tooltip/index.ts +6 -6
- package/src/tree/createTree.ts +261 -246
- package/src/tree/createTreeItem.ts +282 -233
- package/src/tree/createTreeSelectionCheckbox.ts +68 -68
- package/src/tree/index.ts +16 -16
- package/src/tree/types.ts +91 -87
- package/src/utils/env.ts +55 -54
- package/src/utils/platform.ts +16 -6
- package/src/visually-hidden/createVisuallyHidden.ts +139 -124
- package/src/visually-hidden/index.ts +6 -6
|
@@ -1,268 +1,311 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Provides the behavior and accessibility implementation for a number field.
|
|
3
|
-
* Based on @react-aria/numberfield useNumberField.
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { type JSX, createMemo } from 'solid-js';
|
|
7
|
-
import { createLabel } from '../label/createLabel';
|
|
8
|
-
import { filterDOMProps } from '../utils/filterDOMProps';
|
|
9
|
-
import { mergeProps } from '../utils/mergeProps';
|
|
10
|
-
import { createId } from '../ssr';
|
|
11
|
-
import { access, type MaybeAccessor } from '../utils/reactivity';
|
|
12
|
-
import type { NumberFieldState } from '@proyecto-viviana/solid-stately';
|
|
13
|
-
|
|
14
|
-
export interface AriaNumberFieldProps {
|
|
15
|
-
/** A label for the number field. */
|
|
16
|
-
label?: JSX.Element;
|
|
17
|
-
/** An accessible label for the number field. */
|
|
18
|
-
'aria-label'?: string;
|
|
19
|
-
/** The element ID that labels the number field. */
|
|
20
|
-
'aria-labelledby'?: string;
|
|
21
|
-
/** The element ID that describes the number field. */
|
|
22
|
-
'aria-describedby'?: string;
|
|
23
|
-
/** Whether the number field is disabled. */
|
|
24
|
-
isDisabled?: boolean;
|
|
25
|
-
/** Whether the number field is read-only. */
|
|
26
|
-
isReadOnly?: boolean;
|
|
27
|
-
/** Whether the number field is required. */
|
|
28
|
-
isRequired?: boolean;
|
|
29
|
-
/** Whether the number field is invalid. */
|
|
30
|
-
isInvalid?: boolean;
|
|
31
|
-
/** A description for the number field. */
|
|
32
|
-
description?: JSX.Element;
|
|
33
|
-
/** An error message for the number field. */
|
|
34
|
-
errorMessage?: JSX.Element;
|
|
35
|
-
/** The ID of the number field. */
|
|
36
|
-
id?: string;
|
|
37
|
-
/** Whether to auto-focus the input. */
|
|
38
|
-
autoFocus?: boolean;
|
|
39
|
-
/** The name for the form input. */
|
|
40
|
-
name?: string;
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
*/
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
};
|
|
120
|
-
|
|
121
|
-
//
|
|
122
|
-
const
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
}
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
1
|
+
/**
|
|
2
|
+
* Provides the behavior and accessibility implementation for a number field.
|
|
3
|
+
* Based on @react-aria/numberfield useNumberField.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { type JSX, createMemo } from 'solid-js';
|
|
7
|
+
import { createLabel } from '../label/createLabel';
|
|
8
|
+
import { filterDOMProps } from '../utils/filterDOMProps';
|
|
9
|
+
import { mergeProps } from '../utils/mergeProps';
|
|
10
|
+
import { createId } from '../ssr';
|
|
11
|
+
import { access, type MaybeAccessor } from '../utils/reactivity';
|
|
12
|
+
import type { NumberFieldState } from '@proyecto-viviana/solid-stately';
|
|
13
|
+
|
|
14
|
+
export interface AriaNumberFieldProps {
|
|
15
|
+
/** A label for the number field. */
|
|
16
|
+
label?: JSX.Element;
|
|
17
|
+
/** An accessible label for the number field. */
|
|
18
|
+
'aria-label'?: string;
|
|
19
|
+
/** The element ID that labels the number field. */
|
|
20
|
+
'aria-labelledby'?: string;
|
|
21
|
+
/** The element ID that describes the number field. */
|
|
22
|
+
'aria-describedby'?: string;
|
|
23
|
+
/** Whether the number field is disabled. */
|
|
24
|
+
isDisabled?: boolean;
|
|
25
|
+
/** Whether the number field is read-only. */
|
|
26
|
+
isReadOnly?: boolean;
|
|
27
|
+
/** Whether the number field is required. */
|
|
28
|
+
isRequired?: boolean;
|
|
29
|
+
/** Whether the number field is invalid. */
|
|
30
|
+
isInvalid?: boolean;
|
|
31
|
+
/** A description for the number field. */
|
|
32
|
+
description?: JSX.Element;
|
|
33
|
+
/** An error message for the number field. */
|
|
34
|
+
errorMessage?: JSX.Element;
|
|
35
|
+
/** The ID of the number field. */
|
|
36
|
+
id?: string;
|
|
37
|
+
/** Whether to auto-focus the input. */
|
|
38
|
+
autoFocus?: boolean;
|
|
39
|
+
/** The name for the form input. */
|
|
40
|
+
name?: string;
|
|
41
|
+
/** The form element this input belongs to. */
|
|
42
|
+
form?: string;
|
|
43
|
+
/** Handler for focus events. */
|
|
44
|
+
onFocus?: JSX.EventHandler<HTMLInputElement, FocusEvent>;
|
|
45
|
+
/** Handler for blur events. */
|
|
46
|
+
onBlur?: JSX.EventHandler<HTMLInputElement, FocusEvent>;
|
|
47
|
+
/** Handler called when focus state changes. */
|
|
48
|
+
onFocusChange?: (isFocused: boolean) => void;
|
|
49
|
+
/** Handler for key down events. */
|
|
50
|
+
onKeyDown?: JSX.EventHandler<HTMLInputElement, KeyboardEvent>;
|
|
51
|
+
/** Handler for key up events. */
|
|
52
|
+
onKeyUp?: JSX.EventHandler<HTMLInputElement, KeyboardEvent>;
|
|
53
|
+
/** Handler for paste events. */
|
|
54
|
+
onPaste?: JSX.EventHandler<HTMLInputElement, ClipboardEvent>;
|
|
55
|
+
/** Handler for copy events. */
|
|
56
|
+
onCopy?: JSX.EventHandler<HTMLInputElement, ClipboardEvent>;
|
|
57
|
+
/** Handler for cut events. */
|
|
58
|
+
onCut?: JSX.EventHandler<HTMLInputElement, ClipboardEvent>;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export interface NumberFieldAria {
|
|
62
|
+
/** Props for the label element. */
|
|
63
|
+
labelProps: JSX.HTMLAttributes<HTMLElement>;
|
|
64
|
+
/** Props for the input element group. */
|
|
65
|
+
groupProps: JSX.HTMLAttributes<HTMLElement>;
|
|
66
|
+
/** Props for the input element. */
|
|
67
|
+
inputProps: JSX.InputHTMLAttributes<HTMLInputElement>;
|
|
68
|
+
/** Props for the increment button. */
|
|
69
|
+
incrementButtonProps: JSX.ButtonHTMLAttributes<HTMLButtonElement>;
|
|
70
|
+
/** Props for the decrement button. */
|
|
71
|
+
decrementButtonProps: JSX.ButtonHTMLAttributes<HTMLButtonElement>;
|
|
72
|
+
/** Props for the description element. */
|
|
73
|
+
descriptionProps: JSX.HTMLAttributes<HTMLElement>;
|
|
74
|
+
/** Props for the error message element. */
|
|
75
|
+
errorMessageProps: JSX.HTMLAttributes<HTMLElement>;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Provides the behavior and accessibility implementation for a number field.
|
|
80
|
+
*/
|
|
81
|
+
export function createNumberField(
|
|
82
|
+
props: MaybeAccessor<AriaNumberFieldProps>,
|
|
83
|
+
state: NumberFieldState,
|
|
84
|
+
inputRef?: () => HTMLInputElement | null
|
|
85
|
+
): NumberFieldAria {
|
|
86
|
+
const getProps = () => access(props);
|
|
87
|
+
const id = createId(getProps().id);
|
|
88
|
+
|
|
89
|
+
// Generate IDs for associated elements
|
|
90
|
+
const inputId = `${id}-input`;
|
|
91
|
+
const incrementId = `${id}-increment`;
|
|
92
|
+
const decrementId = `${id}-decrement`;
|
|
93
|
+
const descriptionId = `${id}-description`;
|
|
94
|
+
const errorMessageId = `${id}-error`;
|
|
95
|
+
|
|
96
|
+
// Label handling
|
|
97
|
+
const { labelProps, fieldProps } = createLabel({
|
|
98
|
+
get id() {
|
|
99
|
+
return inputId;
|
|
100
|
+
},
|
|
101
|
+
get label() {
|
|
102
|
+
return getProps().label;
|
|
103
|
+
},
|
|
104
|
+
get 'aria-label'() {
|
|
105
|
+
return getProps()['aria-label'];
|
|
106
|
+
},
|
|
107
|
+
get 'aria-labelledby'() {
|
|
108
|
+
return getProps()['aria-labelledby'];
|
|
109
|
+
},
|
|
110
|
+
labelElementType: 'span',
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
// Get the label text for button aria-labels
|
|
114
|
+
const getLabelText = (): string => {
|
|
115
|
+
const p = getProps();
|
|
116
|
+
if (p['aria-label']) return p['aria-label'];
|
|
117
|
+
if (typeof p.label === 'string') return p.label;
|
|
118
|
+
return 'value';
|
|
119
|
+
};
|
|
120
|
+
|
|
121
|
+
// Filter DOM props
|
|
122
|
+
const domProps = createMemo(() =>
|
|
123
|
+
filterDOMProps(getProps() as unknown as Record<string, unknown>, { labelable: true })
|
|
124
|
+
);
|
|
125
|
+
|
|
126
|
+
// Handle input change
|
|
127
|
+
const onInputChange: JSX.EventHandler<HTMLInputElement, Event> = (e) => {
|
|
128
|
+
const value = e.currentTarget.value;
|
|
129
|
+
if (state.validate(value)) {
|
|
130
|
+
state.setInputValue(value);
|
|
131
|
+
}
|
|
132
|
+
};
|
|
133
|
+
|
|
134
|
+
// Handle input blur - commit value
|
|
135
|
+
const onInputBlur: JSX.EventHandler<HTMLInputElement, FocusEvent> = (e) => {
|
|
136
|
+
state.commit();
|
|
137
|
+
const p = getProps();
|
|
138
|
+
p.onBlur?.(e);
|
|
139
|
+
p.onFocusChange?.(false);
|
|
140
|
+
};
|
|
141
|
+
|
|
142
|
+
const onInputFocus: JSX.EventHandler<HTMLInputElement, FocusEvent> = (e) => {
|
|
143
|
+
const p = getProps();
|
|
144
|
+
p.onFocus?.(e);
|
|
145
|
+
p.onFocusChange?.(true);
|
|
146
|
+
};
|
|
147
|
+
|
|
148
|
+
// Handle keyboard events
|
|
149
|
+
const onKeyDown: JSX.EventHandler<HTMLInputElement, KeyboardEvent> = (e) => {
|
|
150
|
+
const p = getProps();
|
|
151
|
+
if (p.isDisabled || p.isReadOnly) {
|
|
152
|
+
p.onKeyDown?.(e);
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
switch (e.key) {
|
|
157
|
+
case 'ArrowUp':
|
|
158
|
+
e.preventDefault();
|
|
159
|
+
state.increment();
|
|
160
|
+
break;
|
|
161
|
+
case 'ArrowDown':
|
|
162
|
+
e.preventDefault();
|
|
163
|
+
state.decrement();
|
|
164
|
+
break;
|
|
165
|
+
case 'PageUp':
|
|
166
|
+
e.preventDefault();
|
|
167
|
+
state.incrementToMax();
|
|
168
|
+
break;
|
|
169
|
+
case 'PageDown':
|
|
170
|
+
e.preventDefault();
|
|
171
|
+
state.decrementToMin();
|
|
172
|
+
break;
|
|
173
|
+
case 'Home':
|
|
174
|
+
e.preventDefault();
|
|
175
|
+
state.decrementToMin();
|
|
176
|
+
break;
|
|
177
|
+
case 'End':
|
|
178
|
+
e.preventDefault();
|
|
179
|
+
state.incrementToMax();
|
|
180
|
+
break;
|
|
181
|
+
case 'Enter':
|
|
182
|
+
state.commit();
|
|
183
|
+
break;
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
p.onKeyDown?.(e);
|
|
187
|
+
};
|
|
188
|
+
|
|
189
|
+
const onKeyUp: JSX.EventHandler<HTMLInputElement, KeyboardEvent> = (e) => {
|
|
190
|
+
getProps().onKeyUp?.(e);
|
|
191
|
+
};
|
|
192
|
+
|
|
193
|
+
// Handle increment button
|
|
194
|
+
const onIncrementPress = () => {
|
|
195
|
+
state.increment();
|
|
196
|
+
// Return focus to input
|
|
197
|
+
inputRef?.()?.focus();
|
|
198
|
+
};
|
|
199
|
+
|
|
200
|
+
// Handle decrement button
|
|
201
|
+
const onDecrementPress = () => {
|
|
202
|
+
state.decrement();
|
|
203
|
+
// Return focus to input
|
|
204
|
+
inputRef?.()?.focus();
|
|
205
|
+
};
|
|
206
|
+
|
|
207
|
+
// Build aria-describedby
|
|
208
|
+
const getAriaDescribedBy = () => {
|
|
209
|
+
const p = getProps();
|
|
210
|
+
const parts: string[] = [];
|
|
211
|
+
if (p['aria-describedby']) parts.push(p['aria-describedby']);
|
|
212
|
+
if (p.description) parts.push(descriptionId);
|
|
213
|
+
if (p.isInvalid && p.errorMessage) parts.push(errorMessageId);
|
|
214
|
+
return parts.length > 0 ? parts.join(' ') : undefined;
|
|
215
|
+
};
|
|
216
|
+
|
|
217
|
+
return {
|
|
218
|
+
get labelProps() {
|
|
219
|
+
return labelProps as JSX.HTMLAttributes<HTMLElement>;
|
|
220
|
+
},
|
|
221
|
+
get groupProps() {
|
|
222
|
+
return {
|
|
223
|
+
role: 'group',
|
|
224
|
+
'aria-disabled': getProps().isDisabled || undefined,
|
|
225
|
+
'aria-invalid': getProps().isInvalid || undefined,
|
|
226
|
+
} as JSX.HTMLAttributes<HTMLElement>;
|
|
227
|
+
},
|
|
228
|
+
get inputProps() {
|
|
229
|
+
const p = getProps();
|
|
230
|
+
const isDisabled = p.isDisabled ?? state.isDisabled();
|
|
231
|
+
const isReadOnly = p.isReadOnly ?? state.isReadOnly();
|
|
232
|
+
|
|
233
|
+
return mergeProps(
|
|
234
|
+
domProps(),
|
|
235
|
+
fieldProps as Record<string, unknown>,
|
|
236
|
+
{
|
|
237
|
+
id: inputId,
|
|
238
|
+
type: 'text',
|
|
239
|
+
inputMode: 'decimal' as const,
|
|
240
|
+
autoComplete: 'off',
|
|
241
|
+
autoCorrect: 'off',
|
|
242
|
+
spellcheck: false,
|
|
243
|
+
role: 'spinbutton',
|
|
244
|
+
'aria-valuenow': isNaN(state.numberValue()) ? undefined : state.numberValue(),
|
|
245
|
+
'aria-valuemin': state.minValue(),
|
|
246
|
+
'aria-valuemax': state.maxValue(),
|
|
247
|
+
'aria-valuetext': isNaN(state.numberValue()) ? undefined : state.inputValue(),
|
|
248
|
+
'aria-invalid': p.isInvalid || undefined,
|
|
249
|
+
'aria-required': p.isRequired || undefined,
|
|
250
|
+
'aria-describedby': getAriaDescribedBy(),
|
|
251
|
+
disabled: isDisabled || undefined,
|
|
252
|
+
readOnly: isReadOnly || undefined,
|
|
253
|
+
value: state.inputValue(),
|
|
254
|
+
onChange: onInputChange,
|
|
255
|
+
onFocus: onInputFocus,
|
|
256
|
+
onBlur: onInputBlur,
|
|
257
|
+
onKeyDown,
|
|
258
|
+
onKeyUp,
|
|
259
|
+
onPaste: p.onPaste,
|
|
260
|
+
onCopy: p.onCopy,
|
|
261
|
+
onCut: p.onCut,
|
|
262
|
+
name: p.name,
|
|
263
|
+
form: p.form,
|
|
264
|
+
autoFocus: p.autoFocus,
|
|
265
|
+
} as Record<string, unknown>
|
|
266
|
+
) as JSX.InputHTMLAttributes<HTMLInputElement>;
|
|
267
|
+
},
|
|
268
|
+
get incrementButtonProps() {
|
|
269
|
+
const p = getProps();
|
|
270
|
+
const isDisabled = p.isDisabled ?? state.isDisabled();
|
|
271
|
+
const canIncrement = state.canIncrement();
|
|
272
|
+
|
|
273
|
+
return {
|
|
274
|
+
id: incrementId,
|
|
275
|
+
type: 'button' as const,
|
|
276
|
+
tabIndex: -1, // Exclude from tab order
|
|
277
|
+
'aria-label': `Increase ${getLabelText()}`,
|
|
278
|
+
'aria-controls': inputId,
|
|
279
|
+
disabled: isDisabled || !canIncrement,
|
|
280
|
+
onClick: onIncrementPress,
|
|
281
|
+
onMouseDown: (e: MouseEvent) => e.preventDefault(), // Prevent focus
|
|
282
|
+
} as JSX.ButtonHTMLAttributes<HTMLButtonElement>;
|
|
283
|
+
},
|
|
284
|
+
get decrementButtonProps() {
|
|
285
|
+
const p = getProps();
|
|
286
|
+
const isDisabled = p.isDisabled ?? state.isDisabled();
|
|
287
|
+
const canDecrement = state.canDecrement();
|
|
288
|
+
|
|
289
|
+
return {
|
|
290
|
+
id: decrementId,
|
|
291
|
+
type: 'button' as const,
|
|
292
|
+
tabIndex: -1, // Exclude from tab order
|
|
293
|
+
'aria-label': `Decrease ${getLabelText()}`,
|
|
294
|
+
'aria-controls': inputId,
|
|
295
|
+
disabled: isDisabled || !canDecrement,
|
|
296
|
+
onClick: onDecrementPress,
|
|
297
|
+
onMouseDown: (e: MouseEvent) => e.preventDefault(), // Prevent focus
|
|
298
|
+
} as JSX.ButtonHTMLAttributes<HTMLButtonElement>;
|
|
299
|
+
},
|
|
300
|
+
get descriptionProps() {
|
|
301
|
+
return {
|
|
302
|
+
id: descriptionId,
|
|
303
|
+
} as JSX.HTMLAttributes<HTMLElement>;
|
|
304
|
+
},
|
|
305
|
+
get errorMessageProps() {
|
|
306
|
+
return {
|
|
307
|
+
id: errorMessageId,
|
|
308
|
+
} as JSX.HTMLAttributes<HTMLElement>;
|
|
309
|
+
},
|
|
310
|
+
};
|
|
311
|
+
}
|
package/src/numberfield/index.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
export {
|
|
2
|
-
createNumberField,
|
|
3
|
-
type AriaNumberFieldProps,
|
|
4
|
-
type NumberFieldAria,
|
|
5
|
-
} from './createNumberField';
|
|
1
|
+
export {
|
|
2
|
+
createNumberField,
|
|
3
|
+
type AriaNumberFieldProps,
|
|
4
|
+
type NumberFieldAria,
|
|
5
|
+
} from './createNumberField';
|