@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,314 +1,380 @@
1
- /**
2
- * createColorArea hook.
3
- *
4
- * Provides ARIA attributes and keyboard/pointer handling for a 2D color area.
5
- */
6
-
7
- import { createMemo, type Accessor } from 'solid-js';
8
- import type { ColorAreaState } from '@proyecto-viviana/solid-stately';
9
- import { createId } from '../ssr';
10
- import type { AriaColorAreaOptions, ColorAreaAria } from './types';
11
-
12
- /**
13
- * Creates ARIA props for a color area.
14
- */
15
- export function createColorArea(
16
- props: Accessor<AriaColorAreaOptions>,
17
- state: Accessor<ColorAreaState>,
18
- areaRef: Accessor<HTMLDivElement | null>
19
- ): ColorAreaAria {
20
- const getProps = () => props();
21
- const getState = () => state();
22
-
23
- // Generate IDs
24
- const xInputId = createId();
25
- const yInputId = createId();
26
-
27
- // Calculate position from pointer event
28
- const getPositionFromEvent = (e: MouseEvent | PointerEvent) => {
29
- const area = areaRef();
30
- if (!area) return null;
31
-
32
- const rect = area.getBoundingClientRect();
33
- const x = Math.max(0, Math.min(1, (e.clientX - rect.left) / rect.width));
34
- const y = Math.max(0, Math.min(1, (e.clientY - rect.top) / rect.height));
35
-
36
- return { x, y };
37
- };
38
-
39
- // Handle pointer down
40
- const onPointerDown = (e: PointerEvent) => {
41
- if (getProps().isDisabled || getState().isDisabled) return;
42
-
43
- const pos = getPositionFromEvent(e);
44
- if (!pos) return;
45
-
46
- getState().setColorFromPoint(pos.x, pos.y);
47
- getState().setDragging(true);
48
-
49
- // Capture pointer for dragging
50
- (e.target as HTMLElement).setPointerCapture?.(e.pointerId);
51
- };
52
-
53
- // Handle pointer move
54
- const onPointerMove = (e: PointerEvent) => {
55
- if (!getState().isDragging) return;
56
-
57
- const pos = getPositionFromEvent(e);
58
- if (!pos) return;
59
-
60
- getState().setColorFromPoint(pos.x, pos.y);
61
- };
62
-
63
- // Handle pointer up
64
- const onPointerUp = (e: PointerEvent) => {
65
- if (getState().isDragging) {
66
- getState().setDragging(false);
67
- (e.target as HTMLElement).releasePointerCapture?.(e.pointerId);
68
- }
69
- };
70
-
71
- // Handle keyboard for X axis
72
- const onKeyDownX = (e: KeyboardEvent) => {
73
- if (getProps().isDisabled || getState().isDisabled) return;
74
-
75
- const s = getState();
76
- let handled = true;
77
-
78
- switch (e.key) {
79
- case 'ArrowRight':
80
- s.incrementX();
81
- break;
82
- case 'ArrowLeft':
83
- s.decrementX();
84
- break;
85
- case 'ArrowUp':
86
- s.incrementY();
87
- break;
88
- case 'ArrowDown':
89
- s.decrementY();
90
- break;
91
- case 'PageUp':
92
- s.incrementY(s.yChannelPageStep);
93
- break;
94
- case 'PageDown':
95
- s.decrementY(s.yChannelPageStep);
96
- break;
97
- case 'Home':
98
- if (e.ctrlKey) {
99
- s.setXValue(0);
100
- s.setYValue(100);
101
- } else {
102
- s.setXValue(0);
103
- }
104
- break;
105
- case 'End':
106
- if (e.ctrlKey) {
107
- s.setXValue(100);
108
- s.setYValue(0);
109
- } else {
110
- s.setXValue(100);
111
- }
112
- break;
113
- default:
114
- handled = false;
115
- }
116
-
117
- if (handled) {
118
- e.preventDefault();
119
- e.stopPropagation();
120
- }
121
- };
122
-
123
- // Handle keyboard for Y axis
124
- const onKeyDownY = (e: KeyboardEvent) => {
125
- if (getProps().isDisabled || getState().isDisabled) return;
126
-
127
- const s = getState();
128
- let handled = true;
129
-
130
- switch (e.key) {
131
- case 'ArrowUp':
132
- s.incrementY();
133
- break;
134
- case 'ArrowDown':
135
- s.decrementY();
136
- break;
137
- case 'ArrowRight':
138
- s.incrementX();
139
- break;
140
- case 'ArrowLeft':
141
- s.decrementX();
142
- break;
143
- case 'PageUp':
144
- s.incrementY(s.yChannelPageStep);
145
- break;
146
- case 'PageDown':
147
- s.decrementY(s.yChannelPageStep);
148
- break;
149
- case 'Home':
150
- s.setYValue(100);
151
- break;
152
- case 'End':
153
- s.setYValue(0);
154
- break;
155
- default:
156
- handled = false;
157
- }
158
-
159
- if (handled) {
160
- e.preventDefault();
161
- e.stopPropagation();
162
- }
163
- };
164
-
165
- // Color area props
166
- const colorAreaProps = createMemo(() => {
167
- const s = getState();
168
- const p = getProps();
169
-
170
- return {
171
- role: 'group' as const,
172
- 'aria-label': p['aria-label'],
173
- 'aria-labelledby': p['aria-labelledby'],
174
- 'aria-describedby': p['aria-describedby'],
175
- 'aria-disabled': s.isDisabled || p.isDisabled || undefined,
176
- onPointerDown,
177
- onPointerMove,
178
- onPointerUp,
179
- style: {
180
- position: 'relative' as const,
181
- 'touch-action': 'none',
182
- },
183
- 'data-disabled': s.isDisabled || p.isDisabled || undefined,
184
- };
185
- });
186
-
187
- // Gradient props (the visual area)
188
- const gradientProps = createMemo(() => {
189
- return {
190
- role: 'presentation' as const,
191
- style: {
192
- width: '100%',
193
- height: '100%',
194
- },
195
- };
196
- });
197
-
198
- // Thumb props
199
- const thumbProps = createMemo(() => {
200
- const s = getState();
201
- const p = getProps();
202
- const pos = s.getThumbPosition();
203
-
204
- return {
205
- role: 'presentation' as const,
206
- style: {
207
- position: 'absolute' as const,
208
- left: `${pos.x * 100}%`,
209
- top: `${pos.y * 100}%`,
210
- transform: 'translate(-50%, -50%)',
211
- 'touch-action': 'none',
212
- },
213
- 'data-dragging': s.isDragging || undefined,
214
- 'data-disabled': s.isDisabled || p.isDisabled || undefined,
215
- };
216
- });
217
-
218
- // X input props (hidden, for accessibility)
219
- const xInputProps = createMemo(() => {
220
- const s = getState();
221
- const p = getProps();
222
- const xRange = s.value.getChannelRange(s.xChannel);
223
- const xName = s.value.getChannelName(s.xChannel, 'en-US');
224
-
225
- return {
226
- type: 'range',
227
- id: xInputId,
228
- 'aria-label': `${xName}`,
229
- 'aria-valuetext': `${xName}: ${s.getXValue()}`,
230
- min: xRange.minValue,
231
- max: xRange.maxValue,
232
- step: xRange.step,
233
- value: s.getXValue(),
234
- disabled: s.isDisabled || p.isDisabled,
235
- onKeyDown: onKeyDownX,
236
- onChange: (e: Event) => {
237
- const target = e.target as HTMLInputElement;
238
- s.setXValue(parseFloat(target.value));
239
- },
240
- onBlur: () => {
241
- if (s.isDragging) {
242
- s.setDragging(false);
243
- }
244
- },
245
- tabIndex: 0,
246
- style: {
247
- position: 'absolute' as const,
248
- width: '1px',
249
- height: '1px',
250
- padding: '0',
251
- margin: '-1px',
252
- overflow: 'hidden',
253
- clip: 'rect(0, 0, 0, 0)',
254
- 'white-space': 'nowrap',
255
- border: '0',
256
- },
257
- };
258
- });
259
-
260
- // Y input props (hidden, for accessibility)
261
- const yInputProps = createMemo(() => {
262
- const s = getState();
263
- const p = getProps();
264
- const yRange = s.value.getChannelRange(s.yChannel);
265
- const yName = s.value.getChannelName(s.yChannel, 'en-US');
266
-
267
- return {
268
- type: 'range',
269
- id: yInputId,
270
- 'aria-label': `${yName}`,
271
- 'aria-valuetext': `${yName}: ${s.getYValue()}`,
272
- min: yRange.minValue,
273
- max: yRange.maxValue,
274
- step: yRange.step,
275
- value: s.getYValue(),
276
- disabled: s.isDisabled || p.isDisabled,
277
- onKeyDown: onKeyDownY,
278
- onChange: (e: Event) => {
279
- const target = e.target as HTMLInputElement;
280
- s.setYValue(parseFloat(target.value));
281
- },
282
- tabIndex: -1, // Only first input is in tab order
283
- style: {
284
- position: 'absolute' as const,
285
- width: '1px',
286
- height: '1px',
287
- padding: '0',
288
- margin: '-1px',
289
- overflow: 'hidden',
290
- clip: 'rect(0, 0, 0, 0)',
291
- 'white-space': 'nowrap',
292
- border: '0',
293
- },
294
- };
295
- });
296
-
297
- return {
298
- get colorAreaProps() {
299
- return colorAreaProps();
300
- },
301
- get gradientProps() {
302
- return gradientProps();
303
- },
304
- get thumbProps() {
305
- return thumbProps();
306
- },
307
- get xInputProps() {
308
- return xInputProps();
309
- },
310
- get yInputProps() {
311
- return yInputProps();
312
- },
313
- };
314
- }
1
+ /**
2
+ * createColorArea hook.
3
+ *
4
+ * Provides ARIA attributes and keyboard/pointer handling for a 2D color area.
5
+ */
6
+
7
+ import { createMemo, type Accessor } from 'solid-js';
8
+ import type { Color, ColorAreaState, ColorChannel } from '@proyecto-viviana/solid-stately';
9
+ import { parseColor } from '@proyecto-viviana/solid-stately';
10
+ import { createId } from '../ssr';
11
+ import type { AriaColorAreaOptions, ColorAreaAria } from './types';
12
+
13
+ /**
14
+ * Creates ARIA props for a color area.
15
+ */
16
+ export function createColorArea(
17
+ props: Accessor<AriaColorAreaOptions>,
18
+ state: Accessor<ColorAreaState>,
19
+ areaRef: Accessor<HTMLDivElement | null>
20
+ ): ColorAreaAria {
21
+ const getProps = () => props();
22
+ const getState = () => state();
23
+
24
+ // Generate IDs
25
+ const xInputId = createId();
26
+ const yInputId = createId();
27
+
28
+ // Calculate position from pointer event
29
+ const getPositionFromEvent = (e: MouseEvent | PointerEvent) => {
30
+ const area = areaRef();
31
+ if (!area) return null;
32
+
33
+ const rect = area.getBoundingClientRect();
34
+ const x = Math.max(0, Math.min(1, (e.clientX - rect.left) / rect.width));
35
+ const y = Math.max(0, Math.min(1, (e.clientY - rect.top) / rect.height));
36
+
37
+ return { x, y };
38
+ };
39
+
40
+ // Handle pointer down
41
+ const onPointerDown = (e: PointerEvent) => {
42
+ if (getProps().isDisabled || getState().isDisabled) return;
43
+
44
+ const pos = getPositionFromEvent(e);
45
+ if (!pos) return;
46
+
47
+ getState().setColorFromPoint(pos.x, pos.y);
48
+ getState().setDragging(true);
49
+
50
+ // Capture pointer for dragging
51
+ (e.target as HTMLElement).setPointerCapture?.(e.pointerId);
52
+ };
53
+
54
+ // Handle pointer move
55
+ const onPointerMove = (e: PointerEvent) => {
56
+ if (!getState().isDragging) return;
57
+
58
+ const pos = getPositionFromEvent(e);
59
+ if (!pos) return;
60
+
61
+ getState().setColorFromPoint(pos.x, pos.y);
62
+ };
63
+
64
+ // Handle pointer up
65
+ const onPointerUp = (e: PointerEvent) => {
66
+ if (getState().isDragging) {
67
+ getState().setDragging(false);
68
+ (e.target as HTMLElement).releasePointerCapture?.(e.pointerId);
69
+ }
70
+ };
71
+
72
+ // Handle keyboard for X axis
73
+ const onKeyDownX = (e: KeyboardEvent) => {
74
+ if (getProps().isDisabled || getState().isDisabled) return;
75
+
76
+ const s = getState();
77
+ let handled = true;
78
+
79
+ switch (e.key) {
80
+ case 'ArrowRight':
81
+ s.incrementX();
82
+ break;
83
+ case 'ArrowLeft':
84
+ s.decrementX();
85
+ break;
86
+ case 'ArrowUp':
87
+ s.incrementY();
88
+ break;
89
+ case 'ArrowDown':
90
+ s.decrementY();
91
+ break;
92
+ case 'PageUp':
93
+ s.incrementY(s.yChannelPageStep);
94
+ break;
95
+ case 'PageDown':
96
+ s.decrementY(s.yChannelPageStep);
97
+ break;
98
+ case 'Home':
99
+ if (e.ctrlKey) {
100
+ s.setXValue(0);
101
+ s.setYValue(100);
102
+ } else {
103
+ s.setXValue(0);
104
+ }
105
+ break;
106
+ case 'End':
107
+ if (e.ctrlKey) {
108
+ s.setXValue(100);
109
+ s.setYValue(0);
110
+ } else {
111
+ s.setXValue(100);
112
+ }
113
+ break;
114
+ default:
115
+ handled = false;
116
+ }
117
+
118
+ if (handled) {
119
+ e.preventDefault();
120
+ e.stopPropagation();
121
+ }
122
+ };
123
+
124
+ // Handle keyboard for Y axis
125
+ const onKeyDownY = (e: KeyboardEvent) => {
126
+ if (getProps().isDisabled || getState().isDisabled) return;
127
+
128
+ const s = getState();
129
+ let handled = true;
130
+
131
+ switch (e.key) {
132
+ case 'ArrowUp':
133
+ s.incrementY();
134
+ break;
135
+ case 'ArrowDown':
136
+ s.decrementY();
137
+ break;
138
+ case 'ArrowRight':
139
+ s.incrementX();
140
+ break;
141
+ case 'ArrowLeft':
142
+ s.decrementX();
143
+ break;
144
+ case 'PageUp':
145
+ s.incrementY(s.yChannelPageStep);
146
+ break;
147
+ case 'PageDown':
148
+ s.decrementY(s.yChannelPageStep);
149
+ break;
150
+ case 'Home':
151
+ s.setYValue(100);
152
+ break;
153
+ case 'End':
154
+ s.setYValue(0);
155
+ break;
156
+ default:
157
+ handled = false;
158
+ }
159
+
160
+ if (handled) {
161
+ e.preventDefault();
162
+ e.stopPropagation();
163
+ }
164
+ };
165
+
166
+ // Color area props
167
+ const colorAreaProps = createMemo(() => {
168
+ const s = getState();
169
+ const p = getProps();
170
+
171
+ return {
172
+ role: 'group' as const,
173
+ 'aria-label': p['aria-label'],
174
+ 'aria-labelledby': p['aria-labelledby'],
175
+ 'aria-describedby': p['aria-describedby'],
176
+ 'aria-disabled': s.isDisabled || p.isDisabled || undefined,
177
+ onPointerDown,
178
+ onPointerMove,
179
+ onPointerUp,
180
+ style: {
181
+ position: 'relative' as const,
182
+ 'touch-action': 'none',
183
+ },
184
+ 'data-disabled': s.isDisabled || p.isDisabled || undefined,
185
+ };
186
+ });
187
+
188
+ // Generate gradient background for the color area
189
+ const generateGradient = (): Record<string, string> => {
190
+ const s = getState();
191
+ const value = s.value;
192
+ const xCh = s.xChannel;
193
+ const yCh = s.yChannel;
194
+ const zCh = s.zChannel;
195
+ const zValue = value.getChannelValue(zCh);
196
+ const end = 'right';
197
+
198
+ const hue = (color: Color) =>
199
+ [0, 60, 120, 180, 240, 300, 360].map(h => color.withChannelValue('hue', h).toString('css')).join(', ');
200
+
201
+ const hslChannels: Record<string, (c: Color) => string> = {
202
+ hue,
203
+ saturation: (color) => `${color.withChannelValue('saturation', 0).toString('css')}, transparent`,
204
+ lightness: () => 'black, transparent, white',
205
+ };
206
+
207
+ const hsbChannels: Record<string, (c: Color) => string> = {
208
+ hue,
209
+ saturation: (color) => `${color.withChannelValue('saturation', 0).toString('css')}, transparent`,
210
+ brightness: () => 'black, transparent',
211
+ };
212
+
213
+ switch (value.getColorSpace()) {
214
+ case 'rgb': {
215
+ const rgb = parseColor('rgb(0, 0, 0)');
216
+ return {
217
+ background: [
218
+ `linear-gradient(to ${end}, ${rgb.withChannelValue(xCh, 0).toString('css')}, ${rgb.withChannelValue(xCh, 255).toString('css')})`,
219
+ `linear-gradient(to top, ${rgb.withChannelValue(yCh, 0).toString('css')}, ${rgb.withChannelValue(yCh, 255).toString('css')})`,
220
+ rgb.withChannelValue(zCh, zValue).toString('css'),
221
+ ].join(','),
222
+ 'background-blend-mode': 'screen',
223
+ };
224
+ }
225
+ case 'hsl': {
226
+ const channels = value.getColorChannels();
227
+ const base = parseColor('hsl(0, 100%, 50%)').withChannelValue(zCh, zValue);
228
+ const bg = channels
229
+ .filter((c: ColorChannel) => c !== zCh)
230
+ .map((c: ColorChannel) => `linear-gradient(to ${c === xCh ? end : 'top'}, ${hslChannels[c]?.(base) ?? ''})`)
231
+ .reverse();
232
+ if (zCh === 'hue') bg.push(base.toString('css'));
233
+ return { background: bg.join(', ') };
234
+ }
235
+ case 'hsb': {
236
+ const channels = value.getColorChannels();
237
+ const base = parseColor('hsb(0, 100%, 100%)').withChannelValue(zCh, zValue);
238
+ const bg = channels
239
+ .filter((c: ColorChannel) => c !== zCh)
240
+ .map((c: ColorChannel) => `linear-gradient(to ${c === xCh ? end : 'top'}, ${hsbChannels[c]?.(base) ?? ''})`)
241
+ .reverse();
242
+ if (zCh === 'hue') bg.push(base.toString('css'));
243
+ return { background: bg.join(', ') };
244
+ }
245
+ default:
246
+ return {};
247
+ }
248
+ };
249
+
250
+ // Gradient props (the visual area)
251
+ const gradientProps = createMemo(() => {
252
+ const gradientStyles = generateGradient();
253
+ return {
254
+ role: 'presentation' as const,
255
+ style: {
256
+ width: '100%',
257
+ height: '100%',
258
+ 'forced-color-adjust': 'none' as const,
259
+ ...gradientStyles,
260
+ },
261
+ };
262
+ });
263
+
264
+ // Thumb props
265
+ const thumbProps = createMemo(() => {
266
+ const s = getState();
267
+ const p = getProps();
268
+ const pos = s.getThumbPosition();
269
+
270
+ return {
271
+ role: 'presentation' as const,
272
+ style: {
273
+ position: 'absolute' as const,
274
+ left: `${pos.x * 100}%`,
275
+ top: `${pos.y * 100}%`,
276
+ transform: 'translate(-50%, -50%)',
277
+ 'touch-action': 'none',
278
+ },
279
+ 'data-dragging': s.isDragging || undefined,
280
+ 'data-disabled': s.isDisabled || p.isDisabled || undefined,
281
+ };
282
+ });
283
+
284
+ // X input props (hidden, for accessibility)
285
+ const xInputProps = createMemo(() => {
286
+ const s = getState();
287
+ const p = getProps();
288
+ const xRange = s.value.getChannelRange(s.xChannel);
289
+ const xName = s.value.getChannelName(s.xChannel, 'en-US');
290
+
291
+ return {
292
+ type: 'range',
293
+ id: xInputId,
294
+ 'aria-label': `${xName}`,
295
+ 'aria-valuetext': `${xName}: ${s.getXValue()}`,
296
+ min: xRange.minValue,
297
+ max: xRange.maxValue,
298
+ step: xRange.step,
299
+ value: s.getXValue(),
300
+ disabled: s.isDisabled || p.isDisabled,
301
+ onKeyDown: onKeyDownX,
302
+ onChange: (e: Event) => {
303
+ const target = e.target as HTMLInputElement;
304
+ s.setXValue(parseFloat(target.value));
305
+ },
306
+ onBlur: () => {
307
+ if (s.isDragging) {
308
+ s.setDragging(false);
309
+ }
310
+ },
311
+ tabIndex: 0,
312
+ style: {
313
+ position: 'absolute' as const,
314
+ width: '1px',
315
+ height: '1px',
316
+ padding: '0',
317
+ margin: '-1px',
318
+ overflow: 'hidden',
319
+ clip: 'rect(0, 0, 0, 0)',
320
+ 'white-space': 'nowrap',
321
+ border: '0',
322
+ },
323
+ };
324
+ });
325
+
326
+ // Y input props (hidden, for accessibility)
327
+ const yInputProps = createMemo(() => {
328
+ const s = getState();
329
+ const p = getProps();
330
+ const yRange = s.value.getChannelRange(s.yChannel);
331
+ const yName = s.value.getChannelName(s.yChannel, 'en-US');
332
+
333
+ return {
334
+ type: 'range',
335
+ id: yInputId,
336
+ 'aria-label': `${yName}`,
337
+ 'aria-valuetext': `${yName}: ${s.getYValue()}`,
338
+ min: yRange.minValue,
339
+ max: yRange.maxValue,
340
+ step: yRange.step,
341
+ value: s.getYValue(),
342
+ disabled: s.isDisabled || p.isDisabled,
343
+ onKeyDown: onKeyDownY,
344
+ onChange: (e: Event) => {
345
+ const target = e.target as HTMLInputElement;
346
+ s.setYValue(parseFloat(target.value));
347
+ },
348
+ tabIndex: -1, // Only first input is in tab order
349
+ style: {
350
+ position: 'absolute' as const,
351
+ width: '1px',
352
+ height: '1px',
353
+ padding: '0',
354
+ margin: '-1px',
355
+ overflow: 'hidden',
356
+ clip: 'rect(0, 0, 0, 0)',
357
+ 'white-space': 'nowrap',
358
+ border: '0',
359
+ },
360
+ };
361
+ });
362
+
363
+ return {
364
+ get colorAreaProps() {
365
+ return colorAreaProps();
366
+ },
367
+ get gradientProps() {
368
+ return gradientProps();
369
+ },
370
+ get thumbProps() {
371
+ return thumbProps();
372
+ },
373
+ get xInputProps() {
374
+ return xInputProps();
375
+ },
376
+ get yInputProps() {
377
+ return yInputProps();
378
+ },
379
+ };
380
+ }