"use client"
import type {
DialogTriggerProps,
PopoverProps as PopoverPrimitiveProps,
} from "react-aria-components"
import {
DialogTrigger as DialogTriggerPrimitive,
OverlayArrow,
Popover as PopoverPrimitive,
} from "react-aria-components"
import { twMerge } from "tailwind-merge"
import { cx } from "@/lib/primitive"
import {
DialogBody,
DialogClose,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "./dialog"
type PopoverProps = DialogTriggerProps
const Popover = (props: PopoverProps) => {
return <DialogTriggerPrimitive {...props} />
}
const PopoverTitle = DialogTitle
const PopoverHeader = DialogHeader
const PopoverBody = DialogBody
const PopoverFooter = ({ className, ...props }: React.ComponentProps<typeof DialogFooter>) => (
<DialogFooter
className={twMerge("justify-start has-[button]:justify-end", className)}
{...props}
/>
)
interface PopoverContentProps extends PopoverPrimitiveProps {
arrow?: boolean
ref?: React.Ref<HTMLDivElement>
}
const PopoverContent = ({
children,
arrow = false,
className,
ref,
...props
}: PopoverContentProps) => {
const offset = props.offset ?? (arrow ? 12 : 8)
return (
<PopoverPrimitive
ref={ref}
offset={offset}
className={cx(
"group/popover min-w-(--trigger-width) max-w-xs origin-(--trigger-anchor-point) rounded-(--popover-radius) border border-fg/10 bg-overlay text-overlay-fg shadow-xs outline-hidden transition-transform [--gutter:--spacing(6)] [--popover-radius:var(--radius-xl)] sm:text-sm dark:backdrop-saturate-200 **:[[role=dialog]]:[--gutter:--spacing(6)]",
"entering:fade-in exiting:fade-out entering:animate-in exiting:animate-out",
"placement-left:entering:slide-in-from-right-1 placement-right:entering:slide-in-from-left-1 placement-top:entering:slide-in-from-bottom-1 placement-bottom:entering:slide-in-from-top-1",
"placement-left:exiting:slide-out-to-right-1 placement-right:exiting:slide-out-to-left-1 placement-top:exiting:slide-out-to-bottom-1 placement-bottom:exiting:slide-out-to-top-1",
"forced-colors:bg-[Canvas]",
className,
)}
{...props}
>
{(values) => (
<>
{arrow && (
<OverlayArrow className="group">
<svg
width={12}
height={12}
viewBox="0 0 12 12"
className="block fill-overlay stroke-border group-placement-bottom:rotate-180 group-placement-left:-rotate-90 group-placement-right:rotate-90 forced-colors:fill-[Canvas] forced-colors:stroke-[ButtonBorder]"
>
<path d="M0 0 L6 6 L12 0" />
</svg>
</OverlayArrow>
)}
<div data-slot="popover-inner" className="max-h-[inherit] overflow-y-auto">
{typeof children === "function" ? children(values) : children}
</div>
</>
)}
</PopoverPrimitive>
)
}
const PopoverTrigger = DialogTrigger
const PopoverClose = DialogClose
const PopoverDescription = DialogDescription
export type { PopoverProps, PopoverContentProps }
export {
Popover,
PopoverTrigger,
PopoverClose,
PopoverDescription,
PopoverContent,
PopoverBody,
PopoverFooter,
PopoverHeader,
PopoverTitle,
}