import React, { Dispatch, SetStateAction, useState } from "react"; import { Slot, VariantProps, vcn } from "../shared"; import ReactDOM from "react-dom"; /** * ========================= * DialogContext * ========================= */ interface DialogContext { opened: boolean; } const initialDialogContext: DialogContext = { opened: false }; const DialogContext = React.createContext< [DialogContext, Dispatch>] >([ initialDialogContext, () => { if (process.env.NODE_ENV && process.env.NODE_ENV === "development") { console.warn( "It seems like you're using DialogContext outside of a provider." ); } }, ]); const useDialogContext = () => React.useContext(DialogContext); /** * ========================= * DialogRoot * ========================= */ interface DialogRootProps { children: React.ReactNode; } const DialogRoot = ({ children }: DialogRootProps) => { const state = useState(initialDialogContext); return ( {children} ); }; /** * ========================= * DialogTrigger * ========================= */ interface DialogTriggerProps { children: React.ReactNode; } const DialogTrigger = ({ children }: DialogTriggerProps) => { const [_, setState] = useDialogContext(); const onClick = () => setState((p) => ({ ...p, opened: true })); const slotProps = { onClick, children, }; return ; }; /** * ========================= * DialogOverlay * ========================= */ const [dialogOverlayVariant, resolveDialogOverlayVariant] = vcn({ base: "fixed inset-0 z-50 w-full h-full max-w-screen transition-all duration-300 flex flex-col justify-center items-center", variants: { opened: { true: "pointer-events-auto opacity-100", false: "pointer-events-none opacity-0", }, blur: { sm: "backdrop-blur-sm", md: "backdrop-blur-md", lg: "backdrop-blur-lg", }, darken: { sm: "backdrop-brightness-90", md: "backdrop-brightness-75", lg: "backdrop-brightness-50", }, padding: { sm: "p-4", md: "p-6", lg: "p-8", }, }, defaults: { opened: false, blur: "md", darken: "md", padding: "md", }, }); interface DialogOverlay extends React.ComponentPropsWithoutRef<"div">, Omit, "opened"> { closeOnClick?: boolean; } const DialogOverlay = React.forwardRef( (props, ref) => { const [{ opened }, 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); }} > {children}
, document.body )} ); } ); /** * ========================= * DialogContent * ========================= */ const [dialogContentVariant, resolveDialogContentVariant] = vcn({ base: "transition-transform duration-300 bg-white dark:bg-black border border-neutral-200 dark:border-neutral-800", variants: { opened: { true: "scale-100", false: "scale-50", }, size: { fit: "w-fit", fullSm: "w-full max-w-sm", fullMd: "w-full max-w-md", fullLg: "w-full max-w-lg", fullXl: "w-full max-w-xl", full2xl: "w-full max-w-2xl", }, rounded: { sm: "rounded-sm", md: "rounded-md", lg: "rounded-lg", xl: "rounded-xl", }, padding: { sm: "p-4", md: "p-6", lg: "p-8", }, gap: { sm: "space-y-4", md: "space-y-6", lg: "space-y-8", }, }, defaults: { opened: false, size: "fit", rounded: "md", padding: "md", gap: "md", }, }); interface DialogContent extends React.ComponentPropsWithoutRef<"div">, Omit, "opened"> {} const DialogContent = React.forwardRef( (props, ref) => { const [{ opened }] = useDialogContext(); const [variantProps, otherPropsCompressed] = resolveDialogContentVariant({ ...props, opened, }); const { children, ...otherPropsExtracted } = otherPropsCompressed; return (
{children}
); } ); /** * ========================= * 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", variants: { gap: { sm: "gap-2", md: "gap-4", lg: "gap-6", }, }, defaults: { gap: "sm", }, }); interface DialogHeaderProps extends React.ComponentPropsWithoutRef<"header">, VariantProps {} const DialogHeader = React.forwardRef( (props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogHeaderVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; return (
{children}
); } ); /** * ========================= * DialogTitle / DialogSubtitle * ========================= */ const [dialogTitleVariant, resolveDialogTitleVariant] = vcn({ variants: { size: { sm: "text-lg", md: "text-xl", lg: "text-2xl", }, weight: { sm: "font-medium", md: "font-semibold", lg: "font-bold", }, }, defaults: { size: "md", weight: "lg", }, }); interface DialogTitleProps extends React.ComponentPropsWithoutRef<"h1">, VariantProps {} const [dialogSubtitleVariant, resolveDialogSubtitleVariant] = vcn({ variants: { size: { sm: "text-sm", md: "text-base", lg: "text-lg", }, opacity: { sm: "opacity-60", md: "opacity-70", lg: "opacity-80", }, weight: { sm: "font-light", md: "font-normal", lg: "font-medium", }, }, defaults: { size: "sm", opacity: "sm", weight: "md", }, }); interface DialogSubtitleProps extends React.ComponentPropsWithoutRef<"h2">, VariantProps {} const DialogTitle = React.forwardRef( (props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogTitleVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; return (

{children}

); } ); const DialogSubtitle = React.forwardRef< HTMLHeadingElement, DialogSubtitleProps >((props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogSubtitleVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; return (

{children}

); }); /** * ========================= * DialogFooter * ========================= */ const [dialogFooterVariant, resolveDialogFooterVariant] = vcn({ base: "flex flex-col items-end sm:flex-row sm:items-center sm:justify-end", variants: { gap: { sm: "gap-2", md: "gap-4", lg: "gap-6", }, }, defaults: { gap: "md", }, }); interface DialogFooterProps extends React.ComponentPropsWithoutRef<"footer">, VariantProps {} const DialogFooter = React.forwardRef( (props, ref) => { const [variantProps, otherPropsCompressed] = resolveDialogFooterVariant(props); const { children, ...otherPropsExtracted } = otherPropsCompressed; return (
{children}
); } ); export { useDialogContext, DialogRoot, DialogTrigger, DialogOverlay, DialogContent, DialogClose, DialogHeader, DialogTitle, DialogSubtitle, DialogFooter, };