@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,289 +1,289 @@
1
- /**
2
- * createMove - Handles move interactions across mouse, touch, pointer, and keyboard.
3
- *
4
- * Port of @react-aria/interactions useMove, adapted for SolidJS.
5
- */
6
-
7
- import { JSX, createSignal, createEffect, onCleanup } from 'solid-js';
8
- import { disableTextSelection, restoreTextSelection, createGlobalListeners } from '../utils';
9
-
10
- export type PointerType = 'mouse' | 'pen' | 'touch' | 'keyboard';
11
-
12
- interface BaseMoveEvent {
13
- pointerType: PointerType;
14
- shiftKey: boolean;
15
- ctrlKey: boolean;
16
- metaKey: boolean;
17
- altKey: boolean;
18
- }
19
-
20
- export interface MoveStartEvent extends BaseMoveEvent {
21
- type: 'movestart';
22
- }
23
-
24
- export interface MoveMoveEvent extends BaseMoveEvent {
25
- type: 'move';
26
- deltaX: number;
27
- deltaY: number;
28
- }
29
-
30
- export interface MoveEndEvent extends BaseMoveEvent {
31
- type: 'moveend';
32
- }
33
-
34
- export interface MoveEvents {
35
- onMoveStart?: (e: MoveStartEvent) => void;
36
- onMove?: (e: MoveMoveEvent) => void;
37
- onMoveEnd?: (e: MoveEndEvent) => void;
38
- }
39
-
40
- export interface MoveResult {
41
- /** Props to spread on the target element. */
42
- moveProps: JSX.HTMLAttributes<HTMLElement>;
43
- }
44
-
45
- interface MoveState {
46
- didMove: boolean;
47
- lastPosition: { pageX: number; pageY: number } | null;
48
- id: number | null;
49
- }
50
-
51
- function createBaseEvent(
52
- originalEvent: { shiftKey: boolean; ctrlKey: boolean; metaKey: boolean; altKey: boolean },
53
- pointerType: PointerType
54
- ): BaseMoveEvent {
55
- return {
56
- pointerType,
57
- shiftKey: originalEvent.shiftKey,
58
- ctrlKey: originalEvent.ctrlKey,
59
- metaKey: originalEvent.metaKey,
60
- altKey: originalEvent.altKey,
61
- };
62
- }
63
-
64
- /**
65
- * Handles move interactions across mouse, touch, pointer, and keyboard.
66
- */
67
- export function createMove(props: MoveEvents = {}): MoveResult {
68
- const { onMoveStart, onMove, onMoveEnd } = props;
69
-
70
- const state: MoveState = {
71
- didMove: false,
72
- lastPosition: null,
73
- id: null,
74
- };
75
-
76
- const { addGlobalListener, removeGlobalListener } = createGlobalListeners();
77
- const [pointerDown, setPointerDown] = createSignal<'pointer' | 'mouse' | 'touch' | null>(null);
78
-
79
- const move = (
80
- originalEvent: { shiftKey: boolean; ctrlKey: boolean; metaKey: boolean; altKey: boolean },
81
- pointerType: PointerType,
82
- deltaX: number,
83
- deltaY: number
84
- ) => {
85
- if (deltaX === 0 && deltaY === 0) {
86
- return;
87
- }
88
-
89
- if (!state.didMove) {
90
- state.didMove = true;
91
- onMoveStart?.({
92
- type: 'movestart',
93
- ...createBaseEvent(originalEvent, pointerType),
94
- });
95
- }
96
-
97
- onMove?.({
98
- type: 'move',
99
- deltaX,
100
- deltaY,
101
- ...createBaseEvent(originalEvent, pointerType),
102
- });
103
- };
104
-
105
- const end = (originalEvent: { shiftKey: boolean; ctrlKey: boolean; metaKey: boolean; altKey: boolean }, pointerType: PointerType) => {
106
- restoreTextSelection();
107
- if (state.didMove) {
108
- onMoveEnd?.({
109
- type: 'moveend',
110
- ...createBaseEvent(originalEvent, pointerType),
111
- });
112
- }
113
- };
114
-
115
- createEffect(() => {
116
- const activePointer = pointerDown();
117
- if (!activePointer) return;
118
-
119
- if (activePointer === 'pointer') {
120
- const onPointerMove = (e: PointerEvent) => {
121
- if (e.pointerId === state.id) {
122
- const pointerType = (e.pointerType || 'mouse') as PointerType;
123
- move(e, pointerType, e.pageX - (state.lastPosition?.pageX ?? 0), e.pageY - (state.lastPosition?.pageY ?? 0));
124
- state.lastPosition = { pageX: e.pageX, pageY: e.pageY };
125
- }
126
- };
127
-
128
- const onPointerUp = (e: PointerEvent) => {
129
- if (e.pointerId === state.id) {
130
- const pointerType = (e.pointerType || 'mouse') as PointerType;
131
- end(e, pointerType);
132
- state.id = null;
133
- removeGlobalListener('pointermove', onPointerMove, { isWindow: true });
134
- removeGlobalListener('pointerup', onPointerUp, { isWindow: true });
135
- removeGlobalListener('pointercancel', onPointerUp, { isWindow: true });
136
- setPointerDown(null);
137
- }
138
- };
139
-
140
- addGlobalListener('pointermove', onPointerMove, { isWindow: true });
141
- addGlobalListener('pointerup', onPointerUp, { isWindow: true });
142
- addGlobalListener('pointercancel', onPointerUp, { isWindow: true });
143
- onCleanup(() => {
144
- removeGlobalListener('pointermove', onPointerMove, { isWindow: true });
145
- removeGlobalListener('pointerup', onPointerUp, { isWindow: true });
146
- removeGlobalListener('pointercancel', onPointerUp, { isWindow: true });
147
- });
148
- }
149
-
150
- // Mouse/touch listeners are attached directly in their handlers.
151
- });
152
-
153
- const start = () => {
154
- disableTextSelection();
155
- state.didMove = false;
156
- };
157
-
158
- const moveProps: JSX.HTMLAttributes<HTMLElement> = {};
159
-
160
- const hasPointerEvents = typeof window !== 'undefined' && typeof window.PointerEvent !== 'undefined';
161
-
162
- moveProps.onMouseDown = (e: MouseEvent) => {
163
- if (pointerDown() != null) {
164
- return;
165
- }
166
- if (e.button === 0) {
167
- start();
168
- e.stopPropagation();
169
- e.preventDefault();
170
- state.lastPosition = { pageX: e.pageX, pageY: e.pageY };
171
- setPointerDown('mouse');
172
-
173
- const onMouseMove = (event: MouseEvent) => {
174
- move(
175
- event,
176
- 'mouse',
177
- event.pageX - (state.lastPosition?.pageX ?? 0),
178
- event.pageY - (state.lastPosition?.pageY ?? 0)
179
- );
180
- state.lastPosition = { pageX: event.pageX, pageY: event.pageY };
181
- };
182
- const onMouseUp = (event: MouseEvent) => {
183
- end(event, 'mouse');
184
- removeGlobalListener('mousemove', onMouseMove);
185
- removeGlobalListener('mouseup', onMouseUp);
186
- setPointerDown(null);
187
- };
188
- addGlobalListener('mousemove', onMouseMove);
189
- addGlobalListener('mouseup', onMouseUp);
190
- }
191
- };
192
-
193
- moveProps.onTouchStart = (e: TouchEvent) => {
194
- if (pointerDown() != null || e.changedTouches.length === 0 || state.id != null) {
195
- return;
196
- }
197
- const { pageX, pageY, identifier } = e.changedTouches[0];
198
- start();
199
- e.stopPropagation();
200
- e.preventDefault();
201
- state.lastPosition = { pageX, pageY };
202
- state.id = identifier;
203
- setPointerDown('touch');
204
-
205
- const onTouchMove = (event: TouchEvent) => {
206
- const touchIndex = [...event.changedTouches].findIndex(
207
- ({ identifier: touchId }) => touchId === state.id
208
- );
209
- if (touchIndex >= 0) {
210
- const { pageX: moveX, pageY: moveY } = event.changedTouches[touchIndex];
211
- move(event, 'touch', moveX - (state.lastPosition?.pageX ?? 0), moveY - (state.lastPosition?.pageY ?? 0));
212
- state.lastPosition = { pageX: moveX, pageY: moveY };
213
- }
214
- };
215
- const onTouchEnd = (event: TouchEvent) => {
216
- const touchIndex = [...event.changedTouches].findIndex(
217
- ({ identifier: touchId }) => touchId === state.id
218
- );
219
- if (touchIndex >= 0) {
220
- end(event, 'touch');
221
- state.id = null;
222
- removeGlobalListener('touchmove', onTouchMove);
223
- removeGlobalListener('touchend', onTouchEnd);
224
- removeGlobalListener('touchcancel', onTouchEnd);
225
- setPointerDown(null);
226
- }
227
- };
228
- addGlobalListener('touchmove', onTouchMove);
229
- addGlobalListener('touchend', onTouchEnd);
230
- addGlobalListener('touchcancel', onTouchEnd);
231
- };
232
-
233
- if (hasPointerEvents) {
234
- moveProps.onPointerDown = (e: PointerEvent) => {
235
- const button = e.button ?? 0;
236
- if (button === 0 && state.id == null) {
237
- start();
238
- e.stopPropagation();
239
- e.preventDefault();
240
- state.lastPosition = { pageX: e.pageX, pageY: e.pageY };
241
- state.id = e.pointerId;
242
- setPointerDown('pointer');
243
- }
244
- };
245
- }
246
-
247
- const triggerKeyboardMove = (
248
- e: KeyboardEvent,
249
- deltaX: number,
250
- deltaY: number
251
- ) => {
252
- start();
253
- move(e, 'keyboard', deltaX, deltaY);
254
- end(e, 'keyboard');
255
- };
256
-
257
- moveProps.onKeyDown = (e: KeyboardEvent) => {
258
- switch (e.key) {
259
- case 'Left':
260
- case 'ArrowLeft':
261
- e.preventDefault();
262
- e.stopPropagation();
263
- triggerKeyboardMove(e, -1, 0);
264
- break;
265
- case 'Right':
266
- case 'ArrowRight':
267
- e.preventDefault();
268
- e.stopPropagation();
269
- triggerKeyboardMove(e, 1, 0);
270
- break;
271
- case 'Up':
272
- case 'ArrowUp':
273
- e.preventDefault();
274
- e.stopPropagation();
275
- triggerKeyboardMove(e, 0, -1);
276
- break;
277
- case 'Down':
278
- case 'ArrowDown':
279
- e.preventDefault();
280
- e.stopPropagation();
281
- triggerKeyboardMove(e, 0, 1);
282
- break;
283
- default:
284
- break;
285
- }
286
- };
287
-
288
- return { moveProps };
289
- }
1
+ /**
2
+ * createMove - Handles move interactions across mouse, touch, pointer, and keyboard.
3
+ *
4
+ * Port of @react-aria/interactions useMove, adapted for SolidJS.
5
+ */
6
+
7
+ import { JSX, createSignal, createEffect, onCleanup } from 'solid-js';
8
+ import { disableTextSelection, restoreTextSelection, createGlobalListeners } from '../utils';
9
+
10
+ export type PointerType = 'mouse' | 'pen' | 'touch' | 'keyboard';
11
+
12
+ interface BaseMoveEvent {
13
+ pointerType: PointerType;
14
+ shiftKey: boolean;
15
+ ctrlKey: boolean;
16
+ metaKey: boolean;
17
+ altKey: boolean;
18
+ }
19
+
20
+ export interface MoveStartEvent extends BaseMoveEvent {
21
+ type: 'movestart';
22
+ }
23
+
24
+ export interface MoveMoveEvent extends BaseMoveEvent {
25
+ type: 'move';
26
+ deltaX: number;
27
+ deltaY: number;
28
+ }
29
+
30
+ export interface MoveEndEvent extends BaseMoveEvent {
31
+ type: 'moveend';
32
+ }
33
+
34
+ export interface MoveEvents {
35
+ onMoveStart?: (e: MoveStartEvent) => void;
36
+ onMove?: (e: MoveMoveEvent) => void;
37
+ onMoveEnd?: (e: MoveEndEvent) => void;
38
+ }
39
+
40
+ export interface MoveResult {
41
+ /** Props to spread on the target element. */
42
+ moveProps: JSX.HTMLAttributes<HTMLElement>;
43
+ }
44
+
45
+ interface MoveState {
46
+ didMove: boolean;
47
+ lastPosition: { pageX: number; pageY: number } | null;
48
+ id: number | null;
49
+ }
50
+
51
+ function createBaseEvent(
52
+ originalEvent: { shiftKey: boolean; ctrlKey: boolean; metaKey: boolean; altKey: boolean },
53
+ pointerType: PointerType
54
+ ): BaseMoveEvent {
55
+ return {
56
+ pointerType,
57
+ shiftKey: originalEvent.shiftKey,
58
+ ctrlKey: originalEvent.ctrlKey,
59
+ metaKey: originalEvent.metaKey,
60
+ altKey: originalEvent.altKey,
61
+ };
62
+ }
63
+
64
+ /**
65
+ * Handles move interactions across mouse, touch, pointer, and keyboard.
66
+ */
67
+ export function createMove(props: MoveEvents = {}): MoveResult {
68
+ const { onMoveStart, onMove, onMoveEnd } = props;
69
+
70
+ const state: MoveState = {
71
+ didMove: false,
72
+ lastPosition: null,
73
+ id: null,
74
+ };
75
+
76
+ const { addGlobalListener, removeGlobalListener } = createGlobalListeners();
77
+ const [pointerDown, setPointerDown] = createSignal<'pointer' | 'mouse' | 'touch' | null>(null);
78
+
79
+ const move = (
80
+ originalEvent: { shiftKey: boolean; ctrlKey: boolean; metaKey: boolean; altKey: boolean },
81
+ pointerType: PointerType,
82
+ deltaX: number,
83
+ deltaY: number
84
+ ) => {
85
+ if (deltaX === 0 && deltaY === 0) {
86
+ return;
87
+ }
88
+
89
+ if (!state.didMove) {
90
+ state.didMove = true;
91
+ onMoveStart?.({
92
+ type: 'movestart',
93
+ ...createBaseEvent(originalEvent, pointerType),
94
+ });
95
+ }
96
+
97
+ onMove?.({
98
+ type: 'move',
99
+ deltaX,
100
+ deltaY,
101
+ ...createBaseEvent(originalEvent, pointerType),
102
+ });
103
+ };
104
+
105
+ const end = (originalEvent: { shiftKey: boolean; ctrlKey: boolean; metaKey: boolean; altKey: boolean }, pointerType: PointerType) => {
106
+ restoreTextSelection();
107
+ if (state.didMove) {
108
+ onMoveEnd?.({
109
+ type: 'moveend',
110
+ ...createBaseEvent(originalEvent, pointerType),
111
+ });
112
+ }
113
+ };
114
+
115
+ createEffect(() => {
116
+ const activePointer = pointerDown();
117
+ if (!activePointer) return;
118
+
119
+ if (activePointer === 'pointer') {
120
+ const onPointerMove = (e: PointerEvent) => {
121
+ if (e.pointerId === state.id) {
122
+ const pointerType = (e.pointerType || 'mouse') as PointerType;
123
+ move(e, pointerType, e.pageX - (state.lastPosition?.pageX ?? 0), e.pageY - (state.lastPosition?.pageY ?? 0));
124
+ state.lastPosition = { pageX: e.pageX, pageY: e.pageY };
125
+ }
126
+ };
127
+
128
+ const onPointerUp = (e: PointerEvent) => {
129
+ if (e.pointerId === state.id) {
130
+ const pointerType = (e.pointerType || 'mouse') as PointerType;
131
+ end(e, pointerType);
132
+ state.id = null;
133
+ removeGlobalListener('pointermove', onPointerMove, { isWindow: true });
134
+ removeGlobalListener('pointerup', onPointerUp, { isWindow: true });
135
+ removeGlobalListener('pointercancel', onPointerUp, { isWindow: true });
136
+ setPointerDown(null);
137
+ }
138
+ };
139
+
140
+ addGlobalListener('pointermove', onPointerMove, { isWindow: true });
141
+ addGlobalListener('pointerup', onPointerUp, { isWindow: true });
142
+ addGlobalListener('pointercancel', onPointerUp, { isWindow: true });
143
+ onCleanup(() => {
144
+ removeGlobalListener('pointermove', onPointerMove, { isWindow: true });
145
+ removeGlobalListener('pointerup', onPointerUp, { isWindow: true });
146
+ removeGlobalListener('pointercancel', onPointerUp, { isWindow: true });
147
+ });
148
+ }
149
+
150
+ // Mouse/touch listeners are attached directly in their handlers.
151
+ });
152
+
153
+ const start = () => {
154
+ disableTextSelection();
155
+ state.didMove = false;
156
+ };
157
+
158
+ const moveProps: JSX.HTMLAttributes<HTMLElement> = {};
159
+
160
+ const hasPointerEvents = typeof window !== 'undefined' && typeof window.PointerEvent !== 'undefined';
161
+
162
+ moveProps.onMouseDown = (e: MouseEvent) => {
163
+ if (pointerDown() != null) {
164
+ return;
165
+ }
166
+ if (e.button === 0) {
167
+ start();
168
+ e.stopPropagation();
169
+ e.preventDefault();
170
+ state.lastPosition = { pageX: e.pageX, pageY: e.pageY };
171
+ setPointerDown('mouse');
172
+
173
+ const onMouseMove = (event: MouseEvent) => {
174
+ move(
175
+ event,
176
+ 'mouse',
177
+ event.pageX - (state.lastPosition?.pageX ?? 0),
178
+ event.pageY - (state.lastPosition?.pageY ?? 0)
179
+ );
180
+ state.lastPosition = { pageX: event.pageX, pageY: event.pageY };
181
+ };
182
+ const onMouseUp = (event: MouseEvent) => {
183
+ end(event, 'mouse');
184
+ removeGlobalListener('mousemove', onMouseMove);
185
+ removeGlobalListener('mouseup', onMouseUp);
186
+ setPointerDown(null);
187
+ };
188
+ addGlobalListener('mousemove', onMouseMove);
189
+ addGlobalListener('mouseup', onMouseUp);
190
+ }
191
+ };
192
+
193
+ moveProps.onTouchStart = (e: TouchEvent) => {
194
+ if (pointerDown() != null || e.changedTouches.length === 0 || state.id != null) {
195
+ return;
196
+ }
197
+ const { pageX, pageY, identifier } = e.changedTouches[0];
198
+ start();
199
+ e.stopPropagation();
200
+ e.preventDefault();
201
+ state.lastPosition = { pageX, pageY };
202
+ state.id = identifier;
203
+ setPointerDown('touch');
204
+
205
+ const onTouchMove = (event: TouchEvent) => {
206
+ const touchIndex = [...event.changedTouches].findIndex(
207
+ ({ identifier: touchId }) => touchId === state.id
208
+ );
209
+ if (touchIndex >= 0) {
210
+ const { pageX: moveX, pageY: moveY } = event.changedTouches[touchIndex];
211
+ move(event, 'touch', moveX - (state.lastPosition?.pageX ?? 0), moveY - (state.lastPosition?.pageY ?? 0));
212
+ state.lastPosition = { pageX: moveX, pageY: moveY };
213
+ }
214
+ };
215
+ const onTouchEnd = (event: TouchEvent) => {
216
+ const touchIndex = [...event.changedTouches].findIndex(
217
+ ({ identifier: touchId }) => touchId === state.id
218
+ );
219
+ if (touchIndex >= 0) {
220
+ end(event, 'touch');
221
+ state.id = null;
222
+ removeGlobalListener('touchmove', onTouchMove);
223
+ removeGlobalListener('touchend', onTouchEnd);
224
+ removeGlobalListener('touchcancel', onTouchEnd);
225
+ setPointerDown(null);
226
+ }
227
+ };
228
+ addGlobalListener('touchmove', onTouchMove);
229
+ addGlobalListener('touchend', onTouchEnd);
230
+ addGlobalListener('touchcancel', onTouchEnd);
231
+ };
232
+
233
+ if (hasPointerEvents) {
234
+ moveProps.onPointerDown = (e: PointerEvent) => {
235
+ const button = e.button ?? 0;
236
+ if (button === 0 && state.id == null) {
237
+ start();
238
+ e.stopPropagation();
239
+ e.preventDefault();
240
+ state.lastPosition = { pageX: e.pageX, pageY: e.pageY };
241
+ state.id = e.pointerId;
242
+ setPointerDown('pointer');
243
+ }
244
+ };
245
+ }
246
+
247
+ const triggerKeyboardMove = (
248
+ e: KeyboardEvent,
249
+ deltaX: number,
250
+ deltaY: number
251
+ ) => {
252
+ start();
253
+ move(e, 'keyboard', deltaX, deltaY);
254
+ end(e, 'keyboard');
255
+ };
256
+
257
+ moveProps.onKeyDown = (e: KeyboardEvent) => {
258
+ switch (e.key) {
259
+ case 'Left':
260
+ case 'ArrowLeft':
261
+ e.preventDefault();
262
+ e.stopPropagation();
263
+ triggerKeyboardMove(e, -1, 0);
264
+ break;
265
+ case 'Right':
266
+ case 'ArrowRight':
267
+ e.preventDefault();
268
+ e.stopPropagation();
269
+ triggerKeyboardMove(e, 1, 0);
270
+ break;
271
+ case 'Up':
272
+ case 'ArrowUp':
273
+ e.preventDefault();
274
+ e.stopPropagation();
275
+ triggerKeyboardMove(e, 0, -1);
276
+ break;
277
+ case 'Down':
278
+ case 'ArrowDown':
279
+ e.preventDefault();
280
+ e.stopPropagation();
281
+ triggerKeyboardMove(e, 0, 1);
282
+ break;
283
+ default:
284
+ break;
285
+ }
286
+ };
287
+
288
+ return { moveProps };
289
+ }
@@ -84,8 +84,8 @@ function isPressedValue(isPressed: Accessor<boolean> | boolean | undefined): boo
84
84
  return isPressed ?? false;
85
85
  }
86
86
 
87
- // Symbol to track if a link click was handled by us
88
- const LINK_CLICKED = Symbol('linkClicked');
87
+ // Track which links have been programmatically clicked to avoid double activation
88
+ const linkClickedSet = new WeakSet<HTMLElement>();
89
89
 
90
90
  // CSS for preventing double-tap zoom delay
91
91
  let pressableCSSInjected = false;
@@ -685,12 +685,12 @@ export function createPress(props: CreatePressProps = {}): PressResult {
685
685
 
686
686
  // Handle link activation with non-Enter keys (Space)
687
687
  // Native links only respond to Enter, but we want Space to work too
688
- if (e.key === ' ' && isHTMLAnchorLink(target) && !(target as any)[LINK_CLICKED]) {
689
- (target as any)[LINK_CLICKED] = true;
688
+ if (e.key === ' ' && isHTMLAnchorLink(target) && !linkClickedSet.has(target as HTMLElement)) {
689
+ linkClickedSet.add(target as HTMLElement);
690
690
  openLink(target as HTMLAnchorElement, e);
691
691
  // Clean up the marker
692
692
  setTimeout(() => {
693
- delete (target as any)[LINK_CLICKED];
693
+ linkClickedSet.delete(target as HTMLElement);
694
694
  }, 0);
695
695
  }
696
696