@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,79 +1,79 @@
1
- /**
2
- * createTooltip hook for Solidaria
3
- *
4
- * Provides the accessibility implementation for a Tooltip component.
5
- *
6
- * Port of @react-aria/tooltip useTooltip.
7
- */
8
-
9
- import { type JSX } from 'solid-js';
10
- import { type TooltipTriggerState } from '@proyecto-viviana/solid-stately';
11
- import { createHover } from '../interactions/createHover';
12
- import { filterDOMProps, mergeProps } from '../utils';
13
-
14
- // ============================================
15
- // TYPES
16
- // ============================================
17
-
18
- export interface TooltipProps {
19
- /** Whether the tooltip is disabled. */
20
- isDisabled?: boolean;
21
- /** Custom aria-label for the tooltip. */
22
- 'aria-label'?: string;
23
- /** ID of an element that labels the tooltip. */
24
- 'aria-labelledby'?: string;
25
- /** ID of an element that describes the tooltip. */
26
- 'aria-describedby'?: string;
27
- }
28
-
29
- export interface TooltipAria {
30
- /** Props to spread on the tooltip element. */
31
- tooltipProps: JSX.HTMLAttributes<HTMLElement>;
32
- }
33
-
34
- // ============================================
35
- // IMPLEMENTATION
36
- // ============================================
37
-
38
- /**
39
- * Provides the accessibility implementation for a Tooltip component.
40
- *
41
- * When hovering over the tooltip itself, it stays open. When the mouse leaves
42
- * the tooltip, it closes.
43
- *
44
- * @example
45
- * ```tsx
46
- * import { createTooltip } from 'solidaria';
47
- * import { createTooltipTriggerState } from 'solid-stately';
48
- *
49
- * function Tooltip(props) {
50
- * const state = props.state;
51
- * const { tooltipProps } = createTooltip(props, state);
52
- *
53
- * return (
54
- * <div {...tooltipProps} role="tooltip">
55
- * {props.children}
56
- * </div>
57
- * );
58
- * }
59
- * ```
60
- */
61
- export function createTooltip(
62
- props: TooltipProps = {},
63
- state?: TooltipTriggerState
64
- ): TooltipAria {
65
- const domProps = filterDOMProps(props, { labelable: true });
66
-
67
- const { hoverProps } = createHover({
68
- onHoverStart: () => state?.open(true),
69
- onHoverEnd: () => state?.close(),
70
- });
71
-
72
- return {
73
- tooltipProps: mergeProps<JSX.HTMLAttributes<HTMLElement>>(
74
- domProps,
75
- hoverProps,
76
- { role: 'tooltip' }
77
- ),
78
- };
79
- }
1
+ /**
2
+ * createTooltip hook for Solidaria
3
+ *
4
+ * Provides the accessibility implementation for a Tooltip component.
5
+ *
6
+ * Port of @react-aria/tooltip useTooltip.
7
+ */
8
+
9
+ import { type JSX } from 'solid-js';
10
+ import { type TooltipTriggerState } from '@proyecto-viviana/solid-stately';
11
+ import { createHover } from '../interactions/createHover';
12
+ import { filterDOMProps, mergeProps } from '../utils';
13
+
14
+ // ============================================
15
+ // TYPES
16
+ // ============================================
17
+
18
+ export interface TooltipProps {
19
+ /** Whether the tooltip is disabled. */
20
+ isDisabled?: boolean;
21
+ /** Custom aria-label for the tooltip. */
22
+ 'aria-label'?: string;
23
+ /** ID of an element that labels the tooltip. */
24
+ 'aria-labelledby'?: string;
25
+ /** ID of an element that describes the tooltip. */
26
+ 'aria-describedby'?: string;
27
+ }
28
+
29
+ export interface TooltipAria {
30
+ /** Props to spread on the tooltip element. */
31
+ tooltipProps: JSX.HTMLAttributes<HTMLElement>;
32
+ }
33
+
34
+ // ============================================
35
+ // IMPLEMENTATION
36
+ // ============================================
37
+
38
+ /**
39
+ * Provides the accessibility implementation for a Tooltip component.
40
+ *
41
+ * When hovering over the tooltip itself, it stays open. When the mouse leaves
42
+ * the tooltip, it closes.
43
+ *
44
+ * @example
45
+ * ```tsx
46
+ * import { createTooltip } from 'solidaria';
47
+ * import { createTooltipTriggerState } from 'solid-stately';
48
+ *
49
+ * function Tooltip(props) {
50
+ * const state = props.state;
51
+ * const { tooltipProps } = createTooltip(props, state);
52
+ *
53
+ * return (
54
+ * <div {...tooltipProps} role="tooltip">
55
+ * {props.children}
56
+ * </div>
57
+ * );
58
+ * }
59
+ * ```
60
+ */
61
+ export function createTooltip(
62
+ props: TooltipProps = {},
63
+ state?: TooltipTriggerState
64
+ ): TooltipAria {
65
+ const domProps = filterDOMProps(props, { labelable: true });
66
+
67
+ const { hoverProps } = createHover({
68
+ onHoverStart: () => state?.open(true),
69
+ onHoverEnd: () => state?.close(),
70
+ });
71
+
72
+ return {
73
+ tooltipProps: mergeProps<JSX.HTMLAttributes<HTMLElement>>(
74
+ domProps,
75
+ hoverProps,
76
+ { role: 'tooltip' }
77
+ ),
78
+ };
79
+ }
@@ -1,222 +1,226 @@
1
- /**
2
- * createTooltipTrigger hook for Solidaria
3
- *
4
- * Provides the behavior and accessibility implementation for a tooltip trigger,
5
- * e.g. a button that shows a description when focused or hovered.
6
- *
7
- * Port of @react-aria/tooltip useTooltipTrigger.
8
- */
9
-
10
- import { type JSX, createEffect, onCleanup } from 'solid-js';
11
- import { type TooltipTriggerState } from '@proyecto-viviana/solid-stately';
12
- import { createHover } from '../interactions/createHover';
13
- import { createFocusable } from '../interactions/createFocusable';
14
- import { mergeProps } from '../utils';
15
- import { createId } from '../ssr';
16
-
17
- // ============================================
18
- // TYPES
19
- // ============================================
20
-
21
- export interface TooltipTriggerProps {
22
- /** Whether the tooltip should be disabled. */
23
- isDisabled?: boolean;
24
- /**
25
- * The trigger mechanism for the tooltip.
26
- * @default 'focus'
27
- */
28
- trigger?: 'focus';
29
- /**
30
- * Whether the tooltip should close when the trigger is pressed.
31
- * @default true
32
- */
33
- shouldCloseOnPress?: boolean;
34
- }
35
-
36
- export interface TooltipTriggerAria {
37
- /** Props to spread on the trigger element. */
38
- triggerProps: JSX.HTMLAttributes<HTMLElement>;
39
- /** Props to spread on the tooltip element (id for accessibility). */
40
- tooltipProps: { id: string };
41
- }
42
-
43
- // ============================================
44
- // GLOBAL STATE
45
- // ============================================
46
-
47
- type Modality = 'keyboard' | 'pointer' | 'virtual';
48
- let currentModality: Modality | null = null;
49
-
50
- // Track interaction modality (pointer vs keyboard)
51
- if (typeof document !== 'undefined') {
52
- document.addEventListener('keydown', () => {
53
- currentModality = 'keyboard';
54
- }, true);
55
- document.addEventListener('pointerdown', () => {
56
- currentModality = 'pointer';
57
- }, true);
58
- document.addEventListener('pointermove', () => {
59
- currentModality = 'pointer';
60
- }, true);
61
- }
62
-
63
- function isFocusVisible(): boolean {
64
- return currentModality === 'keyboard';
65
- }
66
-
67
- // ============================================
68
- // IMPLEMENTATION
69
- // ============================================
70
-
71
- /**
72
- * Provides the behavior and accessibility implementation for a tooltip trigger.
73
- *
74
- * @example
75
- * ```tsx
76
- * import { createTooltipTrigger } from 'solidaria';
77
- * import { createTooltipTriggerState } from 'solid-stately';
78
- *
79
- * function TooltipButton(props) {
80
- * let ref;
81
- * const state = createTooltipTriggerState({ delay: 500 });
82
- * const { triggerProps, tooltipProps } = createTooltipTrigger(
83
- * { isDisabled: props.isDisabled },
84
- * state,
85
- * () => ref
86
- * );
87
- *
88
- * return (
89
- * <>
90
- * <button ref={ref} {...triggerProps}>
91
- * Hover me
92
- * </button>
93
- * <Show when={state.isOpen()}>
94
- * <div {...tooltipProps}>Tooltip content</div>
95
- * </Show>
96
- * </>
97
- * );
98
- * }
99
- * ```
100
- */
101
- export function createTooltipTrigger(
102
- props: TooltipTriggerProps,
103
- state: TooltipTriggerState,
104
- ref: () => HTMLElement | null | undefined
105
- ): TooltipTriggerAria {
106
- const {
107
- isDisabled = false,
108
- trigger,
109
- shouldCloseOnPress = true,
110
- } = props;
111
-
112
- const tooltipId = createId();
113
-
114
- // Track hover and focus state
115
- let isHovered = false;
116
- let isFocused = false;
117
-
118
- const handleShow = () => {
119
- if (isHovered || isFocused) {
120
- state.open(isFocused);
121
- }
122
- };
123
-
124
- const handleHide = (immediate?: boolean) => {
125
- if (!isHovered && !isFocused) {
126
- state.close(immediate);
127
- }
128
- };
129
-
130
- // Handle Escape key to dismiss tooltip
131
- createEffect(() => {
132
- if (!state.isOpen()) return;
133
-
134
- const onKeyDown = (e: KeyboardEvent) => {
135
- const element = ref();
136
- if (element) {
137
- if (e.key === 'Escape') {
138
- e.stopPropagation();
139
- state.close(true);
140
- }
141
- }
142
- };
143
-
144
- document.addEventListener('keydown', onKeyDown, true);
145
- onCleanup(() => {
146
- document.removeEventListener('keydown', onKeyDown, true);
147
- });
148
- });
149
-
150
- const onHoverStart = () => {
151
- if (trigger === 'focus') {
152
- return;
153
- }
154
- // Hover events (onPointerEnter) only fire from pointer interactions,
155
- // so we can always set isHovered to true here
156
- isHovered = true;
157
- handleShow();
158
- };
159
-
160
- const onHoverEnd = () => {
161
- if (trigger === 'focus') {
162
- return;
163
- }
164
- isFocused = false;
165
- isHovered = false;
166
- handleHide();
167
- };
168
-
169
- const onPressStart = () => {
170
- if (!shouldCloseOnPress) {
171
- return;
172
- }
173
- isFocused = false;
174
- isHovered = false;
175
- handleHide(true);
176
- };
177
-
178
- const onFocus = () => {
179
- const visible = isFocusVisible();
180
- if (visible) {
181
- isFocused = true;
182
- handleShow();
183
- }
184
- };
185
-
186
- const onBlur = () => {
187
- isFocused = false;
188
- isHovered = false;
189
- handleHide(true);
190
- };
191
-
192
- const { hoverProps } = createHover({
193
- isDisabled,
194
- onHoverStart,
195
- onHoverEnd,
196
- });
197
-
198
- const { focusableProps } = createFocusable({
199
- isDisabled,
200
- onFocus,
201
- onBlur,
202
- });
203
-
204
- const triggerProps = {
205
- ...focusableProps,
206
- ...hoverProps,
207
- get 'aria-describedby'() {
208
- return state.isOpen() ? tooltipId : undefined;
209
- },
210
- onPointerDown: onPressStart,
211
- onKeyDown: onPressStart,
212
- // Remove tabIndex set by focusableProps to avoid overriding
213
- tabIndex: undefined,
214
- };
215
-
216
- return {
217
- triggerProps: triggerProps as JSX.HTMLAttributes<HTMLElement>,
218
- tooltipProps: {
219
- id: tooltipId,
220
- },
221
- };
222
- }
1
+ /**
2
+ * createTooltipTrigger hook for Solidaria
3
+ *
4
+ * Provides the behavior and accessibility implementation for a tooltip trigger,
5
+ * e.g. a button that shows a description when focused or hovered.
6
+ *
7
+ * Port of @react-aria/tooltip useTooltipTrigger.
8
+ */
9
+
10
+ import { type JSX, createEffect, onCleanup } from 'solid-js';
11
+ import { type TooltipTriggerState } from '@proyecto-viviana/solid-stately';
12
+ import { createHover } from '../interactions/createHover';
13
+ import { createFocusable } from '../interactions/createFocusable';
14
+ import { createId } from '../ssr';
15
+
16
+ // ============================================
17
+ // TYPES
18
+ // ============================================
19
+
20
+ export interface TooltipTriggerProps {
21
+ /** Whether the tooltip should be disabled. */
22
+ isDisabled?: boolean;
23
+ /**
24
+ * The trigger mechanism for the tooltip.
25
+ * @default 'focus'
26
+ */
27
+ trigger?: 'focus';
28
+ /**
29
+ * Whether the tooltip should close when the trigger is pressed.
30
+ * @default true
31
+ */
32
+ shouldCloseOnPress?: boolean;
33
+ }
34
+
35
+ export interface TooltipTriggerAria {
36
+ /** Props to spread on the trigger element. */
37
+ triggerProps: JSX.HTMLAttributes<HTMLElement>;
38
+ /** Props to spread on the tooltip element (id for accessibility). */
39
+ tooltipProps: { id: string };
40
+ }
41
+
42
+ // ============================================
43
+ // GLOBAL STATE
44
+ // ============================================
45
+
46
+ type Modality = 'keyboard' | 'pointer' | 'virtual';
47
+ let currentModality: Modality | null = null;
48
+
49
+ // Track interaction modality (pointer vs keyboard)
50
+ if (typeof document !== 'undefined') {
51
+ document.addEventListener('keydown', () => {
52
+ currentModality = 'keyboard';
53
+ }, true);
54
+ document.addEventListener('pointerdown', () => {
55
+ currentModality = 'pointer';
56
+ }, true);
57
+ document.addEventListener('pointermove', () => {
58
+ currentModality = 'pointer';
59
+ }, true);
60
+ }
61
+
62
+ function isFocusVisible(): boolean {
63
+ return currentModality === 'keyboard';
64
+ }
65
+
66
+ // ============================================
67
+ // IMPLEMENTATION
68
+ // ============================================
69
+
70
+ /**
71
+ * Provides the behavior and accessibility implementation for a tooltip trigger.
72
+ *
73
+ * @example
74
+ * ```tsx
75
+ * import { createTooltipTrigger } from 'solidaria';
76
+ * import { createTooltipTriggerState } from 'solid-stately';
77
+ *
78
+ * function TooltipButton(props) {
79
+ * let ref;
80
+ * const state = createTooltipTriggerState({ delay: 500 });
81
+ * const { triggerProps, tooltipProps } = createTooltipTrigger(
82
+ * { isDisabled: props.isDisabled },
83
+ * state,
84
+ * () => ref
85
+ * );
86
+ *
87
+ * return (
88
+ * <>
89
+ * <button ref={ref} {...triggerProps}>
90
+ * Hover me
91
+ * </button>
92
+ * <Show when={state.isOpen()}>
93
+ * <div {...tooltipProps}>Tooltip content</div>
94
+ * </Show>
95
+ * </>
96
+ * );
97
+ * }
98
+ * ```
99
+ */
100
+ export function createTooltipTrigger(
101
+ props: TooltipTriggerProps,
102
+ state: TooltipTriggerState,
103
+ ref: () => HTMLElement | null | undefined
104
+ ): TooltipTriggerAria {
105
+ const isDisabled = () => props.isDisabled ?? false;
106
+ const trigger = () => props.trigger;
107
+ const shouldCloseOnPress = () => props.shouldCloseOnPress ?? true;
108
+
109
+ const tooltipId = createId();
110
+
111
+ // Track hover and focus state
112
+ let isHovered = false;
113
+ let isFocused = false;
114
+
115
+ const handleShow = () => {
116
+ if (isHovered || isFocused) {
117
+ state.open(isFocused);
118
+ }
119
+ };
120
+
121
+ const handleHide = (immediate?: boolean) => {
122
+ if (!isHovered && !isFocused) {
123
+ state.close(immediate);
124
+ }
125
+ };
126
+
127
+ // Handle Escape key to dismiss tooltip
128
+ createEffect(() => {
129
+ if (!state.isOpen()) return;
130
+
131
+ const onKeyDown = (e: KeyboardEvent) => {
132
+ const element = ref();
133
+ if (element) {
134
+ if (e.key === 'Escape') {
135
+ e.stopPropagation();
136
+ state.close(true);
137
+ }
138
+ }
139
+ };
140
+
141
+ document.addEventListener('keydown', onKeyDown, true);
142
+ onCleanup(() => {
143
+ document.removeEventListener('keydown', onKeyDown, true);
144
+ });
145
+ });
146
+
147
+ const onHoverStart = () => {
148
+ if (trigger() === 'focus') {
149
+ return;
150
+ }
151
+ // Hover events (onPointerEnter) only fire from pointer interactions,
152
+ // so we can always set isHovered to true here
153
+ isHovered = true;
154
+ handleShow();
155
+ };
156
+
157
+ const onHoverEnd = () => {
158
+ if (trigger() === 'focus') {
159
+ return;
160
+ }
161
+ isFocused = false;
162
+ isHovered = false;
163
+ handleHide();
164
+ };
165
+
166
+ const closeOnPress = () => {
167
+ if (!shouldCloseOnPress()) {
168
+ return;
169
+ }
170
+ isFocused = false;
171
+ isHovered = false;
172
+ handleHide(true);
173
+ };
174
+
175
+ const onKeyDownPress = (event: KeyboardEvent) => {
176
+ if (event.key !== 'Enter' && event.key !== ' ' && event.key !== 'Spacebar') {
177
+ return;
178
+ }
179
+ closeOnPress();
180
+ };
181
+
182
+ const onFocus = () => {
183
+ const visible = isFocusVisible();
184
+ if (visible) {
185
+ isFocused = true;
186
+ handleShow();
187
+ }
188
+ };
189
+
190
+ const onBlur = () => {
191
+ isFocused = false;
192
+ isHovered = false;
193
+ handleHide(true);
194
+ };
195
+
196
+ const { hoverProps } = createHover({
197
+ isDisabled: isDisabled(),
198
+ onHoverStart,
199
+ onHoverEnd,
200
+ });
201
+
202
+ const { focusableProps } = createFocusable({
203
+ isDisabled: isDisabled(),
204
+ onFocus,
205
+ onBlur,
206
+ });
207
+
208
+ const triggerProps = {
209
+ ...focusableProps,
210
+ ...hoverProps,
211
+ get 'aria-describedby'() {
212
+ return state.isOpen() ? tooltipId : undefined;
213
+ },
214
+ onPointerDown: closeOnPress,
215
+ onKeyDown: onKeyDownPress,
216
+ // Remove tabIndex set by focusableProps to avoid overriding
217
+ tabIndex: undefined,
218
+ };
219
+
220
+ return {
221
+ triggerProps: triggerProps as JSX.HTMLAttributes<HTMLElement>,
222
+ tooltipProps: {
223
+ id: tooltipId,
224
+ },
225
+ };
226
+ }
@@ -1,6 +1,6 @@
1
- export { createTooltip, type TooltipProps, type TooltipAria } from './createTooltip';
2
- export {
3
- createTooltipTrigger,
4
- type TooltipTriggerProps,
5
- type TooltipTriggerAria,
6
- } from './createTooltipTrigger';
1
+ export { createTooltip, type TooltipProps, type TooltipAria } from './createTooltip';
2
+ export {
3
+ createTooltipTrigger,
4
+ type TooltipTriggerProps,
5
+ type TooltipTriggerAria,
6
+ } from './createTooltipTrigger';