@proyecto-viviana/solidaria 0.2.4 → 0.2.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (219) hide show
  1. package/LICENSE +21 -0
  2. package/dist/actiongroup/createActionGroup.d.ts +29 -0
  3. package/dist/actiongroup/createActionGroup.d.ts.map +1 -0
  4. package/dist/actiongroup/index.d.ts +2 -0
  5. package/dist/actiongroup/index.d.ts.map +1 -0
  6. package/dist/autocomplete/createAutocomplete.d.ts +6 -2
  7. package/dist/autocomplete/createAutocomplete.d.ts.map +1 -1
  8. package/dist/breadcrumbs/createBreadcrumbs.d.ts +2 -0
  9. package/dist/breadcrumbs/createBreadcrumbs.d.ts.map +1 -1
  10. package/dist/button/createToggleButtonGroup.d.ts +32 -0
  11. package/dist/button/createToggleButtonGroup.d.ts.map +1 -0
  12. package/dist/button/index.d.ts +2 -0
  13. package/dist/button/index.d.ts.map +1 -1
  14. package/dist/calendar/createCalendarCell.d.ts +2 -0
  15. package/dist/calendar/createCalendarCell.d.ts.map +1 -1
  16. package/dist/calendar/createCalendarGrid.d.ts.map +1 -1
  17. package/dist/calendar/createRangeCalendarCell.d.ts +3 -1
  18. package/dist/calendar/createRangeCalendarCell.d.ts.map +1 -1
  19. package/dist/checkbox/createCheckboxGroup.d.ts +5 -1
  20. package/dist/checkbox/createCheckboxGroup.d.ts.map +1 -1
  21. package/dist/collections/index.d.ts +56 -0
  22. package/dist/collections/index.d.ts.map +1 -0
  23. package/dist/color/createColorArea.d.ts.map +1 -1
  24. package/dist/color/createColorSlider.d.ts.map +1 -1
  25. package/dist/color/createColorWheel.d.ts.map +1 -1
  26. package/dist/combobox/createComboBox.d.ts +6 -0
  27. package/dist/combobox/createComboBox.d.ts.map +1 -1
  28. package/dist/datepicker/createDatePicker.d.ts +6 -0
  29. package/dist/datepicker/createDatePicker.d.ts.map +1 -1
  30. package/dist/datepicker/createDateRangePicker.d.ts +40 -0
  31. package/dist/datepicker/createDateRangePicker.d.ts.map +1 -0
  32. package/dist/datepicker/createDateSegment.d.ts +1 -1
  33. package/dist/datepicker/createDateSegment.d.ts.map +1 -1
  34. package/dist/datepicker/createTimeSegment.d.ts +29 -0
  35. package/dist/datepicker/createTimeSegment.d.ts.map +1 -0
  36. package/dist/datepicker/index.d.ts +2 -0
  37. package/dist/datepicker/index.d.ts.map +1 -1
  38. package/dist/disclosure/createDisclosureGroup.d.ts +2 -1
  39. package/dist/disclosure/createDisclosureGroup.d.ts.map +1 -1
  40. package/dist/dnd/createDrag.d.ts.map +1 -1
  41. package/dist/dnd/createDraggableCollection.d.ts +4 -0
  42. package/dist/dnd/createDraggableCollection.d.ts.map +1 -1
  43. package/dist/dnd/createDraggableItem.d.ts.map +1 -1
  44. package/dist/dnd/createDrop.d.ts.map +1 -1
  45. package/dist/dnd/createDroppableCollection.d.ts +32 -1
  46. package/dist/dnd/createDroppableCollection.d.ts.map +1 -1
  47. package/dist/dnd/createDroppableItem.d.ts.map +1 -1
  48. package/dist/dnd/index.d.ts +1 -1
  49. package/dist/dnd/index.d.ts.map +1 -1
  50. package/dist/grid/createGrid.d.ts.map +1 -1
  51. package/dist/gridlist/createGridList.d.ts.map +1 -1
  52. package/dist/index.d.ts +6 -4
  53. package/dist/index.d.ts.map +1 -1
  54. package/dist/index.js +4659 -3452
  55. package/dist/index.js.map +1 -7
  56. package/dist/index.ssr.js +4659 -3452
  57. package/dist/index.ssr.js.map +1 -7
  58. package/dist/interactions/createFocus.d.ts.map +1 -1
  59. package/dist/interactions/createFocusWithin.d.ts.map +1 -1
  60. package/dist/link/createLink.d.ts +10 -0
  61. package/dist/link/createLink.d.ts.map +1 -1
  62. package/dist/listbox/createListBox.d.ts +1 -0
  63. package/dist/listbox/createListBox.d.ts.map +1 -1
  64. package/dist/listbox/createOption.d.ts.map +1 -1
  65. package/dist/menu/createMenu.d.ts +1 -0
  66. package/dist/menu/createMenu.d.ts.map +1 -1
  67. package/dist/meter/createMeter.d.ts.map +1 -1
  68. package/dist/numberfield/createNumberField.d.ts +18 -0
  69. package/dist/numberfield/createNumberField.d.ts.map +1 -1
  70. package/dist/overlays/createModal.d.ts +16 -0
  71. package/dist/overlays/createModal.d.ts.map +1 -1
  72. package/dist/overlays/createOverlay.d.ts.map +1 -1
  73. package/dist/overlays/index.d.ts +1 -1
  74. package/dist/overlays/index.d.ts.map +1 -1
  75. package/dist/popover/createOverlayPosition.d.ts.map +1 -1
  76. package/dist/popover/createPopover.d.ts.map +1 -1
  77. package/dist/progress/createProgressBar.d.ts.map +1 -1
  78. package/dist/radio/createRadioGroup.d.ts +2 -2
  79. package/dist/radio/createRadioGroup.d.ts.map +1 -1
  80. package/dist/searchfield/createSearchField.d.ts.map +1 -1
  81. package/dist/select/createHiddenSelect.d.ts.map +1 -1
  82. package/dist/select/createSelect.d.ts.map +1 -1
  83. package/dist/slider/createSlider.d.ts.map +1 -1
  84. package/dist/table/createTable.d.ts.map +1 -1
  85. package/dist/tabs/createTabs.d.ts +1 -1
  86. package/dist/tabs/createTabs.d.ts.map +1 -1
  87. package/dist/tag/createTag.d.ts.map +1 -1
  88. package/dist/tag/createTagGroup.d.ts.map +1 -1
  89. package/dist/toast/createToast.d.ts +4 -0
  90. package/dist/toast/createToast.d.ts.map +1 -1
  91. package/dist/toast/createToastRegion.d.ts.map +1 -1
  92. package/dist/toolbar/createToolbar.d.ts.map +1 -1
  93. package/dist/tooltip/createTooltipTrigger.d.ts.map +1 -1
  94. package/dist/tree/createTree.d.ts.map +1 -1
  95. package/dist/tree/createTreeItem.d.ts.map +1 -1
  96. package/dist/tree/types.d.ts +4 -0
  97. package/dist/tree/types.d.ts.map +1 -1
  98. package/dist/utils/env.d.ts +1 -1
  99. package/dist/utils/env.d.ts.map +1 -1
  100. package/dist/utils/platform.d.ts.map +1 -1
  101. package/dist/visually-hidden/createVisuallyHidden.d.ts.map +1 -1
  102. package/package.json +8 -6
  103. package/src/actiongroup/createActionGroup.ts +324 -0
  104. package/src/actiongroup/index.ts +8 -0
  105. package/src/autocomplete/createAutocomplete.ts +32 -9
  106. package/src/breadcrumbs/createBreadcrumbs.ts +10 -15
  107. package/src/button/createButton.ts +1 -1
  108. package/src/button/createToggleButtonGroup.ts +128 -0
  109. package/src/button/index.ts +9 -0
  110. package/src/calendar/createCalendarCell.ts +6 -4
  111. package/src/calendar/createCalendarGrid.ts +27 -18
  112. package/src/calendar/createRangeCalendarCell.ts +26 -9
  113. package/src/checkbox/createCheckboxGroup.ts +21 -4
  114. package/src/collections/index.ts +242 -0
  115. package/src/color/createColorArea.ts +380 -314
  116. package/src/color/createColorField.ts +137 -137
  117. package/src/color/createColorSlider.ts +286 -197
  118. package/src/color/createColorSwatch.ts +40 -40
  119. package/src/color/createColorWheel.ts +218 -208
  120. package/src/color/index.ts +24 -24
  121. package/src/color/types.ts +116 -116
  122. package/src/combobox/createComboBox.ts +670 -647
  123. package/src/combobox/index.ts +6 -6
  124. package/src/datepicker/createDatePicker.ts +54 -16
  125. package/src/datepicker/createDateRangePicker.ts +246 -0
  126. package/src/datepicker/createDateSegment.ts +185 -31
  127. package/src/datepicker/createTimeSegment.ts +370 -0
  128. package/src/datepicker/index.ts +14 -0
  129. package/src/dialog/createDialog.ts +120 -120
  130. package/src/dialog/index.ts +2 -2
  131. package/src/dialog/types.ts +19 -19
  132. package/src/disclosure/createDisclosureGroup.ts +5 -2
  133. package/src/dnd/createDrag.ts +224 -209
  134. package/src/dnd/createDraggableCollection.ts +96 -63
  135. package/src/dnd/createDraggableItem.ts +259 -243
  136. package/src/dnd/createDrop.ts +322 -321
  137. package/src/dnd/createDroppableCollection.ts +682 -293
  138. package/src/dnd/createDroppableItem.ts +215 -213
  139. package/src/dnd/index.ts +55 -47
  140. package/src/dnd/types.ts +89 -89
  141. package/src/dnd/utils.ts +294 -294
  142. package/src/focus/createAutoFocus.ts +321 -321
  143. package/src/focus/createFocusRestore.ts +313 -313
  144. package/src/focus/createVirtualFocus.ts +396 -396
  145. package/src/form/createFormValidation.ts +224 -224
  146. package/src/form/index.ts +11 -11
  147. package/src/grid/createGrid.ts +3 -1
  148. package/src/gridlist/createGridList.ts +16 -0
  149. package/src/gridlist/createGridListItem.ts +1 -1
  150. package/src/i18n/NumberFormatter.ts +266 -266
  151. package/src/i18n/createCollator.ts +79 -79
  152. package/src/i18n/createDateFormatter.ts +83 -83
  153. package/src/i18n/createFilter.ts +131 -131
  154. package/src/i18n/createNumberFormatter.ts +52 -52
  155. package/src/i18n/index.ts +40 -40
  156. package/src/i18n/locale.tsx +188 -188
  157. package/src/i18n/utils.ts +99 -99
  158. package/src/index.ts +51 -0
  159. package/src/interactions/createFocus.ts +6 -5
  160. package/src/interactions/createFocusWithin.ts +6 -5
  161. package/src/interactions/createLongPress.ts +174 -174
  162. package/src/interactions/createMove.ts +289 -289
  163. package/src/interactions/createPress.ts +5 -5
  164. package/src/landmark/createLandmark.ts +377 -377
  165. package/src/landmark/index.ts +8 -8
  166. package/src/link/createLink.ts +23 -8
  167. package/src/listbox/createListBox.ts +308 -269
  168. package/src/listbox/createOption.ts +162 -151
  169. package/src/listbox/index.ts +12 -12
  170. package/src/live-announcer/announce.ts +322 -322
  171. package/src/live-announcer/index.ts +9 -9
  172. package/src/menu/createMenu.ts +405 -396
  173. package/src/menu/createMenuItem.ts +149 -149
  174. package/src/menu/createMenuTrigger.ts +88 -88
  175. package/src/menu/index.ts +18 -18
  176. package/src/meter/createMeter.ts +1 -6
  177. package/src/numberfield/createNumberField.ts +311 -268
  178. package/src/numberfield/index.ts +5 -5
  179. package/src/overlays/ariaHideOutside.ts +219 -219
  180. package/src/overlays/createInteractOutside.ts +149 -149
  181. package/src/overlays/createModal.tsx +238 -202
  182. package/src/overlays/createOverlay.ts +165 -155
  183. package/src/overlays/createOverlayTrigger.ts +85 -85
  184. package/src/overlays/createPreventScroll.ts +266 -266
  185. package/src/overlays/index.ts +48 -44
  186. package/src/popover/calculatePosition.ts +6 -6
  187. package/src/popover/createOverlayPosition.ts +7 -4
  188. package/src/popover/createPopover.ts +21 -7
  189. package/src/progress/createProgressBar.ts +6 -1
  190. package/src/radio/createRadioGroup.ts +88 -14
  191. package/src/searchfield/createSearchField.ts +241 -186
  192. package/src/searchfield/index.ts +2 -2
  193. package/src/select/createHiddenSelect.tsx +263 -236
  194. package/src/select/createSelect.ts +373 -395
  195. package/src/select/index.ts +14 -14
  196. package/src/slider/createSlider.ts +364 -349
  197. package/src/slider/index.ts +2 -2
  198. package/src/ssr/index.tsx +370 -370
  199. package/src/table/createTable.ts +3 -1
  200. package/src/table/createTableColumnHeader.ts +1 -1
  201. package/src/table/createTableRow.ts +1 -1
  202. package/src/tabs/createTabs.ts +80 -51
  203. package/src/tag/createTag.ts +135 -6
  204. package/src/tag/createTagGroup.ts +7 -2
  205. package/src/toast/createToast.ts +8 -2
  206. package/src/toast/createToastRegion.ts +0 -1
  207. package/src/toolbar/createToolbar.ts +75 -1
  208. package/src/tooltip/createTooltip.ts +79 -79
  209. package/src/tooltip/createTooltipTrigger.ts +226 -222
  210. package/src/tooltip/index.ts +6 -6
  211. package/src/tree/createTree.ts +261 -246
  212. package/src/tree/createTreeItem.ts +282 -233
  213. package/src/tree/createTreeSelectionCheckbox.ts +68 -68
  214. package/src/tree/index.ts +16 -16
  215. package/src/tree/types.ts +91 -87
  216. package/src/utils/env.ts +55 -54
  217. package/src/utils/platform.ts +16 -6
  218. package/src/visually-hidden/createVisuallyHidden.ts +139 -124
  219. package/src/visually-hidden/index.ts +6 -6
@@ -1,321 +1,321 @@
1
- /**
2
- * Auto-focus management for solidaria
3
- *
4
- * Provides priority-based auto-focus with deferred execution
5
- * and conflict resolution for multiple auto-focus elements.
6
- */
7
-
8
- import { createEffect, onCleanup, onMount } from 'solid-js';
9
- import { isServer } from 'solid-js/web';
10
- import { focusSafely } from '../utils/focus';
11
-
12
- // ============================================
13
- // TYPES
14
- // ============================================
15
-
16
- export interface AutoFocusOptions {
17
- /**
18
- * Whether auto-focus is enabled.
19
- * @default true
20
- */
21
- isEnabled?: boolean;
22
- /**
23
- * Priority level (higher = more important).
24
- * When multiple elements request auto-focus, the highest priority wins.
25
- * @default 0
26
- */
27
- priority?: number;
28
- /**
29
- * Delay in milliseconds before focusing.
30
- * Useful for animations or transitions.
31
- * @default 0
32
- */
33
- delay?: number;
34
- /**
35
- * Whether to focus even if another element is already focused.
36
- * @default false
37
- */
38
- force?: boolean;
39
- /**
40
- * Whether to prevent scrolling when focusing.
41
- * @default true
42
- */
43
- preventScroll?: boolean;
44
- /**
45
- * Callback when focus is applied.
46
- */
47
- onFocus?: (element: HTMLElement) => void;
48
- /**
49
- * Callback when focus is skipped (due to lower priority or other reasons).
50
- */
51
- onSkip?: () => void;
52
- }
53
-
54
- export interface AutoFocusResult {
55
- /**
56
- * Manually trigger the auto-focus.
57
- */
58
- focus: () => void;
59
- /**
60
- * Cancel any pending auto-focus.
61
- */
62
- cancel: () => void;
63
- }
64
-
65
- // ============================================
66
- // AUTO-FOCUS QUEUE
67
- // ============================================
68
-
69
- interface QueuedFocus {
70
- ref: () => HTMLElement | null | undefined;
71
- priority: number;
72
- delay: number;
73
- force: boolean;
74
- preventScroll: boolean;
75
- onFocus?: (element: HTMLElement) => void;
76
- onSkip?: () => void;
77
- }
78
-
79
- // Global queue for managing auto-focus requests
80
- let autoFocusQueue: QueuedFocus[] = [];
81
- let processingTimeout: ReturnType<typeof setTimeout> | null = null;
82
-
83
- /**
84
- * Process the auto-focus queue and focus the highest priority element.
85
- */
86
- function processAutoFocusQueue(): void {
87
- if (processingTimeout) {
88
- clearTimeout(processingTimeout);
89
- processingTimeout = null;
90
- }
91
-
92
- if (autoFocusQueue.length === 0) return;
93
-
94
- // Sort by priority (highest first)
95
- autoFocusQueue.sort((a, b) => b.priority - a.priority);
96
-
97
- // Get the highest priority item
98
- const winner = autoFocusQueue[0];
99
- const losers = autoFocusQueue.slice(1);
100
-
101
- // Clear the queue
102
- autoFocusQueue = [];
103
-
104
- // Notify losers
105
- for (const loser of losers) {
106
- loser.onSkip?.();
107
- }
108
-
109
- // Focus the winner
110
- const element = winner.ref();
111
- if (!element) {
112
- winner.onSkip?.();
113
- return;
114
- }
115
-
116
- // Check if we should focus
117
- const activeElement = document.activeElement;
118
- const shouldFocus =
119
- winner.force ||
120
- !activeElement ||
121
- activeElement === document.body ||
122
- activeElement === document.documentElement;
123
-
124
- if (!shouldFocus) {
125
- winner.onSkip?.();
126
- return;
127
- }
128
-
129
- // Apply focus with optional delay
130
- if (winner.delay > 0) {
131
- setTimeout(() => {
132
- const el = winner.ref();
133
- if (el && document.body.contains(el)) {
134
- if (winner.preventScroll) {
135
- focusSafely(el);
136
- } else {
137
- el.focus();
138
- }
139
- winner.onFocus?.(el);
140
- }
141
- }, winner.delay);
142
- } else {
143
- if (winner.preventScroll) {
144
- focusSafely(element);
145
- } else {
146
- element.focus();
147
- }
148
- winner.onFocus?.(element);
149
- }
150
- }
151
-
152
- /**
153
- * Queue an element for auto-focus.
154
- */
155
- function queueAutoFocus(item: QueuedFocus): void {
156
- autoFocusQueue.push(item);
157
-
158
- // Schedule processing on next frame to allow all components to register
159
- if (processingTimeout === null) {
160
- processingTimeout = setTimeout(processAutoFocusQueue, 0);
161
- }
162
- }
163
-
164
- /**
165
- * Remove an item from the auto-focus queue.
166
- */
167
- function removeFromQueue(ref: () => HTMLElement | null | undefined): void {
168
- autoFocusQueue = autoFocusQueue.filter((item) => item.ref !== ref);
169
- }
170
-
171
- // ============================================
172
- // HOOK
173
- // ============================================
174
-
175
- /**
176
- * Creates auto-focus behavior for an element.
177
- *
178
- * This hook registers the element for auto-focus when mounted. If multiple
179
- * elements request auto-focus, the one with the highest priority wins.
180
- *
181
- * @param ref - Accessor for the element to focus
182
- * @param options - Auto-focus options
183
- *
184
- * @example
185
- * ```tsx
186
- * function Dialog(props) {
187
- * let contentRef: HTMLDivElement | undefined;
188
- *
189
- * createAutoFocus(() => contentRef, {
190
- * priority: 10, // High priority for dialogs
191
- * onFocus: () => console.log('Dialog focused'),
192
- * });
193
- *
194
- * return (
195
- * <div ref={contentRef} tabIndex={-1}>
196
- * {props.children}
197
- * </div>
198
- * );
199
- * }
200
- * ```
201
- *
202
- * @example
203
- * ```tsx
204
- * // With delay for animations
205
- * function AnimatedPanel() {
206
- * let panelRef: HTMLDivElement | undefined;
207
- *
208
- * createAutoFocus(() => panelRef, {
209
- * delay: 300, // Wait for animation
210
- * });
211
- *
212
- * return <div ref={panelRef} class="animated-panel">...</div>;
213
- * }
214
- * ```
215
- *
216
- * @example
217
- * ```tsx
218
- * // Conditional auto-focus
219
- * function Input(props) {
220
- * let inputRef: HTMLInputElement | undefined;
221
- *
222
- * createAutoFocus(() => inputRef, {
223
- * isEnabled: props.autoFocus,
224
- * });
225
- *
226
- * return <input ref={inputRef} />;
227
- * }
228
- * ```
229
- */
230
- export function createAutoFocus(
231
- ref: () => HTMLElement | null | undefined,
232
- options: AutoFocusOptions = {}
233
- ): AutoFocusResult {
234
- const {
235
- isEnabled = true,
236
- priority = 0,
237
- delay = 0,
238
- force = false,
239
- preventScroll = true,
240
- onFocus,
241
- onSkip,
242
- } = options;
243
-
244
- // During SSR, return no-op functions
245
- if (isServer) {
246
- return {
247
- focus: () => {},
248
- cancel: () => {},
249
- };
250
- }
251
-
252
- let canceled = false;
253
-
254
- // Queue auto-focus on mount
255
- onMount(() => {
256
- if (!isEnabled || canceled) return;
257
-
258
- queueAutoFocus({
259
- ref,
260
- priority,
261
- delay,
262
- force,
263
- preventScroll,
264
- onFocus,
265
- onSkip,
266
- });
267
- });
268
-
269
- // Remove from queue on cleanup
270
- onCleanup(() => {
271
- removeFromQueue(ref);
272
- });
273
-
274
- const focus = (): void => {
275
- if (canceled) return;
276
-
277
- const element = ref();
278
- if (!element) return;
279
-
280
- if (preventScroll) {
281
- focusSafely(element);
282
- } else {
283
- element.focus();
284
- }
285
- onFocus?.(element);
286
- };
287
-
288
- const cancel = (): void => {
289
- canceled = true;
290
- removeFromQueue(ref);
291
- };
292
-
293
- return {
294
- focus,
295
- cancel,
296
- };
297
- }
298
-
299
- // ============================================
300
- // UTILITIES
301
- // ============================================
302
-
303
- /**
304
- * Clears all pending auto-focus requests.
305
- * Useful for testing or when navigating away.
306
- */
307
- export function clearAutoFocusQueue(): void {
308
- if (processingTimeout) {
309
- clearTimeout(processingTimeout);
310
- processingTimeout = null;
311
- }
312
- autoFocusQueue = [];
313
- }
314
-
315
- /**
316
- * Gets the current auto-focus queue length.
317
- * Useful for debugging.
318
- */
319
- export function getAutoFocusQueueLength(): number {
320
- return autoFocusQueue.length;
321
- }
1
+ /**
2
+ * Auto-focus management for solidaria
3
+ *
4
+ * Provides priority-based auto-focus with deferred execution
5
+ * and conflict resolution for multiple auto-focus elements.
6
+ */
7
+
8
+ import { createEffect, onCleanup, onMount } from 'solid-js';
9
+ import { isServer } from 'solid-js/web';
10
+ import { focusSafely } from '../utils/focus';
11
+
12
+ // ============================================
13
+ // TYPES
14
+ // ============================================
15
+
16
+ export interface AutoFocusOptions {
17
+ /**
18
+ * Whether auto-focus is enabled.
19
+ * @default true
20
+ */
21
+ isEnabled?: boolean;
22
+ /**
23
+ * Priority level (higher = more important).
24
+ * When multiple elements request auto-focus, the highest priority wins.
25
+ * @default 0
26
+ */
27
+ priority?: number;
28
+ /**
29
+ * Delay in milliseconds before focusing.
30
+ * Useful for animations or transitions.
31
+ * @default 0
32
+ */
33
+ delay?: number;
34
+ /**
35
+ * Whether to focus even if another element is already focused.
36
+ * @default false
37
+ */
38
+ force?: boolean;
39
+ /**
40
+ * Whether to prevent scrolling when focusing.
41
+ * @default true
42
+ */
43
+ preventScroll?: boolean;
44
+ /**
45
+ * Callback when focus is applied.
46
+ */
47
+ onFocus?: (element: HTMLElement) => void;
48
+ /**
49
+ * Callback when focus is skipped (due to lower priority or other reasons).
50
+ */
51
+ onSkip?: () => void;
52
+ }
53
+
54
+ export interface AutoFocusResult {
55
+ /**
56
+ * Manually trigger the auto-focus.
57
+ */
58
+ focus: () => void;
59
+ /**
60
+ * Cancel any pending auto-focus.
61
+ */
62
+ cancel: () => void;
63
+ }
64
+
65
+ // ============================================
66
+ // AUTO-FOCUS QUEUE
67
+ // ============================================
68
+
69
+ interface QueuedFocus {
70
+ ref: () => HTMLElement | null | undefined;
71
+ priority: number;
72
+ delay: number;
73
+ force: boolean;
74
+ preventScroll: boolean;
75
+ onFocus?: (element: HTMLElement) => void;
76
+ onSkip?: () => void;
77
+ }
78
+
79
+ // Global queue for managing auto-focus requests
80
+ let autoFocusQueue: QueuedFocus[] = [];
81
+ let processingTimeout: ReturnType<typeof setTimeout> | null = null;
82
+
83
+ /**
84
+ * Process the auto-focus queue and focus the highest priority element.
85
+ */
86
+ function processAutoFocusQueue(): void {
87
+ if (processingTimeout) {
88
+ clearTimeout(processingTimeout);
89
+ processingTimeout = null;
90
+ }
91
+
92
+ if (autoFocusQueue.length === 0) return;
93
+
94
+ // Sort by priority (highest first)
95
+ autoFocusQueue.sort((a, b) => b.priority - a.priority);
96
+
97
+ // Get the highest priority item
98
+ const winner = autoFocusQueue[0];
99
+ const losers = autoFocusQueue.slice(1);
100
+
101
+ // Clear the queue
102
+ autoFocusQueue = [];
103
+
104
+ // Notify losers
105
+ for (const loser of losers) {
106
+ loser.onSkip?.();
107
+ }
108
+
109
+ // Focus the winner
110
+ const element = winner.ref();
111
+ if (!element) {
112
+ winner.onSkip?.();
113
+ return;
114
+ }
115
+
116
+ // Check if we should focus
117
+ const activeElement = document.activeElement;
118
+ const shouldFocus =
119
+ winner.force ||
120
+ !activeElement ||
121
+ activeElement === document.body ||
122
+ activeElement === document.documentElement;
123
+
124
+ if (!shouldFocus) {
125
+ winner.onSkip?.();
126
+ return;
127
+ }
128
+
129
+ // Apply focus with optional delay
130
+ if (winner.delay > 0) {
131
+ setTimeout(() => {
132
+ const el = winner.ref();
133
+ if (el && document.body.contains(el)) {
134
+ if (winner.preventScroll) {
135
+ focusSafely(el);
136
+ } else {
137
+ el.focus();
138
+ }
139
+ winner.onFocus?.(el);
140
+ }
141
+ }, winner.delay);
142
+ } else {
143
+ if (winner.preventScroll) {
144
+ focusSafely(element);
145
+ } else {
146
+ element.focus();
147
+ }
148
+ winner.onFocus?.(element);
149
+ }
150
+ }
151
+
152
+ /**
153
+ * Queue an element for auto-focus.
154
+ */
155
+ function queueAutoFocus(item: QueuedFocus): void {
156
+ autoFocusQueue.push(item);
157
+
158
+ // Schedule processing on next frame to allow all components to register
159
+ if (processingTimeout === null) {
160
+ processingTimeout = setTimeout(processAutoFocusQueue, 0);
161
+ }
162
+ }
163
+
164
+ /**
165
+ * Remove an item from the auto-focus queue.
166
+ */
167
+ function removeFromQueue(ref: () => HTMLElement | null | undefined): void {
168
+ autoFocusQueue = autoFocusQueue.filter((item) => item.ref !== ref);
169
+ }
170
+
171
+ // ============================================
172
+ // HOOK
173
+ // ============================================
174
+
175
+ /**
176
+ * Creates auto-focus behavior for an element.
177
+ *
178
+ * This hook registers the element for auto-focus when mounted. If multiple
179
+ * elements request auto-focus, the one with the highest priority wins.
180
+ *
181
+ * @param ref - Accessor for the element to focus
182
+ * @param options - Auto-focus options
183
+ *
184
+ * @example
185
+ * ```tsx
186
+ * function Dialog(props) {
187
+ * let contentRef: HTMLDivElement | undefined;
188
+ *
189
+ * createAutoFocus(() => contentRef, {
190
+ * priority: 10, // High priority for dialogs
191
+ * onFocus: () => console.log('Dialog focused'),
192
+ * });
193
+ *
194
+ * return (
195
+ * <div ref={contentRef} tabIndex={-1}>
196
+ * {props.children}
197
+ * </div>
198
+ * );
199
+ * }
200
+ * ```
201
+ *
202
+ * @example
203
+ * ```tsx
204
+ * // With delay for animations
205
+ * function AnimatedPanel() {
206
+ * let panelRef: HTMLDivElement | undefined;
207
+ *
208
+ * createAutoFocus(() => panelRef, {
209
+ * delay: 300, // Wait for animation
210
+ * });
211
+ *
212
+ * return <div ref={panelRef} class="animated-panel">...</div>;
213
+ * }
214
+ * ```
215
+ *
216
+ * @example
217
+ * ```tsx
218
+ * // Conditional auto-focus
219
+ * function Input(props) {
220
+ * let inputRef: HTMLInputElement | undefined;
221
+ *
222
+ * createAutoFocus(() => inputRef, {
223
+ * isEnabled: props.autoFocus,
224
+ * });
225
+ *
226
+ * return <input ref={inputRef} />;
227
+ * }
228
+ * ```
229
+ */
230
+ export function createAutoFocus(
231
+ ref: () => HTMLElement | null | undefined,
232
+ options: AutoFocusOptions = {}
233
+ ): AutoFocusResult {
234
+ const {
235
+ isEnabled = true,
236
+ priority = 0,
237
+ delay = 0,
238
+ force = false,
239
+ preventScroll = true,
240
+ onFocus,
241
+ onSkip,
242
+ } = options;
243
+
244
+ // During SSR, return no-op functions
245
+ if (isServer) {
246
+ return {
247
+ focus: () => {},
248
+ cancel: () => {},
249
+ };
250
+ }
251
+
252
+ let canceled = false;
253
+
254
+ // Queue auto-focus on mount
255
+ onMount(() => {
256
+ if (!isEnabled || canceled) return;
257
+
258
+ queueAutoFocus({
259
+ ref,
260
+ priority,
261
+ delay,
262
+ force,
263
+ preventScroll,
264
+ onFocus,
265
+ onSkip,
266
+ });
267
+ });
268
+
269
+ // Remove from queue on cleanup
270
+ onCleanup(() => {
271
+ removeFromQueue(ref);
272
+ });
273
+
274
+ const focus = (): void => {
275
+ if (canceled) return;
276
+
277
+ const element = ref();
278
+ if (!element) return;
279
+
280
+ if (preventScroll) {
281
+ focusSafely(element);
282
+ } else {
283
+ element.focus();
284
+ }
285
+ onFocus?.(element);
286
+ };
287
+
288
+ const cancel = (): void => {
289
+ canceled = true;
290
+ removeFromQueue(ref);
291
+ };
292
+
293
+ return {
294
+ focus,
295
+ cancel,
296
+ };
297
+ }
298
+
299
+ // ============================================
300
+ // UTILITIES
301
+ // ============================================
302
+
303
+ /**
304
+ * Clears all pending auto-focus requests.
305
+ * Useful for testing or when navigating away.
306
+ */
307
+ export function clearAutoFocusQueue(): void {
308
+ if (processingTimeout) {
309
+ clearTimeout(processingTimeout);
310
+ processingTimeout = null;
311
+ }
312
+ autoFocusQueue = [];
313
+ }
314
+
315
+ /**
316
+ * Gets the current auto-focus queue length.
317
+ * Useful for debugging.
318
+ */
319
+ export function getAutoFocusQueueLength(): number {
320
+ return autoFocusQueue.length;
321
+ }