import { ServerSideDocumentFallback, Slot, type VariantProps, vcn, } from "@pswui-lib"; import React, { type ReactNode, useId, useState } from "react"; import ReactDOM from "react-dom"; import { DialogContext, type IDialogContext, initialDialogContext, useDialogContext, } from "./Context"; /** * ========================= * DialogRoot * ========================= */ interface DialogRootProps { children: React.ReactNode; } const DialogRoot = ({ children }: DialogRootProps) => { const state = useState({ ...initialDialogContext, ids: { dialog: useId(), title: useId(), description: useId() }, }); return ( {children} ); }; /** * ========================= * DialogTrigger * ========================= */ interface DialogTriggerProps { children: React.ReactNode; } const DialogTrigger = ({ children }: DialogTriggerProps) => { const [{ ids }, setState] = useDialogContext(); const onClick = () => setState((p) => ({ ...p, opened: true })); return ( {children} ); }; /** * ========================= * DialogOverlay * ========================= */ const [dialogOverlayVariant, resolveDialogOverlayVariant] = vcn({ base: "fixed inset-0 z-50 w-full h-screen overflow-y-auto max-w-screen transition-all duration-300 backdrop-blur-md backdrop-brightness-75 [&>div]:p-6", variants: { opened: { true: "pointer-events-auto opacity-100", false: "pointer-events-none opacity-0", }, }, defaults: { opened: false, }, }); interface DialogOverlay extends React.ComponentPropsWithoutRef<"div">, Omit, "opened"> { closeOnClick?: boolean; } const DialogOverlay = React.forwardRef( (props, ref) => { const [{ opened, ids }, setContext] = useDialogContext(); const [variantProps, otherPropsCompressed] = resolveDialogOverlayVariant({ ...props, opened, }); const { children, closeOnClick, onClick, ...otherPropsExtracted } = otherPropsCompressed; return ( {() => ReactDOM.createPortal(
{ if (closeOnClick) { setContext((p) => ({ ...p, opened: false })); } onClick?.(e); }} >
{/* Layer for overflow positioning */} {children}
, document.body, ) }
); }, ); DialogOverlay.displayName = "DialogOverlay"; /** * ========================= * DialogContent * ========================= */ const [dialogContentVariant, resolveDialogContentVariant] = vcn({ base: "transition-transform duration-300 bg-white dark:bg-black border border-neutral-200 dark:border-neutral-800 p-6 w-full max-w-xl rounded-md flex flex-col justify-start items-start gap-6", variants: { opened: { true: "scale-100", false: "scale-50", }, }, defaults: { opened: false, }, }); interface DialogContentProps extends React.ComponentPropsWithoutRef<"div">, Omit, "opened"> {} const DialogContent = React.forwardRef( (props, ref) => { const [{ opened, ids }] = useDialogContext(); const [variantProps, otherPropsCompressed] = resolveDialogContentVariant({ ...props, opened, }); const { children, onClick, ...otherPropsExtracted } = otherPropsCompressed; return (
{ e.stopPropagation(); onClick?.(e); }} > {children}
); }, ); DialogContent.displayName = "DialogContent"; /** * ========================= * DialogClose * ========================= */ interface DialogCloseProps { children: React.ReactNode; } const DialogClose = ({ children }: DialogCloseProps) => { const [_, setState] = useDialogContext(); const onClick = () => setState((p) => ({ ...p, opened: false })); const slotProps = { onClick, children, }; return ; }; /** * ========================= * DialogHeader * ========================= */ const [dialogHeaderVariant, resolveDialogHeaderVariant] = vcn({ base: "flex flex-col gap-2", variants: {}, defaults: {}, }); interface DialogHeaderProps extends React.ComponentPropsWithoutRef<"header">, VariantProps {} const DialogHeader = React.forwardRef( (props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogHeaderVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; return (
{children}
); }, ); DialogHeader.displayName = "DialogHeader"; /** * ========================= * DialogTitle / DialogSubtitle * ========================= */ const [dialogTitleVariant, resolveDialogTitleVariant] = vcn({ base: "text-xl font-bold", variants: {}, defaults: {}, }); interface DialogTitleProps extends React.ComponentPropsWithoutRef<"h1">, VariantProps {} const [dialogDescriptionVariant, resolveDialogDescriptionVariant] = vcn({ base: "text-sm opacity-60 font-normal", variants: {}, defaults: {}, }); interface DialogDescriptionProps extends React.ComponentPropsWithoutRef<"h2">, VariantProps {} const DialogTitle = React.forwardRef( (props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogTitleVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; const [{ ids }] = useDialogContext(); return (

{children}

); }, ); DialogTitle.displayName = "DialogTitle"; const DialogDescription = React.forwardRef< HTMLHeadingElement, DialogDescriptionProps >((props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogDescriptionVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; const [{ ids }] = useDialogContext(); return (

{children}

); }); DialogDescription.displayName = "DialogDescription"; // renamed DialogSubtitle -> DialogDescription // keep DialogSubtitle for backward compatibility const DialogSubtitle = DialogDescription; /** * ========================= * DialogFooter * ========================= */ const [dialogFooterVariant, resolveDialogFooterVariant] = vcn({ base: "flex w-full flex-col items-end sm:flex-row sm:items-center sm:justify-end gap-2", variants: {}, defaults: {}, }); interface DialogFooterProps extends React.ComponentPropsWithoutRef<"footer">, VariantProps {} const DialogFooter = React.forwardRef( (props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogFooterVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; return (
{children}
); }, ); DialogFooter.displayName = "DialogFooter"; interface DialogControllers { context: IDialogContext; setContext: React.Dispatch>; close: () => void; } interface DialogControllerProps { children: (controllers: DialogControllers) => ReactNode; } const DialogController = (props: DialogControllerProps) => { return ( {([context, setContext]) => props.children({ context, setContext, close() { setContext((p) => ({ ...p, opened: false })); }, }) } ); }; export { DialogRoot, DialogTrigger, DialogOverlay, DialogContent, DialogClose, DialogHeader, DialogTitle, DialogSubtitle, DialogDescription, DialogFooter, DialogController, };