Files
LangBot/web/src/components/ui/tooltip.tsx
T
RockChinQ b3848c9d05 feat(web): make tooltips tap-toggleable on touch devices
Radix tooltips open on hover/focus only and stay closed on touch input,
so on mobile every hover tooltip was unreachable. Detect coarse/no-hover
pointers via matchMedia and drive the tooltip's open state ourselves so a
tap on the trigger toggles it. Desktop hover/focus behaviour is unchanged
(we only intercept the tap when the device has no hover capability). Fixes
all tooltips app-wide from the shared primitive.
2026-06-21 12:46:18 -04:00

135 lines
4.1 KiB
TypeScript

import * as React from 'react';
import * as TooltipPrimitive from '@radix-ui/react-tooltip';
import { cn } from '@/lib/utils';
// Radix tooltips open on hover/focus only and deliberately stay closed on
// touch input. To make every tooltip usable on mobile, we expose a small
// context so the trigger can toggle the tooltip open on tap when the device
// has no hover capability (coarse / touch pointer).
interface TooltipTouchContextValue {
isTouch: boolean;
open: boolean;
setOpen: (open: boolean) => void;
}
const TooltipTouchContext =
React.createContext<TooltipTouchContextValue | null>(null);
function useIsTouchDevice(): boolean {
const [isTouch, setIsTouch] = React.useState(false);
React.useEffect(() => {
if (typeof window === 'undefined' || !window.matchMedia) return;
const mq = window.matchMedia('(hover: none), (pointer: coarse)');
const update = () => setIsTouch(mq.matches);
update();
mq.addEventListener?.('change', update);
return () => mq.removeEventListener?.('change', update);
}, []);
return isTouch;
}
function TooltipProvider({
delayDuration = 0,
...props
}: React.ComponentProps<typeof TooltipPrimitive.Provider>) {
return (
<TooltipPrimitive.Provider
data-slot="tooltip-provider"
delayDuration={delayDuration}
{...props}
/>
);
}
function Tooltip({
open: openProp,
onOpenChange,
...props
}: React.ComponentProps<typeof TooltipPrimitive.Root>) {
const isTouch = useIsTouchDevice();
const [openState, setOpenState] = React.useState(false);
const isControlled = openProp !== undefined;
const open = isControlled ? (openProp ?? false) : openState;
const setOpen = React.useCallback(
(next: boolean) => {
if (!isControlled) setOpenState(next);
onOpenChange?.(next);
},
[isControlled, onOpenChange],
);
return (
<TooltipProvider>
<TooltipTouchContext.Provider value={{ isTouch, open, setOpen }}>
{/* Drive open state ourselves so we can toggle on tap for touch
devices while still forwarding Radix's hover/focus changes on
desktop. */}
<TooltipPrimitive.Root
data-slot="tooltip"
open={open}
onOpenChange={setOpen}
{...props}
/>
</TooltipTouchContext.Provider>
</TooltipProvider>
);
}
function TooltipTrigger({
onClick,
...props
}: React.ComponentProps<typeof TooltipPrimitive.Trigger>) {
const ctx = React.useContext(TooltipTouchContext);
const handleClick = React.useCallback(
(event: React.MouseEvent<HTMLButtonElement>) => {
// On touch devices Radix never opens the tooltip via hover, so a tap on
// the trigger toggles it. The underlying element's own onClick still
// fires (e.g. an actionable button keeps working).
if (ctx?.isTouch) {
ctx.setOpen(!ctx.open);
}
onClick?.(event);
},
[ctx, onClick],
);
return (
<TooltipPrimitive.Trigger
data-slot="tooltip-trigger"
onClick={handleClick}
{...props}
/>
);
}
function TooltipContent({
className,
sideOffset = 0,
children,
...props
}: React.ComponentProps<typeof TooltipPrimitive.Content>) {
return (
<TooltipPrimitive.Portal>
<TooltipPrimitive.Content
data-slot="tooltip-content"
sideOffset={sideOffset}
className={cn(
'bg-primary text-primary-foreground animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-fit origin-(--radix-tooltip-content-transform-origin) rounded-md px-3 py-1.5 text-xs text-balance',
className,
)}
{...props}
>
{children}
<TooltipPrimitive.Arrow className="bg-primary fill-primary z-50 size-2.5 translate-y-[calc(-50%_-_2px)] rotate-45 rounded-[2px]" />
</TooltipPrimitive.Content>
</TooltipPrimitive.Portal>
);
}
export { Tooltip, TooltipTrigger, TooltipContent, TooltipProvider };