@avenue-ticketing/ui 0.4.0 → 0.6.0
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.
- package/dist/react/avatar.d.ts +42 -0
- package/dist/react/avatar.js +159 -0
- package/dist/react/avatar.js.map +1 -0
- package/dist/react/badge.d.ts +12 -0
- package/dist/react/badge.js +35 -1
- package/dist/react/badge.js.map +1 -1
- package/dist/react/button.d.ts +1 -1
- package/dist/react/button.js +3 -3
- package/dist/react/button.js.map +1 -1
- package/dist/react/calendar.d.ts +13 -0
- package/dist/react/calendar.js +4639 -0
- package/dist/react/calendar.js.map +1 -0
- package/dist/react/card.d.ts +11 -0
- package/dist/react/card.js +113 -0
- package/dist/react/card.js.map +1 -0
- package/dist/react/checkbox.d.ts +11 -0
- package/dist/react/checkbox.js +129 -0
- package/dist/react/checkbox.js.map +1 -0
- package/dist/react/datetime-picker.d.ts +21 -0
- package/dist/react/datetime-picker.js +6124 -0
- package/dist/react/datetime-picker.js.map +1 -0
- package/dist/react/dialog.js +1 -1
- package/dist/react/dialog.js.map +1 -1
- package/dist/react/dropdown.d.ts +34 -9
- package/dist/react/dropdown.js +295 -144
- package/dist/react/dropdown.js.map +1 -1
- package/dist/react/input.d.ts +7 -0
- package/dist/react/input.js +15 -2
- package/dist/react/input.js.map +1 -1
- package/dist/react/pagination.d.ts +28 -0
- package/dist/react/pagination.js +262 -0
- package/dist/react/pagination.js.map +1 -0
- package/dist/react/popover.d.ts +76 -0
- package/dist/react/popover.js +564 -0
- package/dist/react/popover.js.map +1 -0
- package/dist/react/scroll-header.js +13 -1
- package/dist/react/scroll-header.js.map +1 -1
- package/dist/react/scroll-wheel.d.ts +45 -0
- package/dist/react/scroll-wheel.js +557 -0
- package/dist/react/scroll-wheel.js.map +1 -0
- package/dist/react/select.d.ts +62 -0
- package/dist/react/select.js +889 -0
- package/dist/react/select.js.map +1 -0
- package/dist/react/sheet.js +1 -1
- package/dist/react/sheet.js.map +1 -1
- package/dist/react/switch.d.ts +38 -0
- package/dist/react/switch.js +117 -0
- package/dist/react/switch.js.map +1 -0
- package/dist/react/table-pagination.d.ts +15 -0
- package/dist/react/table-pagination.js +1153 -0
- package/dist/react/table-pagination.js.map +1 -0
- package/dist/react/table-view/column-menu.d.ts +15 -0
- package/dist/react/table-view/column-menu.js +955 -0
- package/dist/react/table-view/column-menu.js.map +1 -0
- package/dist/react/table-view/index.d.ts +70 -0
- package/dist/react/table-view/index.js +2190 -0
- package/dist/react/table-view/index.js.map +1 -0
- package/dist/react/table.d.ts +86 -0
- package/dist/react/table.js +414 -0
- package/dist/react/table.js.map +1 -0
- package/dist/react/tabs.d.ts +9 -3
- package/dist/react/tabs.js +204 -48
- package/dist/react/tabs.js.map +1 -1
- package/dist/react/textarea.d.ts +6 -0
- package/dist/react/textarea.js +33 -0
- package/dist/react/textarea.js.map +1 -0
- package/dist/react/time-picker.d.ts +22 -0
- package/dist/react/time-picker.js +856 -0
- package/dist/react/time-picker.js.map +1 -0
- package/dist/react/tooltip.d.ts +45 -0
- package/dist/react/tooltip.js +540 -0
- package/dist/react/tooltip.js.map +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,540 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import { createPortal } from 'react-dom';
|
|
3
|
+
import { clsx } from 'clsx';
|
|
4
|
+
import { twMerge } from 'tailwind-merge';
|
|
5
|
+
import { jsx, jsxs } from 'react/jsx-runtime';
|
|
6
|
+
|
|
7
|
+
function cn(...inputs) {
|
|
8
|
+
return twMerge(clsx(inputs));
|
|
9
|
+
}
|
|
10
|
+
var ARROW_SPAN_PX = 14;
|
|
11
|
+
var ARROW_HALF_PX = ARROW_SPAN_PX / 2;
|
|
12
|
+
var ARROW_EDGE_PAD_PX = 5;
|
|
13
|
+
function computeArrowCenter(triggerRect, layerRect, side) {
|
|
14
|
+
const midX = triggerRect.left + triggerRect.width / 2;
|
|
15
|
+
const midY = triggerRect.top + triggerRect.height / 2;
|
|
16
|
+
const min = ARROW_EDGE_PAD_PX + ARROW_HALF_PX;
|
|
17
|
+
if (side === "top" || side === "bottom") {
|
|
18
|
+
return Math.max(min, Math.min(midX - layerRect.left, layerRect.width - min));
|
|
19
|
+
}
|
|
20
|
+
return Math.max(min, Math.min(midY - layerRect.top, layerRect.height - min));
|
|
21
|
+
}
|
|
22
|
+
function computePosition(trigger, layer, side, align, offset, viewportPadding) {
|
|
23
|
+
const triggerRect = trigger.getBoundingClientRect();
|
|
24
|
+
const pw = layer.offsetWidth;
|
|
25
|
+
const ph = layer.offsetHeight;
|
|
26
|
+
const vw = window.innerWidth;
|
|
27
|
+
const vh = window.innerHeight;
|
|
28
|
+
let top = 0;
|
|
29
|
+
let left = 0;
|
|
30
|
+
let effectiveSide = side;
|
|
31
|
+
const place = (s) => {
|
|
32
|
+
switch (s) {
|
|
33
|
+
case "bottom":
|
|
34
|
+
top = triggerRect.bottom + offset;
|
|
35
|
+
break;
|
|
36
|
+
case "top":
|
|
37
|
+
top = triggerRect.top - ph - offset;
|
|
38
|
+
break;
|
|
39
|
+
case "right":
|
|
40
|
+
left = triggerRect.right + offset;
|
|
41
|
+
top = align === "start" ? triggerRect.top : align === "end" ? triggerRect.bottom - ph : triggerRect.top + triggerRect.height / 2 - ph / 2;
|
|
42
|
+
return;
|
|
43
|
+
case "left":
|
|
44
|
+
left = triggerRect.left - pw - offset;
|
|
45
|
+
top = align === "start" ? triggerRect.top : align === "end" ? triggerRect.bottom - ph : triggerRect.top + triggerRect.height / 2 - ph / 2;
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
if (align === "start") left = triggerRect.left;
|
|
49
|
+
else if (align === "end") left = triggerRect.right - pw;
|
|
50
|
+
else left = triggerRect.left + triggerRect.width / 2 - pw / 2;
|
|
51
|
+
};
|
|
52
|
+
place(side);
|
|
53
|
+
const tryFlip = () => {
|
|
54
|
+
if (side === "bottom" && top + ph > vh - viewportPadding) {
|
|
55
|
+
const newTop = triggerRect.top - ph - offset;
|
|
56
|
+
if (newTop >= viewportPadding) {
|
|
57
|
+
effectiveSide = "top";
|
|
58
|
+
top = newTop;
|
|
59
|
+
}
|
|
60
|
+
} else if (side === "top" && top < viewportPadding) {
|
|
61
|
+
const newTop = triggerRect.bottom + offset;
|
|
62
|
+
if (newTop + ph <= vh - viewportPadding) {
|
|
63
|
+
effectiveSide = "bottom";
|
|
64
|
+
top = newTop;
|
|
65
|
+
}
|
|
66
|
+
} else if (side === "right" && left + pw > vw - viewportPadding) {
|
|
67
|
+
const newLeft = triggerRect.left - pw - offset;
|
|
68
|
+
if (newLeft >= viewportPadding) {
|
|
69
|
+
effectiveSide = "left";
|
|
70
|
+
left = newLeft;
|
|
71
|
+
}
|
|
72
|
+
} else if (side === "left" && left < viewportPadding) {
|
|
73
|
+
const newLeft = triggerRect.right + offset;
|
|
74
|
+
if (newLeft + pw <= vw - viewportPadding) {
|
|
75
|
+
effectiveSide = "right";
|
|
76
|
+
left = newLeft;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
};
|
|
80
|
+
tryFlip();
|
|
81
|
+
left = Math.max(
|
|
82
|
+
viewportPadding,
|
|
83
|
+
Math.min(left, vw - pw - viewportPadding)
|
|
84
|
+
);
|
|
85
|
+
top = Math.max(
|
|
86
|
+
viewportPadding,
|
|
87
|
+
Math.min(top, vh - ph - viewportPadding)
|
|
88
|
+
);
|
|
89
|
+
return {
|
|
90
|
+
top,
|
|
91
|
+
left,
|
|
92
|
+
side: effectiveSide,
|
|
93
|
+
arrowCenter: 0
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
function layerRectAtPosition(top, left, width, height) {
|
|
97
|
+
return new DOMRect(left, top, width, height);
|
|
98
|
+
}
|
|
99
|
+
var arrowStrokeProps = {
|
|
100
|
+
fill: "none",
|
|
101
|
+
stroke: "currentColor",
|
|
102
|
+
strokeWidth: 1.25,
|
|
103
|
+
strokeLinecap: "round",
|
|
104
|
+
strokeLinejoin: "round",
|
|
105
|
+
vectorEffect: "non-scaling-stroke",
|
|
106
|
+
className: "text-primary/10"
|
|
107
|
+
};
|
|
108
|
+
function TooltipArrowGraphic({ side }) {
|
|
109
|
+
const fillCls = "fill-background";
|
|
110
|
+
if (side === "bottom") {
|
|
111
|
+
const w2 = ARROW_SPAN_PX;
|
|
112
|
+
const h2 = 8;
|
|
113
|
+
const pts2 = `0,${h2} ${w2 / 2},0 ${w2},${h2}`;
|
|
114
|
+
return /* @__PURE__ */ jsxs(
|
|
115
|
+
"svg",
|
|
116
|
+
{
|
|
117
|
+
width: w2,
|
|
118
|
+
height: h2,
|
|
119
|
+
viewBox: `0 0 ${w2} ${h2}`,
|
|
120
|
+
className: "block shrink-0",
|
|
121
|
+
"aria-hidden": true,
|
|
122
|
+
children: [
|
|
123
|
+
/* @__PURE__ */ jsx("polygon", { points: pts2, className: fillCls }),
|
|
124
|
+
/* @__PURE__ */ jsx("polyline", { points: pts2, ...arrowStrokeProps })
|
|
125
|
+
]
|
|
126
|
+
}
|
|
127
|
+
);
|
|
128
|
+
}
|
|
129
|
+
if (side === "top") {
|
|
130
|
+
const w2 = ARROW_SPAN_PX;
|
|
131
|
+
const h2 = 8;
|
|
132
|
+
const pts2 = `0,0 ${w2 / 2},${h2} ${w2},0`;
|
|
133
|
+
return /* @__PURE__ */ jsxs(
|
|
134
|
+
"svg",
|
|
135
|
+
{
|
|
136
|
+
width: w2,
|
|
137
|
+
height: h2,
|
|
138
|
+
viewBox: `0 0 ${w2} ${h2}`,
|
|
139
|
+
className: "block shrink-0",
|
|
140
|
+
"aria-hidden": true,
|
|
141
|
+
children: [
|
|
142
|
+
/* @__PURE__ */ jsx("polygon", { points: pts2, className: fillCls }),
|
|
143
|
+
/* @__PURE__ */ jsx("polyline", { points: pts2, ...arrowStrokeProps })
|
|
144
|
+
]
|
|
145
|
+
}
|
|
146
|
+
);
|
|
147
|
+
}
|
|
148
|
+
if (side === "right") {
|
|
149
|
+
const w2 = 8;
|
|
150
|
+
const h2 = ARROW_SPAN_PX;
|
|
151
|
+
const pts2 = `${w2},0 0,${h2 / 2} ${w2},${h2}`;
|
|
152
|
+
return /* @__PURE__ */ jsxs(
|
|
153
|
+
"svg",
|
|
154
|
+
{
|
|
155
|
+
width: w2,
|
|
156
|
+
height: h2,
|
|
157
|
+
viewBox: `0 0 ${w2} ${h2}`,
|
|
158
|
+
className: "block shrink-0",
|
|
159
|
+
"aria-hidden": true,
|
|
160
|
+
children: [
|
|
161
|
+
/* @__PURE__ */ jsx("polygon", { points: pts2, className: fillCls }),
|
|
162
|
+
/* @__PURE__ */ jsx("polyline", { points: pts2, ...arrowStrokeProps })
|
|
163
|
+
]
|
|
164
|
+
}
|
|
165
|
+
);
|
|
166
|
+
}
|
|
167
|
+
const w = 8;
|
|
168
|
+
const h = ARROW_SPAN_PX;
|
|
169
|
+
const pts = `0,0 ${w},${h / 2} 0,${h}`;
|
|
170
|
+
return /* @__PURE__ */ jsxs(
|
|
171
|
+
"svg",
|
|
172
|
+
{
|
|
173
|
+
width: w,
|
|
174
|
+
height: h,
|
|
175
|
+
viewBox: `0 0 ${w} ${h}`,
|
|
176
|
+
className: "block shrink-0",
|
|
177
|
+
"aria-hidden": true,
|
|
178
|
+
children: [
|
|
179
|
+
/* @__PURE__ */ jsx("polygon", { points: pts, className: fillCls }),
|
|
180
|
+
/* @__PURE__ */ jsx("polyline", { points: pts, ...arrowStrokeProps })
|
|
181
|
+
]
|
|
182
|
+
}
|
|
183
|
+
);
|
|
184
|
+
}
|
|
185
|
+
function tooltipArrowWrapperStyle(side, arrowCenter) {
|
|
186
|
+
const base = {
|
|
187
|
+
position: "absolute",
|
|
188
|
+
zIndex: 1,
|
|
189
|
+
pointerEvents: "none",
|
|
190
|
+
lineHeight: 0
|
|
191
|
+
};
|
|
192
|
+
switch (side) {
|
|
193
|
+
case "bottom":
|
|
194
|
+
return {
|
|
195
|
+
...base,
|
|
196
|
+
left: arrowCenter,
|
|
197
|
+
top: 0,
|
|
198
|
+
transform: "translate(-50%, -100%)"
|
|
199
|
+
};
|
|
200
|
+
case "top":
|
|
201
|
+
return {
|
|
202
|
+
...base,
|
|
203
|
+
left: arrowCenter,
|
|
204
|
+
bottom: 0,
|
|
205
|
+
transform: "translate(-50%, 100%)"
|
|
206
|
+
};
|
|
207
|
+
case "right":
|
|
208
|
+
return {
|
|
209
|
+
...base,
|
|
210
|
+
top: arrowCenter,
|
|
211
|
+
left: 0,
|
|
212
|
+
transform: "translate(-100%, -50%)"
|
|
213
|
+
};
|
|
214
|
+
case "left":
|
|
215
|
+
return {
|
|
216
|
+
...base,
|
|
217
|
+
top: arrowCenter,
|
|
218
|
+
right: 0,
|
|
219
|
+
transform: "translate(100%, -50%)"
|
|
220
|
+
};
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
var TooltipContext = React.createContext(null);
|
|
224
|
+
function useTooltip() {
|
|
225
|
+
const ctx = React.useContext(TooltipContext);
|
|
226
|
+
if (!ctx) {
|
|
227
|
+
throw new Error("Tooltip components must be used within <Tooltip>");
|
|
228
|
+
}
|
|
229
|
+
return ctx;
|
|
230
|
+
}
|
|
231
|
+
var Tooltip = ({
|
|
232
|
+
children,
|
|
233
|
+
delayDuration = 0,
|
|
234
|
+
closeDelay = 40,
|
|
235
|
+
disabled = false,
|
|
236
|
+
open: controlledOpen,
|
|
237
|
+
onOpenChange
|
|
238
|
+
}) => {
|
|
239
|
+
const [uncontrolledOpen, setUncontrolledOpen] = React.useState(false);
|
|
240
|
+
const isControlled = controlledOpen !== void 0;
|
|
241
|
+
const open = disabled ? false : isControlled ? controlledOpen : uncontrolledOpen;
|
|
242
|
+
const triggerRef = React.useRef(null);
|
|
243
|
+
const contentId = React.useId();
|
|
244
|
+
const openTimerRef = React.useRef(null);
|
|
245
|
+
const closeTimerRef = React.useRef(null);
|
|
246
|
+
const setOpen = React.useCallback(
|
|
247
|
+
(next) => {
|
|
248
|
+
if (disabled) return;
|
|
249
|
+
if (!isControlled) setUncontrolledOpen(next);
|
|
250
|
+
onOpenChange?.(next);
|
|
251
|
+
},
|
|
252
|
+
[disabled, isControlled, onOpenChange]
|
|
253
|
+
);
|
|
254
|
+
const clearTimers = React.useCallback(() => {
|
|
255
|
+
if (openTimerRef.current) clearTimeout(openTimerRef.current);
|
|
256
|
+
if (closeTimerRef.current) clearTimeout(closeTimerRef.current);
|
|
257
|
+
openTimerRef.current = null;
|
|
258
|
+
closeTimerRef.current = null;
|
|
259
|
+
}, []);
|
|
260
|
+
const scheduleOpen = React.useCallback(
|
|
261
|
+
(instant) => {
|
|
262
|
+
clearTimers();
|
|
263
|
+
if (instant || delayDuration <= 0) {
|
|
264
|
+
setOpen(true);
|
|
265
|
+
return;
|
|
266
|
+
}
|
|
267
|
+
openTimerRef.current = setTimeout(() => {
|
|
268
|
+
openTimerRef.current = null;
|
|
269
|
+
setOpen(true);
|
|
270
|
+
}, delayDuration);
|
|
271
|
+
},
|
|
272
|
+
[clearTimers, delayDuration, setOpen]
|
|
273
|
+
);
|
|
274
|
+
const scheduleClose = React.useCallback(() => {
|
|
275
|
+
if (openTimerRef.current) {
|
|
276
|
+
clearTimeout(openTimerRef.current);
|
|
277
|
+
openTimerRef.current = null;
|
|
278
|
+
}
|
|
279
|
+
closeTimerRef.current = setTimeout(() => {
|
|
280
|
+
closeTimerRef.current = null;
|
|
281
|
+
setOpen(false);
|
|
282
|
+
}, closeDelay);
|
|
283
|
+
}, [closeDelay, setOpen]);
|
|
284
|
+
const cancelClose = React.useCallback(() => {
|
|
285
|
+
if (closeTimerRef.current) {
|
|
286
|
+
clearTimeout(closeTimerRef.current);
|
|
287
|
+
closeTimerRef.current = null;
|
|
288
|
+
}
|
|
289
|
+
}, []);
|
|
290
|
+
React.useEffect(() => () => clearTimers(), [clearTimers]);
|
|
291
|
+
const value = React.useMemo(
|
|
292
|
+
() => ({
|
|
293
|
+
open,
|
|
294
|
+
setOpen,
|
|
295
|
+
triggerRef,
|
|
296
|
+
contentId,
|
|
297
|
+
delayDuration,
|
|
298
|
+
closeDelay,
|
|
299
|
+
disabled: Boolean(disabled),
|
|
300
|
+
openTimerRef,
|
|
301
|
+
closeTimerRef,
|
|
302
|
+
clearTimers,
|
|
303
|
+
scheduleOpen,
|
|
304
|
+
scheduleClose,
|
|
305
|
+
cancelClose
|
|
306
|
+
}),
|
|
307
|
+
[
|
|
308
|
+
open,
|
|
309
|
+
setOpen,
|
|
310
|
+
contentId,
|
|
311
|
+
delayDuration,
|
|
312
|
+
closeDelay,
|
|
313
|
+
disabled,
|
|
314
|
+
clearTimers,
|
|
315
|
+
scheduleOpen,
|
|
316
|
+
scheduleClose,
|
|
317
|
+
cancelClose
|
|
318
|
+
]
|
|
319
|
+
);
|
|
320
|
+
return /* @__PURE__ */ jsx(TooltipContext.Provider, { value, children });
|
|
321
|
+
};
|
|
322
|
+
function mergeRefs(...refs) {
|
|
323
|
+
return (node) => {
|
|
324
|
+
for (const ref of refs) {
|
|
325
|
+
if (!ref) continue;
|
|
326
|
+
if (typeof ref === "function") ref(node);
|
|
327
|
+
else ref.current = node;
|
|
328
|
+
}
|
|
329
|
+
};
|
|
330
|
+
}
|
|
331
|
+
var TooltipTrigger = React.forwardRef((props, ref) => {
|
|
332
|
+
const {
|
|
333
|
+
children,
|
|
334
|
+
asChild,
|
|
335
|
+
className,
|
|
336
|
+
onPointerEnter,
|
|
337
|
+
onPointerLeave,
|
|
338
|
+
onFocus,
|
|
339
|
+
onBlur,
|
|
340
|
+
...restProps
|
|
341
|
+
} = props;
|
|
342
|
+
const {
|
|
343
|
+
open,
|
|
344
|
+
triggerRef,
|
|
345
|
+
contentId,
|
|
346
|
+
disabled,
|
|
347
|
+
scheduleOpen,
|
|
348
|
+
scheduleClose,
|
|
349
|
+
cancelClose,
|
|
350
|
+
clearTimers
|
|
351
|
+
} = useTooltip();
|
|
352
|
+
const describedBy = open ? contentId : void 0;
|
|
353
|
+
const handlePointerEnter = (e) => {
|
|
354
|
+
onPointerEnter?.(e);
|
|
355
|
+
if (disabled || e.defaultPrevented) return;
|
|
356
|
+
cancelClose();
|
|
357
|
+
scheduleOpen(false);
|
|
358
|
+
};
|
|
359
|
+
const handlePointerLeave = (e) => {
|
|
360
|
+
onPointerLeave?.(e);
|
|
361
|
+
if (disabled || e.defaultPrevented) return;
|
|
362
|
+
scheduleClose();
|
|
363
|
+
};
|
|
364
|
+
const handleFocus = (e) => {
|
|
365
|
+
onFocus?.(e);
|
|
366
|
+
if (disabled || e.defaultPrevented) return;
|
|
367
|
+
clearTimers();
|
|
368
|
+
scheduleOpen(true);
|
|
369
|
+
};
|
|
370
|
+
const handleBlur = (e) => {
|
|
371
|
+
onBlur?.(e);
|
|
372
|
+
if (disabled || e.defaultPrevented) return;
|
|
373
|
+
scheduleClose();
|
|
374
|
+
};
|
|
375
|
+
if (asChild && React.isValidElement(children)) {
|
|
376
|
+
const child = children;
|
|
377
|
+
return React.cloneElement(child, {
|
|
378
|
+
...child.props,
|
|
379
|
+
...restProps,
|
|
380
|
+
ref: mergeRefs(ref, triggerRef, child.props.ref),
|
|
381
|
+
className: cn(child.props.className, className),
|
|
382
|
+
"aria-describedby": [
|
|
383
|
+
describedBy,
|
|
384
|
+
child.props["aria-describedby"],
|
|
385
|
+
restProps["aria-describedby"]
|
|
386
|
+
].filter(Boolean).join(" ") || void 0,
|
|
387
|
+
onPointerEnter: (e) => {
|
|
388
|
+
child.props.onPointerEnter?.(e);
|
|
389
|
+
handlePointerEnter(e);
|
|
390
|
+
},
|
|
391
|
+
onPointerLeave: (e) => {
|
|
392
|
+
child.props.onPointerLeave?.(e);
|
|
393
|
+
handlePointerLeave(e);
|
|
394
|
+
},
|
|
395
|
+
onFocus: (e) => {
|
|
396
|
+
child.props.onFocus?.(e);
|
|
397
|
+
handleFocus(e);
|
|
398
|
+
},
|
|
399
|
+
onBlur: (e) => {
|
|
400
|
+
child.props.onBlur?.(e);
|
|
401
|
+
handleBlur(e);
|
|
402
|
+
}
|
|
403
|
+
});
|
|
404
|
+
}
|
|
405
|
+
return /* @__PURE__ */ jsx(
|
|
406
|
+
"span",
|
|
407
|
+
{
|
|
408
|
+
ref: mergeRefs(ref, triggerRef),
|
|
409
|
+
className: cn("inline-flex", className),
|
|
410
|
+
"aria-describedby": describedBy,
|
|
411
|
+
onPointerEnter: handlePointerEnter,
|
|
412
|
+
onPointerLeave: handlePointerLeave,
|
|
413
|
+
onFocus: handleFocus,
|
|
414
|
+
onBlur: handleBlur,
|
|
415
|
+
...restProps,
|
|
416
|
+
children
|
|
417
|
+
}
|
|
418
|
+
);
|
|
419
|
+
});
|
|
420
|
+
TooltipTrigger.displayName = "TooltipTrigger";
|
|
421
|
+
var TooltipContent = React.forwardRef(
|
|
422
|
+
({
|
|
423
|
+
children,
|
|
424
|
+
side = "top",
|
|
425
|
+
align = "center",
|
|
426
|
+
offset = 12,
|
|
427
|
+
viewportPadding = 8,
|
|
428
|
+
container,
|
|
429
|
+
showArrow = true,
|
|
430
|
+
className,
|
|
431
|
+
style,
|
|
432
|
+
...props
|
|
433
|
+
}, ref) => {
|
|
434
|
+
const {
|
|
435
|
+
open,
|
|
436
|
+
setOpen,
|
|
437
|
+
triggerRef,
|
|
438
|
+
contentId,
|
|
439
|
+
disabled,
|
|
440
|
+
scheduleClose,
|
|
441
|
+
cancelClose
|
|
442
|
+
} = useTooltip();
|
|
443
|
+
const [position, setPosition] = React.useState({
|
|
444
|
+
top: -9999,
|
|
445
|
+
left: -9999,
|
|
446
|
+
side,
|
|
447
|
+
arrowCenter: 0
|
|
448
|
+
});
|
|
449
|
+
const layerRef = React.useRef(null);
|
|
450
|
+
const mergedRef = mergeRefs(ref, layerRef);
|
|
451
|
+
React.useLayoutEffect(() => {
|
|
452
|
+
if (!open || disabled || !triggerRef.current || !layerRef.current) return;
|
|
453
|
+
const update = () => {
|
|
454
|
+
if (!triggerRef.current || !layerRef.current) return;
|
|
455
|
+
const trigger = triggerRef.current;
|
|
456
|
+
const layer = layerRef.current;
|
|
457
|
+
const next = computePosition(
|
|
458
|
+
trigger,
|
|
459
|
+
layer,
|
|
460
|
+
side,
|
|
461
|
+
align,
|
|
462
|
+
offset,
|
|
463
|
+
viewportPadding
|
|
464
|
+
);
|
|
465
|
+
const tr = trigger.getBoundingClientRect();
|
|
466
|
+
const lrSynth = layerRectAtPosition(
|
|
467
|
+
next.top,
|
|
468
|
+
next.left,
|
|
469
|
+
layer.offsetWidth,
|
|
470
|
+
layer.offsetHeight
|
|
471
|
+
);
|
|
472
|
+
setPosition({
|
|
473
|
+
...next,
|
|
474
|
+
arrowCenter: computeArrowCenter(tr, lrSynth, next.side)
|
|
475
|
+
});
|
|
476
|
+
};
|
|
477
|
+
update();
|
|
478
|
+
window.addEventListener("resize", update);
|
|
479
|
+
window.addEventListener("scroll", update, true);
|
|
480
|
+
return () => {
|
|
481
|
+
window.removeEventListener("resize", update);
|
|
482
|
+
window.removeEventListener("scroll", update, true);
|
|
483
|
+
};
|
|
484
|
+
}, [open, disabled, side, align, offset, viewportPadding, triggerRef]);
|
|
485
|
+
React.useEffect(() => {
|
|
486
|
+
if (!open || disabled) return;
|
|
487
|
+
const onKey = (e) => {
|
|
488
|
+
if (e.key === "Escape") setOpen(false);
|
|
489
|
+
};
|
|
490
|
+
window.addEventListener("keydown", onKey);
|
|
491
|
+
return () => window.removeEventListener("keydown", onKey);
|
|
492
|
+
}, [open, disabled, setOpen]);
|
|
493
|
+
const resolvedContainer = container !== void 0 ? container : typeof document !== "undefined" ? document.body : null;
|
|
494
|
+
if (!open || !resolvedContainer) return null;
|
|
495
|
+
return createPortal(
|
|
496
|
+
/* @__PURE__ */ jsxs(
|
|
497
|
+
"div",
|
|
498
|
+
{
|
|
499
|
+
...props,
|
|
500
|
+
ref: mergedRef,
|
|
501
|
+
id: contentId,
|
|
502
|
+
role: "tooltip",
|
|
503
|
+
"data-slot": "tooltip-content",
|
|
504
|
+
"data-side": position.side,
|
|
505
|
+
className: cn(
|
|
506
|
+
"bg-background text-primary border-primary/10 pointer-events-auto fixed z-50 max-w-xs overflow-visible rounded-md border px-3 py-1.5 text-xs leading-snug shadow-md",
|
|
507
|
+
className
|
|
508
|
+
),
|
|
509
|
+
style: {
|
|
510
|
+
top: position.top,
|
|
511
|
+
left: position.left,
|
|
512
|
+
...style
|
|
513
|
+
},
|
|
514
|
+
onPointerEnter: () => cancelClose(),
|
|
515
|
+
onPointerLeave: () => scheduleClose(),
|
|
516
|
+
children: [
|
|
517
|
+
showArrow ? /* @__PURE__ */ jsx(
|
|
518
|
+
"span",
|
|
519
|
+
{
|
|
520
|
+
"aria-hidden": true,
|
|
521
|
+
style: tooltipArrowWrapperStyle(
|
|
522
|
+
position.side,
|
|
523
|
+
position.arrowCenter
|
|
524
|
+
),
|
|
525
|
+
children: /* @__PURE__ */ jsx(TooltipArrowGraphic, { side: position.side })
|
|
526
|
+
}
|
|
527
|
+
) : null,
|
|
528
|
+
/* @__PURE__ */ jsx("div", { className: "relative z-[2] max-w-full", children })
|
|
529
|
+
]
|
|
530
|
+
}
|
|
531
|
+
),
|
|
532
|
+
resolvedContainer
|
|
533
|
+
);
|
|
534
|
+
}
|
|
535
|
+
);
|
|
536
|
+
TooltipContent.displayName = "TooltipContent";
|
|
537
|
+
|
|
538
|
+
export { Tooltip, TooltipContent, TooltipTrigger };
|
|
539
|
+
//# sourceMappingURL=tooltip.js.map
|
|
540
|
+
//# sourceMappingURL=tooltip.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/lib/utils.ts","../../src/react/tooltip.tsx"],"names":["w","h","pts"],"mappings":"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","file":"tooltip.js","sourcesContent":["import { clsx, type ClassValue } from \"clsx\";\nimport { twMerge } from \"tailwind-merge\";\n\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs));\n}\n","\"use client\";\n\nimport * as React from \"react\";\nimport { createPortal } from \"react-dom\";\n\nimport { cn } from \"@/lib/utils\";\n\ntype Side = \"top\" | \"right\" | \"bottom\" | \"left\";\ntype Align = \"start\" | \"center\" | \"end\";\n\ninterface Position {\n top: number;\n left: number;\n side: Side;\n arrowCenter: number;\n}\n\n/** span of the arrow along the tooltip edge (matches svg width/height) */\nconst ARROW_SPAN_PX = 14;\nconst ARROW_HALF_PX = ARROW_SPAN_PX / 2;\nconst ARROW_EDGE_PAD_PX = 5;\n\nfunction computeArrowCenter(\n triggerRect: DOMRectReadOnly,\n layerRect: DOMRectReadOnly,\n side: Side,\n): number {\n const midX = triggerRect.left + triggerRect.width / 2;\n const midY = triggerRect.top + triggerRect.height / 2;\n const min = ARROW_EDGE_PAD_PX + ARROW_HALF_PX;\n if (side === \"top\" || side === \"bottom\") {\n return Math.max(min, Math.min(midX - layerRect.left, layerRect.width - min));\n }\n return Math.max(min, Math.min(midY - layerRect.top, layerRect.height - min));\n}\n\nfunction computePosition(\n trigger: HTMLElement,\n layer: HTMLElement,\n side: Side,\n align: Align,\n offset: number,\n viewportPadding: number,\n): Position {\n const triggerRect = trigger.getBoundingClientRect();\n const pw = layer.offsetWidth;\n const ph = layer.offsetHeight;\n const vw = window.innerWidth;\n const vh = window.innerHeight;\n\n let top = 0;\n let left = 0;\n let effectiveSide: Side = side;\n\n const place = (s: Side) => {\n switch (s) {\n case \"bottom\":\n top = triggerRect.bottom + offset;\n break;\n case \"top\":\n top = triggerRect.top - ph - offset;\n break;\n case \"right\":\n left = triggerRect.right + offset;\n top =\n align === \"start\"\n ? triggerRect.top\n : align === \"end\"\n ? triggerRect.bottom - ph\n : triggerRect.top + triggerRect.height / 2 - ph / 2;\n return;\n case \"left\":\n left = triggerRect.left - pw - offset;\n top =\n align === \"start\"\n ? triggerRect.top\n : align === \"end\"\n ? triggerRect.bottom - ph\n : triggerRect.top + triggerRect.height / 2 - ph / 2;\n return;\n }\n if (align === \"start\") left = triggerRect.left;\n else if (align === \"end\") left = triggerRect.right - pw;\n else left = triggerRect.left + triggerRect.width / 2 - pw / 2;\n };\n\n place(side);\n\n const tryFlip = () => {\n if (side === \"bottom\" && top + ph > vh - viewportPadding) {\n const newTop = triggerRect.top - ph - offset;\n if (newTop >= viewportPadding) {\n effectiveSide = \"top\";\n top = newTop;\n }\n } else if (side === \"top\" && top < viewportPadding) {\n const newTop = triggerRect.bottom + offset;\n if (newTop + ph <= vh - viewportPadding) {\n effectiveSide = \"bottom\";\n top = newTop;\n }\n } else if (side === \"right\" && left + pw > vw - viewportPadding) {\n const newLeft = triggerRect.left - pw - offset;\n if (newLeft >= viewportPadding) {\n effectiveSide = \"left\";\n left = newLeft;\n }\n } else if (side === \"left\" && left < viewportPadding) {\n const newLeft = triggerRect.right + offset;\n if (newLeft + pw <= vw - viewportPadding) {\n effectiveSide = \"right\";\n left = newLeft;\n }\n }\n };\n tryFlip();\n\n left = Math.max(\n viewportPadding,\n Math.min(left, vw - pw - viewportPadding),\n );\n top = Math.max(\n viewportPadding,\n Math.min(top, vh - ph - viewportPadding),\n );\n\n return {\n top,\n left,\n side: effectiveSide,\n arrowCenter: 0,\n };\n}\n\nfunction layerRectAtPosition(\n top: number,\n left: number,\n width: number,\n height: number,\n): DOMRectReadOnly {\n return new DOMRect(left, top, width, height);\n}\n\nconst arrowStrokeProps = {\n fill: \"none\" as const,\n stroke: \"currentColor\",\n strokeWidth: 1.25,\n strokeLinecap: \"round\" as const,\n strokeLinejoin: \"round\" as const,\n vectorEffect: \"non-scaling-stroke\" as const,\n className: \"text-primary/10\",\n};\n\n/** filled caret + open polyline so stroke follows outer legs only (no diamond inner corners) */\nfunction TooltipArrowGraphic({ side }: { side: Side }) {\n const fillCls = \"fill-background\";\n\n if (side === \"bottom\") {\n const w = ARROW_SPAN_PX;\n const h = 8;\n const pts = `0,${h} ${w / 2},0 ${w},${h}`;\n return (\n <svg\n width={w}\n height={h}\n viewBox={`0 0 ${w} ${h}`}\n className=\"block shrink-0\"\n aria-hidden\n >\n <polygon points={pts} className={fillCls} />\n <polyline points={pts} {...arrowStrokeProps} />\n </svg>\n );\n }\n if (side === \"top\") {\n const w = ARROW_SPAN_PX;\n const h = 8;\n const pts = `0,0 ${w / 2},${h} ${w},0`;\n return (\n <svg\n width={w}\n height={h}\n viewBox={`0 0 ${w} ${h}`}\n className=\"block shrink-0\"\n aria-hidden\n >\n <polygon points={pts} className={fillCls} />\n <polyline points={pts} {...arrowStrokeProps} />\n </svg>\n );\n }\n if (side === \"right\") {\n const w = 8;\n const h = ARROW_SPAN_PX;\n const pts = `${w},0 0,${h / 2} ${w},${h}`;\n return (\n <svg\n width={w}\n height={h}\n viewBox={`0 0 ${w} ${h}`}\n className=\"block shrink-0\"\n aria-hidden\n >\n <polygon points={pts} className={fillCls} />\n <polyline points={pts} {...arrowStrokeProps} />\n </svg>\n );\n }\n const w = 8;\n const h = ARROW_SPAN_PX;\n const pts = `0,0 ${w},${h / 2} 0,${h}`;\n return (\n <svg\n width={w}\n height={h}\n viewBox={`0 0 ${w} ${h}`}\n className=\"block shrink-0\"\n aria-hidden\n >\n <polygon points={pts} className={fillCls} />\n <polyline points={pts} {...arrowStrokeProps} />\n </svg>\n );\n}\n\nfunction tooltipArrowWrapperStyle(\n side: Side,\n arrowCenter: number,\n): React.CSSProperties {\n const base: React.CSSProperties = {\n position: \"absolute\",\n zIndex: 1,\n pointerEvents: \"none\",\n lineHeight: 0,\n };\n switch (side) {\n case \"bottom\":\n return {\n ...base,\n left: arrowCenter,\n top: 0,\n transform: \"translate(-50%, -100%)\",\n };\n case \"top\":\n return {\n ...base,\n left: arrowCenter,\n bottom: 0,\n transform: \"translate(-50%, 100%)\",\n };\n case \"right\":\n return {\n ...base,\n top: arrowCenter,\n left: 0,\n transform: \"translate(-100%, -50%)\",\n };\n case \"left\":\n return {\n ...base,\n top: arrowCenter,\n right: 0,\n transform: \"translate(100%, -50%)\",\n };\n }\n}\n\ntype TooltipContextValue = {\n open: boolean;\n setOpen: (next: boolean) => void;\n triggerRef: React.RefObject<HTMLElement | null>;\n contentId: string;\n delayDuration: number;\n closeDelay: number;\n disabled: boolean;\n openTimerRef: React.MutableRefObject<ReturnType<typeof setTimeout> | null>;\n closeTimerRef: React.MutableRefObject<ReturnType<typeof setTimeout> | null>;\n clearTimers: () => void;\n scheduleOpen: (instant?: boolean) => void;\n scheduleClose: () => void;\n cancelClose: () => void;\n};\n\nconst TooltipContext = React.createContext<TooltipContextValue | null>(null);\n\nfunction useTooltip() {\n const ctx = React.useContext(TooltipContext);\n if (!ctx) {\n throw new Error(\"Tooltip components must be used within <Tooltip>\");\n }\n return ctx;\n}\n\nexport type TooltipProps = {\n children: React.ReactNode;\n /** delay before showing on hover (ms); `0` is instant; focus always opens immediately */\n delayDuration?: number;\n /** delay after pointer leaves trigger/content before hiding (ms) */\n closeDelay?: number;\n disabled?: boolean;\n open?: boolean;\n onOpenChange?: (open: boolean) => void;\n};\n\nexport const Tooltip: React.FC<TooltipProps> = ({\n children,\n delayDuration = 0,\n closeDelay = 40,\n disabled = false,\n open: controlledOpen,\n onOpenChange,\n}) => {\n const [uncontrolledOpen, setUncontrolledOpen] = React.useState(false);\n const isControlled = controlledOpen !== undefined;\n const open = disabled ? false : isControlled ? controlledOpen : uncontrolledOpen;\n const triggerRef = React.useRef<HTMLElement | null>(null);\n const contentId = React.useId();\n const openTimerRef = React.useRef<ReturnType<typeof setTimeout> | null>(null);\n const closeTimerRef = React.useRef<ReturnType<typeof setTimeout> | null>(null);\n\n const setOpen = React.useCallback(\n (next: boolean) => {\n if (disabled) return;\n if (!isControlled) setUncontrolledOpen(next);\n onOpenChange?.(next);\n },\n [disabled, isControlled, onOpenChange],\n );\n\n const clearTimers = React.useCallback(() => {\n if (openTimerRef.current) clearTimeout(openTimerRef.current);\n if (closeTimerRef.current) clearTimeout(closeTimerRef.current);\n openTimerRef.current = null;\n closeTimerRef.current = null;\n }, []);\n\n const scheduleOpen = React.useCallback(\n (instant?: boolean) => {\n clearTimers();\n if (instant || delayDuration <= 0) {\n setOpen(true);\n return;\n }\n openTimerRef.current = setTimeout(() => {\n openTimerRef.current = null;\n setOpen(true);\n }, delayDuration);\n },\n [clearTimers, delayDuration, setOpen],\n );\n\n const scheduleClose = React.useCallback(() => {\n if (openTimerRef.current) {\n clearTimeout(openTimerRef.current);\n openTimerRef.current = null;\n }\n closeTimerRef.current = setTimeout(() => {\n closeTimerRef.current = null;\n setOpen(false);\n }, closeDelay);\n }, [closeDelay, setOpen]);\n\n const cancelClose = React.useCallback(() => {\n if (closeTimerRef.current) {\n clearTimeout(closeTimerRef.current);\n closeTimerRef.current = null;\n }\n }, []);\n\n React.useEffect(() => () => clearTimers(), [clearTimers]);\n\n const value = React.useMemo(\n () => ({\n open,\n setOpen,\n triggerRef,\n contentId,\n delayDuration,\n closeDelay,\n disabled: Boolean(disabled),\n openTimerRef,\n closeTimerRef,\n clearTimers,\n scheduleOpen,\n scheduleClose,\n cancelClose,\n }),\n [\n open,\n setOpen,\n contentId,\n delayDuration,\n closeDelay,\n disabled,\n clearTimers,\n scheduleOpen,\n scheduleClose,\n cancelClose,\n ],\n );\n\n return (\n <TooltipContext.Provider value={value}>{children}</TooltipContext.Provider>\n );\n};\n\nexport type TooltipTriggerProps = React.HTMLAttributes<HTMLElement> & {\n children: React.ReactNode;\n asChild?: boolean;\n};\n\nfunction mergeRefs<T>(\n ...refs: Array<React.Ref<T> | undefined>\n): React.RefCallback<T> {\n return (node) => {\n for (const ref of refs) {\n if (!ref) continue;\n if (typeof ref === \"function\") ref(node);\n else (ref as React.MutableRefObject<T | null>).current = node;\n }\n };\n}\n\nexport const TooltipTrigger = React.forwardRef<\n HTMLElement,\n TooltipTriggerProps\n>((props, ref) => {\n /* eslint-disable react/prop-types -- props typed as TooltipTriggerProps */\n const {\n children,\n asChild,\n className,\n onPointerEnter,\n onPointerLeave,\n onFocus,\n onBlur,\n ...restProps\n } = props;\n /* eslint-enable react/prop-types */\n const {\n open,\n triggerRef,\n contentId,\n disabled,\n scheduleOpen,\n scheduleClose,\n cancelClose,\n clearTimers,\n } = useTooltip();\n\n const describedBy = open ? contentId : undefined;\n\n const handlePointerEnter = (e: React.PointerEvent<HTMLElement>) => {\n onPointerEnter?.(e);\n if (disabled || e.defaultPrevented) return;\n cancelClose();\n scheduleOpen(false);\n };\n\n const handlePointerLeave = (e: React.PointerEvent<HTMLElement>) => {\n onPointerLeave?.(e);\n if (disabled || e.defaultPrevented) return;\n scheduleClose();\n };\n\n const handleFocus = (e: React.FocusEvent<HTMLElement>) => {\n onFocus?.(e);\n if (disabled || e.defaultPrevented) return;\n clearTimers();\n scheduleOpen(true);\n };\n\n const handleBlur = (e: React.FocusEvent<HTMLElement>) => {\n onBlur?.(e);\n if (disabled || e.defaultPrevented) return;\n scheduleClose();\n };\n\n if (asChild && React.isValidElement(children)) {\n const child = children as React.ReactElement<{\n ref?: React.Ref<unknown>;\n onPointerEnter?: React.PointerEventHandler<HTMLElement>;\n onPointerLeave?: React.PointerEventHandler<HTMLElement>;\n onFocus?: React.FocusEventHandler<HTMLElement>;\n onBlur?: React.FocusEventHandler<HTMLElement>;\n className?: string;\n \"aria-describedby\"?: string;\n }>;\n return React.cloneElement(child, {\n ...child.props,\n ...restProps,\n ref: mergeRefs(ref, triggerRef, child.props.ref) as React.Ref<unknown>,\n className: cn(child.props.className, className),\n \"aria-describedby\":\n [\n describedBy,\n child.props[\"aria-describedby\"],\n restProps[\"aria-describedby\"],\n ]\n .filter(Boolean)\n .join(\" \") || undefined,\n onPointerEnter: (e: React.PointerEvent<HTMLElement>) => {\n child.props.onPointerEnter?.(e);\n handlePointerEnter(e);\n },\n onPointerLeave: (e: React.PointerEvent<HTMLElement>) => {\n child.props.onPointerLeave?.(e);\n handlePointerLeave(e);\n },\n onFocus: (e: React.FocusEvent<HTMLElement>) => {\n child.props.onFocus?.(e);\n handleFocus(e);\n },\n onBlur: (e: React.FocusEvent<HTMLElement>) => {\n child.props.onBlur?.(e);\n handleBlur(e);\n },\n } as Partial<typeof child.props>);\n }\n\n return (\n <span\n ref={mergeRefs(ref, triggerRef) as React.Ref<HTMLSpanElement>}\n className={cn(\"inline-flex\", className)}\n aria-describedby={describedBy}\n onPointerEnter={handlePointerEnter}\n onPointerLeave={handlePointerLeave}\n onFocus={handleFocus}\n onBlur={handleBlur}\n {...restProps}\n >\n {children}\n </span>\n );\n});\nTooltipTrigger.displayName = \"TooltipTrigger\";\n\nexport type TooltipContentProps = Omit<\n React.HTMLAttributes<HTMLDivElement>,\n \"children\"\n> & {\n children: React.ReactNode;\n side?: Side;\n align?: Align;\n offset?: number;\n viewportPadding?: number;\n container?: HTMLElement | null;\n /** small diamond at the trigger edge; set `false` to hide */\n showArrow?: boolean;\n};\n\nexport const TooltipContent = React.forwardRef<\n HTMLDivElement,\n TooltipContentProps\n>(\n (\n {\n children,\n side = \"top\",\n align = \"center\",\n offset = 12,\n viewportPadding = 8,\n container,\n showArrow = true,\n className,\n style,\n ...props\n },\n ref,\n ) => {\n const {\n open,\n setOpen,\n triggerRef,\n contentId,\n disabled,\n scheduleClose,\n cancelClose,\n } = useTooltip();\n\n const [position, setPosition] = React.useState<Position>({\n top: -9999,\n left: -9999,\n side,\n arrowCenter: 0,\n });\n const layerRef = React.useRef<HTMLDivElement>(null);\n const mergedRef = mergeRefs(ref, layerRef);\n\n React.useLayoutEffect(() => {\n if (!open || disabled || !triggerRef.current || !layerRef.current) return;\n\n const update = () => {\n if (!triggerRef.current || !layerRef.current) return;\n const trigger = triggerRef.current;\n const layer = layerRef.current;\n const next = computePosition(\n trigger,\n layer,\n side,\n align,\n offset,\n viewportPadding,\n );\n const tr = trigger.getBoundingClientRect();\n const lrSynth = layerRectAtPosition(\n next.top,\n next.left,\n layer.offsetWidth,\n layer.offsetHeight,\n );\n setPosition({\n ...next,\n arrowCenter: computeArrowCenter(tr, lrSynth, next.side),\n });\n };\n\n update();\n window.addEventListener(\"resize\", update);\n window.addEventListener(\"scroll\", update, true);\n return () => {\n window.removeEventListener(\"resize\", update);\n window.removeEventListener(\"scroll\", update, true);\n };\n }, [open, disabled, side, align, offset, viewportPadding, triggerRef]);\n\n React.useEffect(() => {\n if (!open || disabled) return;\n const onKey = (e: KeyboardEvent) => {\n if (e.key === \"Escape\") setOpen(false);\n };\n window.addEventListener(\"keydown\", onKey);\n return () => window.removeEventListener(\"keydown\", onKey);\n }, [open, disabled, setOpen]);\n\n const resolvedContainer =\n container !== undefined\n ? container\n : typeof document !== \"undefined\"\n ? document.body\n : null;\n\n if (!open || !resolvedContainer) return null;\n\n return createPortal(\n <div\n {...props}\n ref={mergedRef}\n id={contentId}\n role=\"tooltip\"\n data-slot=\"tooltip-content\"\n data-side={position.side}\n className={cn(\n \"bg-background text-primary border-primary/10 pointer-events-auto fixed z-50 max-w-xs overflow-visible rounded-md border px-3 py-1.5 text-xs leading-snug shadow-md\",\n className,\n )}\n style={{\n top: position.top,\n left: position.left,\n ...style,\n }}\n onPointerEnter={() => cancelClose()}\n onPointerLeave={() => scheduleClose()}\n >\n {showArrow ? (\n <span\n aria-hidden\n style={tooltipArrowWrapperStyle(\n position.side,\n position.arrowCenter,\n )}\n >\n <TooltipArrowGraphic side={position.side} />\n </span>\n ) : null}\n <div className=\"relative z-[2] max-w-full\">{children}</div>\n </div>,\n resolvedContainer,\n );\n },\n);\nTooltipContent.displayName = \"TooltipContent\";\n"]}
|