@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,243 +1,259 @@
1
- /**
2
- * createDraggableItem - ARIA hook for draggable items within a collection.
3
- *
4
- * Provides accessibility props for items that can be dragged from a collection.
5
- */
6
-
7
- import { createMemo, type Accessor } from 'solid-js';
8
- import type { JSX } from 'solid-js';
9
- import type {
10
- DraggableCollectionState,
11
- DragPreviewRenderer,
12
- } from '@proyecto-viviana/solid-stately';
13
- import {
14
- writeToDataTransfer,
15
- DROP_OPERATION,
16
- EFFECT_ALLOWED,
17
- DROP_EFFECT_TO_DROP_OPERATION,
18
- setGlobalAllowedDropOperations,
19
- setGlobalDropEffect,
20
- getGlobalDropEffect,
21
- } from './utils';
22
-
23
- export interface DraggableItemOptions {
24
- /** The unique key of the item. */
25
- key: string | number;
26
- /** Whether the item has a separate drag button affordance. */
27
- hasDragButton?: boolean;
28
- /** Whether this item is disabled for dragging. */
29
- isDisabled?: boolean;
30
- /** Preview renderer function ref. */
31
- preview?: { current: DragPreviewRenderer | null };
32
- }
33
-
34
- export interface DraggableItemAria {
35
- /** Props for the draggable item element. */
36
- dragProps: JSX.HTMLAttributes<HTMLElement>;
37
- /** Props for the explicit drag button affordance, if any. */
38
- dragButtonProps: JSX.ButtonHTMLAttributes<HTMLButtonElement>;
39
- /** Whether the item is currently being dragged. */
40
- isDragging: boolean;
41
- }
42
-
43
- /**
44
- * Creates ARIA props for a draggable item within a collection.
45
- *
46
- * @param options - Accessor returning item options
47
- * @param state - Draggable collection state
48
- * @returns Draggable item ARIA props
49
- */
50
- export function createDraggableItem(
51
- options: Accessor<DraggableItemOptions>,
52
- state: DraggableCollectionState
53
- ): DraggableItemAria {
54
- const getOptions = createMemo(() => options());
55
-
56
- // Track position for drag move
57
- let lastX = 0;
58
- let lastY = 0;
59
-
60
- const isDragging = createMemo(() => {
61
- const key = getOptions().key;
62
- return state.draggingKeys.has(key);
63
- });
64
-
65
- const getKeysForDrag = (): Set<string | number> => {
66
- const { key } = getOptions();
67
- // If the key is not selected, only drag that item
68
- // If it is selected, drag all selected items
69
- // For now, just return the single key
70
- return new Set([key]);
71
- };
72
-
73
- const onDragStart = (e: DragEvent) => {
74
- if (e.defaultPrevented) return;
75
- e.stopPropagation();
76
-
77
- const opts = getOptions();
78
- if (opts.isDisabled || state.isDisabled) return;
79
-
80
- const keys = getKeysForDrag();
81
-
82
- // Start drag state
83
- state.startDrag(keys, e.clientX, e.clientY);
84
-
85
- // Get items and write to data transfer
86
- const items = state.getItems(keys);
87
- e.dataTransfer?.clearData?.();
88
- if (e.dataTransfer) {
89
- writeToDataTransfer(e.dataTransfer, items);
90
- }
91
-
92
- // Set allowed drop operations
93
- let allowed = DROP_OPERATION.all;
94
- const allowedOps = state.getAllowedDropOperations();
95
- if (allowedOps.length > 0) {
96
- allowed = DROP_OPERATION.none;
97
- for (const op of allowedOps) {
98
- allowed |= DROP_OPERATION[op] || DROP_OPERATION.none;
99
- }
100
- }
101
-
102
- setGlobalAllowedDropOperations(allowed);
103
- const effectAllowed = EFFECT_ALLOWED[allowed] || 'none';
104
- if (e.dataTransfer) {
105
- e.dataTransfer.effectAllowed =
106
- (effectAllowed === 'cancel' ? 'none' : effectAllowed) as DataTransfer['effectAllowed'];
107
- }
108
-
109
- // Handle custom preview
110
- if (typeof opts.preview?.current === 'function' && e.dataTransfer) {
111
- opts.preview.current(items, (node, userX, userY) => {
112
- if (!node || !e.dataTransfer) return;
113
-
114
- const size = node.getBoundingClientRect();
115
- const rect = (e.currentTarget as HTMLElement).getBoundingClientRect();
116
- let defaultX = e.clientX - rect.x;
117
- let defaultY = e.clientY - rect.y;
118
-
119
- if (defaultX > size.width || defaultY > size.height) {
120
- defaultX = size.width / 2;
121
- defaultY = size.height / 2;
122
- }
123
-
124
- let offsetX = typeof userX === 'number' ? userX : defaultX;
125
- let offsetY = typeof userY === 'number' ? userY : defaultY;
126
-
127
- offsetX = Math.max(0, Math.min(offsetX, size.width));
128
- offsetY = Math.max(0, Math.min(offsetY, size.height));
129
-
130
- e.dataTransfer.setDragImage(node, offsetX, offsetY);
131
- });
132
- }
133
-
134
- lastX = e.clientX;
135
- lastY = e.clientY;
136
- };
137
-
138
- const onDrag = (e: DragEvent) => {
139
- e.stopPropagation();
140
-
141
- if (e.clientX === lastX && e.clientY === lastY) {
142
- return;
143
- }
144
-
145
- state.moveDrag(e.clientX, e.clientY);
146
-
147
- lastX = e.clientX;
148
- lastY = e.clientY;
149
- };
150
-
151
- const onDragEnd = (e: DragEvent) => {
152
- e.stopPropagation();
153
-
154
- let dropEffect: string = e.dataTransfer?.dropEffect ?? 'none';
155
- // Chrome Android fix - use global drop effect
156
- if (getGlobalDropEffect()) {
157
- dropEffect = getGlobalDropEffect()!;
158
- }
159
-
160
- const dropOperation = DROP_EFFECT_TO_DROP_OPERATION[dropEffect];
161
- const isInternal = false; // Would check global state
162
- state.endDrag(e.clientX, e.clientY, dropOperation, isInternal);
163
-
164
- setGlobalAllowedDropOperations(DROP_OPERATION.none);
165
- setGlobalDropEffect(undefined);
166
- };
167
-
168
- // Keyboard/screen reader drag initiation
169
- const onKeyDown = (e: KeyboardEvent) => {
170
- if (e.key === 'Enter' && e.target === e.currentTarget) {
171
- e.preventDefault();
172
- e.stopPropagation();
173
- }
174
- };
175
-
176
- const onKeyUp = (e: KeyboardEvent) => {
177
- if (e.key === 'Enter' && e.target === e.currentTarget) {
178
- e.preventDefault();
179
- e.stopPropagation();
180
-
181
- const opts = getOptions();
182
- if (opts.isDisabled || state.isDisabled) return;
183
-
184
- const keys = getKeysForDrag();
185
- const rect = (e.target as HTMLElement).getBoundingClientRect();
186
- state.startDrag(keys, rect.x + rect.width / 2, rect.y + rect.height / 2);
187
- }
188
- };
189
-
190
- const dragProps = createMemo(() => {
191
- const opts = getOptions();
192
-
193
- if (opts.isDisabled || state.isDisabled) {
194
- return {
195
- draggable: false as const,
196
- };
197
- }
198
-
199
- const baseProps: Record<string, unknown> = {
200
- draggable: true as const,
201
- onDragStart,
202
- onDrag,
203
- onDragEnd,
204
- };
205
-
206
- // Add keyboard handlers if no separate drag button
207
- if (!opts.hasDragButton) {
208
- baseProps.onKeyDown = onKeyDown;
209
- baseProps.onKeyUp = onKeyUp;
210
- }
211
-
212
- return baseProps;
213
- });
214
-
215
- const dragButtonProps = createMemo(() => {
216
- const opts = getOptions();
217
-
218
- if (opts.isDisabled || state.isDisabled) {
219
- return {
220
- disabled: true,
221
- };
222
- }
223
-
224
- return {
225
- type: 'button' as const,
226
- 'aria-label': 'Drag',
227
- onKeyDown,
228
- onKeyUp,
229
- };
230
- });
231
-
232
- return {
233
- get dragProps() {
234
- return dragProps() as DraggableItemAria['dragProps'];
235
- },
236
- get dragButtonProps() {
237
- return dragButtonProps() as DraggableItemAria['dragButtonProps'];
238
- },
239
- get isDragging() {
240
- return isDragging();
241
- },
242
- };
243
- }
1
+ /**
2
+ * createDraggableItem - ARIA hook for draggable items within a collection.
3
+ *
4
+ * Provides accessibility props for items that can be dragged from a collection.
5
+ */
6
+
7
+ import { createMemo, type Accessor } from 'solid-js';
8
+ import type { JSX } from 'solid-js';
9
+ import type {
10
+ DraggableCollectionState,
11
+ DragPreviewRenderer,
12
+ } from '@proyecto-viviana/solid-stately';
13
+ import {
14
+ getTypes,
15
+ writeToDataTransfer,
16
+ DROP_OPERATION,
17
+ EFFECT_ALLOWED,
18
+ DROP_EFFECT_TO_DROP_OPERATION,
19
+ setGlobalAllowedDropOperations,
20
+ setGlobalDropEffect,
21
+ getGlobalDropEffect,
22
+ } from './utils';
23
+ import { setGlobalDraggingTypes } from './createDraggableCollection';
24
+
25
+ export interface DraggableItemOptions {
26
+ /** The unique key of the item. */
27
+ key: string | number;
28
+ /** Whether the item has a separate drag button affordance. */
29
+ hasDragButton?: boolean;
30
+ /** Whether this item is disabled for dragging. */
31
+ isDisabled?: boolean;
32
+ /** Preview renderer function ref. */
33
+ preview?: { current: DragPreviewRenderer | null };
34
+ }
35
+
36
+ export interface DraggableItemAria {
37
+ /** Props for the draggable item element. */
38
+ dragProps: JSX.HTMLAttributes<HTMLElement>;
39
+ /** Props for the explicit drag button affordance, if any. */
40
+ dragButtonProps: JSX.ButtonHTMLAttributes<HTMLButtonElement>;
41
+ /** Whether the item is currently being dragged. */
42
+ isDragging: boolean;
43
+ }
44
+
45
+ /**
46
+ * Creates ARIA props for a draggable item within a collection.
47
+ *
48
+ * @param options - Accessor returning item options
49
+ * @param state - Draggable collection state
50
+ * @returns Draggable item ARIA props
51
+ */
52
+ export function createDraggableItem(
53
+ options: Accessor<DraggableItemOptions>,
54
+ state: DraggableCollectionState
55
+ ): DraggableItemAria {
56
+ const getOptions = createMemo(() => options());
57
+
58
+ // Track position for drag move
59
+ let lastX = 0;
60
+ let lastY = 0;
61
+
62
+ const isDragging = createMemo(() => {
63
+ const key = getOptions().key;
64
+ return state.draggingKeys.has(key);
65
+ });
66
+
67
+ const getKeysForDrag = (): Set<string | number> => {
68
+ const { key } = getOptions();
69
+ // If the key is not selected, only drag that item
70
+ // If it is selected, drag all selected items
71
+ // For now, just return the single key
72
+ return new Set([key]);
73
+ };
74
+
75
+ const onDragStart = (e: DragEvent) => {
76
+ if (e.defaultPrevented) return;
77
+ e.stopPropagation();
78
+
79
+ const opts = getOptions();
80
+ if (opts.isDisabled || state.isDisabled) return;
81
+
82
+ const keys = getKeysForDrag();
83
+
84
+ // Start drag state
85
+ state.startDrag(keys, e.clientX, e.clientY);
86
+
87
+ // Get items and write to data transfer
88
+ const items = state.getItems(keys);
89
+ setGlobalDraggingTypes(getTypes(items));
90
+ e.dataTransfer?.clearData?.();
91
+ if (e.dataTransfer) {
92
+ writeToDataTransfer(e.dataTransfer, items);
93
+ }
94
+
95
+ // Set allowed drop operations
96
+ let allowed = DROP_OPERATION.all;
97
+ const allowedOps = state.getAllowedDropOperations();
98
+ if (allowedOps.length > 0) {
99
+ allowed = DROP_OPERATION.none;
100
+ for (const op of allowedOps) {
101
+ allowed |= DROP_OPERATION[op] || DROP_OPERATION.none;
102
+ }
103
+ }
104
+
105
+ setGlobalAllowedDropOperations(allowed);
106
+ const effectAllowed = EFFECT_ALLOWED[allowed] || 'none';
107
+ if (e.dataTransfer) {
108
+ e.dataTransfer.effectAllowed =
109
+ (effectAllowed === 'cancel' ? 'none' : effectAllowed) as DataTransfer['effectAllowed'];
110
+ }
111
+
112
+ // Handle custom preview from item options or collection state.
113
+ const preview = opts.preview ?? state.preview;
114
+ if (typeof preview?.current === 'function' && e.dataTransfer) {
115
+ preview.current(items, (node, userX, userY) => {
116
+ if (!node || !e.dataTransfer) return;
117
+
118
+ const size = node.getBoundingClientRect();
119
+ const rect = (e.currentTarget as HTMLElement).getBoundingClientRect();
120
+ let defaultX = e.clientX - rect.x;
121
+ let defaultY = e.clientY - rect.y;
122
+
123
+ if (defaultX > size.width || defaultY > size.height) {
124
+ defaultX = size.width / 2;
125
+ defaultY = size.height / 2;
126
+ }
127
+
128
+ let offsetX = typeof userX === 'number' ? userX : defaultX;
129
+ let offsetY = typeof userY === 'number' ? userY : defaultY;
130
+
131
+ offsetX = Math.max(0, Math.min(offsetX, size.width));
132
+ offsetY = Math.max(0, Math.min(offsetY, size.height));
133
+
134
+ e.dataTransfer.setDragImage(node, offsetX, offsetY);
135
+ });
136
+ }
137
+
138
+ lastX = e.clientX;
139
+ lastY = e.clientY;
140
+ };
141
+
142
+ const onDrag = (e: DragEvent) => {
143
+ e.stopPropagation();
144
+
145
+ if (e.clientX === lastX && e.clientY === lastY) {
146
+ return;
147
+ }
148
+
149
+ state.moveDrag(e.clientX, e.clientY);
150
+
151
+ lastX = e.clientX;
152
+ lastY = e.clientY;
153
+ };
154
+
155
+ const onDragEnd = (e: DragEvent) => {
156
+ e.stopPropagation();
157
+
158
+ let dropEffect: string = e.dataTransfer?.dropEffect ?? 'none';
159
+ // Chrome Android fix - use global drop effect
160
+ if (getGlobalDropEffect()) {
161
+ dropEffect = getGlobalDropEffect()!;
162
+ }
163
+
164
+ const dropOperation = DROP_EFFECT_TO_DROP_OPERATION[dropEffect];
165
+ const isInternal = false; // Would check global state
166
+ state.endDrag(e.clientX, e.clientY, dropOperation, isInternal);
167
+
168
+ setGlobalAllowedDropOperations(DROP_OPERATION.none);
169
+ setGlobalDraggingTypes(new Set());
170
+ setGlobalDropEffect(undefined);
171
+ };
172
+
173
+ // Keyboard/screen reader drag initiation
174
+ const onKeyDown = (e: KeyboardEvent) => {
175
+ if (e.key === 'Enter' && e.target === e.currentTarget) {
176
+ e.preventDefault();
177
+ e.stopPropagation();
178
+ }
179
+ };
180
+
181
+ const onKeyUp = (e: KeyboardEvent) => {
182
+ if (e.key === 'Enter' && e.target === e.currentTarget) {
183
+ e.preventDefault();
184
+ e.stopPropagation();
185
+
186
+ const opts = getOptions();
187
+ if (opts.isDisabled || state.isDisabled) return;
188
+
189
+ const keys = getKeysForDrag();
190
+ const rect = (e.target as HTMLElement).getBoundingClientRect();
191
+ state.startDrag(keys, rect.x + rect.width / 2, rect.y + rect.height / 2);
192
+ const items = state.getItems(keys);
193
+ setGlobalDraggingTypes(getTypes(items));
194
+ let allowed = DROP_OPERATION.all;
195
+ const allowedOps = state.getAllowedDropOperations();
196
+ if (allowedOps.length > 0) {
197
+ allowed = DROP_OPERATION.none;
198
+ for (const op of allowedOps) {
199
+ allowed |= DROP_OPERATION[op] || DROP_OPERATION.none;
200
+ }
201
+ }
202
+ setGlobalAllowedDropOperations(allowed);
203
+ }
204
+ };
205
+
206
+ const dragProps = createMemo(() => {
207
+ const opts = getOptions();
208
+
209
+ if (opts.isDisabled || state.isDisabled) {
210
+ return {
211
+ draggable: false as const,
212
+ };
213
+ }
214
+
215
+ const baseProps: Record<string, unknown> = {
216
+ draggable: true as const,
217
+ onDragStart,
218
+ onDrag,
219
+ onDragEnd,
220
+ };
221
+
222
+ // Add keyboard handlers if no separate drag button
223
+ if (!opts.hasDragButton) {
224
+ baseProps.onKeyDown = onKeyDown;
225
+ baseProps.onKeyUp = onKeyUp;
226
+ }
227
+
228
+ return baseProps;
229
+ });
230
+
231
+ const dragButtonProps = createMemo(() => {
232
+ const opts = getOptions();
233
+
234
+ if (opts.isDisabled || state.isDisabled) {
235
+ return {
236
+ disabled: true,
237
+ };
238
+ }
239
+
240
+ return {
241
+ type: 'button' as const,
242
+ 'aria-label': 'Drag',
243
+ onKeyDown,
244
+ onKeyUp,
245
+ };
246
+ });
247
+
248
+ return {
249
+ get dragProps() {
250
+ return dragProps() as DraggableItemAria['dragProps'];
251
+ },
252
+ get dragButtonProps() {
253
+ return dragButtonProps() as DraggableItemAria['dragButtonProps'];
254
+ },
255
+ get isDragging() {
256
+ return isDragging();
257
+ },
258
+ };
259
+ }