223 lines
6.9 KiB
JavaScript
223 lines
6.9 KiB
JavaScript
"use client"
|
|
|
|
import * as React from "react"
|
|
import { Drawer as DrawerPrimitive } from "vaul"
|
|
|
|
import { cn } from "@/lib/utils"
|
|
|
|
function parseVhToPx(vhStr) {
|
|
if (typeof vhStr === "number") return vhStr
|
|
const match = String(vhStr).match(/^([\d.]+)\s*vh$/)
|
|
if (!match) return null
|
|
return (window.innerHeight * Number(match[1])) / 100
|
|
}
|
|
|
|
function Drawer({
|
|
...props
|
|
}) {
|
|
return <DrawerPrimitive.Root data-slot="drawer" {...props} />;
|
|
}
|
|
|
|
function DrawerTrigger({
|
|
...props
|
|
}) {
|
|
return <DrawerPrimitive.Trigger data-slot="drawer-trigger" {...props} />;
|
|
}
|
|
|
|
function DrawerPortal({
|
|
...props
|
|
}) {
|
|
return <DrawerPrimitive.Portal data-slot="drawer-portal" {...props} />;
|
|
}
|
|
|
|
function DrawerClose({
|
|
...props
|
|
}) {
|
|
return <DrawerPrimitive.Close data-slot="drawer-close" {...props} />;
|
|
}
|
|
|
|
function DrawerOverlay({
|
|
className,
|
|
...props
|
|
}) {
|
|
return (
|
|
<DrawerPrimitive.Overlay
|
|
data-slot="drawer-overlay"
|
|
className={cn(
|
|
"fixed inset-0 z-50 bg-[var(--drawer-overlay)] backdrop-blur-[4px] data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:animate-in data-[state=open]:fade-in-0",
|
|
className
|
|
)}
|
|
{...props} />
|
|
);
|
|
}
|
|
|
|
function DrawerContent({
|
|
className,
|
|
children,
|
|
defaultHeight = "77vh",
|
|
minHeight = "20vh",
|
|
maxHeight = "90vh",
|
|
...props
|
|
}) {
|
|
const [heightPx, setHeightPx] = React.useState(() =>
|
|
typeof window !== "undefined" ? parseVhToPx(defaultHeight) : null
|
|
);
|
|
const [isDragging, setIsDragging] = React.useState(false);
|
|
const dragRef = React.useRef({ startY: 0, startHeight: 0 });
|
|
|
|
const minPx = React.useMemo(() => parseVhToPx(minHeight), [minHeight]);
|
|
const maxPx = React.useMemo(() => parseVhToPx(maxHeight), [maxHeight]);
|
|
|
|
React.useEffect(() => {
|
|
const px = parseVhToPx(defaultHeight);
|
|
if (px != null) setHeightPx(px);
|
|
}, [defaultHeight]);
|
|
|
|
React.useEffect(() => {
|
|
const sync = () => {
|
|
const max = parseVhToPx(maxHeight);
|
|
const min = parseVhToPx(minHeight);
|
|
setHeightPx((prev) => {
|
|
if (prev == null) return parseVhToPx(defaultHeight);
|
|
const clamped = Math.min(prev, max ?? prev);
|
|
return Math.max(clamped, min ?? clamped);
|
|
});
|
|
};
|
|
window.addEventListener("resize", sync);
|
|
return () => window.removeEventListener("resize", sync);
|
|
}, [defaultHeight, minHeight, maxHeight]);
|
|
|
|
const handlePointerDown = React.useCallback(
|
|
(e) => {
|
|
e.preventDefault();
|
|
setIsDragging(true);
|
|
dragRef.current = { startY: e.clientY ?? e.touches?.[0]?.clientY, startHeight: heightPx ?? parseVhToPx(defaultHeight) ?? 0 };
|
|
},
|
|
[heightPx, defaultHeight]
|
|
);
|
|
|
|
React.useEffect(() => {
|
|
if (!isDragging) return;
|
|
const move = (e) => {
|
|
const clientY = e.clientY ?? e.touches?.[0]?.clientY;
|
|
const { startY, startHeight } = dragRef.current;
|
|
const delta = startY - clientY;
|
|
const next = Math.min(maxPx ?? Infinity, Math.max(minPx ?? 0, startHeight + delta));
|
|
setHeightPx(next);
|
|
};
|
|
const up = () => setIsDragging(false);
|
|
document.addEventListener("mousemove", move, { passive: true });
|
|
document.addEventListener("mouseup", up);
|
|
document.addEventListener("touchmove", move, { passive: true });
|
|
document.addEventListener("touchend", up);
|
|
return () => {
|
|
document.removeEventListener("mousemove", move);
|
|
document.removeEventListener("mouseup", up);
|
|
document.removeEventListener("touchmove", move);
|
|
document.removeEventListener("touchend", up);
|
|
};
|
|
}, [isDragging, minPx, maxPx]);
|
|
|
|
const contentStyle = React.useMemo(() => {
|
|
if (heightPx == null) return undefined;
|
|
return { height: `${heightPx}px`, maxHeight: maxPx != null ? `${maxPx}px` : undefined };
|
|
}, [heightPx, maxPx]);
|
|
|
|
return (
|
|
<DrawerPortal data-slot="drawer-portal">
|
|
<DrawerOverlay />
|
|
<DrawerPrimitive.Content
|
|
data-slot="drawer-content"
|
|
style={contentStyle}
|
|
className={cn(
|
|
"group/drawer-content fixed z-50 flex h-auto flex-col bg-[var(--card)] text-[var(--text)] border-[var(--border)]",
|
|
"data-[vaul-drawer-direction=top]:inset-x-0 data-[vaul-drawer-direction=top]:top-0 data-[vaul-drawer-direction=top]:mb-24 data-[vaul-drawer-direction=top]:max-h-[80vh] data-[vaul-drawer-direction=top]:rounded-b-[var(--radius)] data-[vaul-drawer-direction=top]:border-b drawer-shadow-top",
|
|
"data-[vaul-drawer-direction=bottom]:inset-x-0 data-[vaul-drawer-direction=bottom]:bottom-0 data-[vaul-drawer-direction=bottom]:mt-24 data-[vaul-drawer-direction=bottom]:max-h-[88vh] data-[vaul-drawer-direction=bottom]:rounded-t-[20px] data-[vaul-drawer-direction=bottom]:border-t drawer-shadow-bottom",
|
|
"data-[vaul-drawer-direction=right]:inset-y-0 data-[vaul-drawer-direction=right]:right-0 data-[vaul-drawer-direction=right]:w-3/4 data-[vaul-drawer-direction=right]:border-l data-[vaul-drawer-direction=right]:sm:max-w-sm",
|
|
"data-[vaul-drawer-direction=left]:inset-y-0 data-[vaul-drawer-direction=left]:left-0 data-[vaul-drawer-direction=left]:w-3/4 data-[vaul-drawer-direction=left]:border-r data-[vaul-drawer-direction=left]:sm:max-w-sm",
|
|
"drawer-content-theme",
|
|
className
|
|
)}
|
|
{...props}>
|
|
<div
|
|
role="separator"
|
|
aria-label="拖动调整高度"
|
|
onMouseDown={handlePointerDown}
|
|
onTouchStart={handlePointerDown}
|
|
className={cn(
|
|
"mx-auto mt-4 hidden h-2 w-[100px] shrink-0 rounded-full bg-[var(--muted)] cursor-n-resize touch-none select-none",
|
|
"group-data-[vaul-drawer-direction=bottom]/drawer-content:block",
|
|
"hover:bg-[var(--muted-foreground)/0.4] active:bg-[var(--muted-foreground)/0.6]"
|
|
)}
|
|
/>
|
|
{children}
|
|
</DrawerPrimitive.Content>
|
|
</DrawerPortal>
|
|
);
|
|
}
|
|
|
|
function DrawerHeader({
|
|
className,
|
|
...props
|
|
}) {
|
|
return (
|
|
<div
|
|
data-slot="drawer-header"
|
|
className={cn(
|
|
"flex flex-col gap-0.5 p-4 border-b border-[var(--border)] group-data-[vaul-drawer-direction=bottom]/drawer-content:text-center group-data-[vaul-drawer-direction=top]/drawer-content:text-center md:gap-1.5 md:text-left",
|
|
"drawer-header-theme",
|
|
className
|
|
)}
|
|
{...props} />
|
|
);
|
|
}
|
|
|
|
function DrawerFooter({
|
|
className,
|
|
...props
|
|
}) {
|
|
return (
|
|
<div
|
|
data-slot="drawer-footer"
|
|
className={cn("mt-auto flex flex-col gap-2 p-4", className)}
|
|
{...props} />
|
|
);
|
|
}
|
|
|
|
function DrawerTitle({
|
|
className,
|
|
...props
|
|
}) {
|
|
return (
|
|
<DrawerPrimitive.Title
|
|
data-slot="drawer-title"
|
|
className={cn("font-semibold text-[var(--text)]", className)}
|
|
{...props} />
|
|
);
|
|
}
|
|
|
|
function DrawerDescription({
|
|
className,
|
|
...props
|
|
}) {
|
|
return (
|
|
<DrawerPrimitive.Description
|
|
data-slot="drawer-description"
|
|
className={cn("text-sm text-[var(--muted)]", className)}
|
|
{...props} />
|
|
);
|
|
}
|
|
|
|
export {
|
|
Drawer,
|
|
DrawerPortal,
|
|
DrawerOverlay,
|
|
DrawerTrigger,
|
|
DrawerClose,
|
|
DrawerContent,
|
|
DrawerHeader,
|
|
DrawerFooter,
|
|
DrawerTitle,
|
|
DrawerDescription,
|
|
}
|