diff --git a/packages/react/components/Dialog/Component.tsx b/packages/react/components/Dialog/Component.tsx index 888efdd..d2dbea8 100644 --- a/packages/react/components/Dialog/Component.tsx +++ b/packages/react/components/Dialog/Component.tsx @@ -1,12 +1,20 @@ -import { Slot, type VariantProps, useDocument, vcn } from "@pswui-lib"; -import React, { type ReactNode, useId, useState } from "react"; +import { + Slot, + type VariantProps, + useAnimatedMount, + useDocument, + vcn, +} from "@pswui-lib"; +import React, { type ReactNode, useId, useRef, useState } from "react"; import ReactDOM from "react-dom"; import { DialogContext, type IDialogContext, + InnerDialogContext, initialDialogContext, useDialogContext, + useInnerDialogContext, } from "./Context"; /** @@ -81,40 +89,56 @@ interface DialogOverlay const DialogOverlay = React.forwardRef( (props, ref) => { const [{ opened, ids }, setContext] = useDialogContext(); - const [variantProps, otherPropsCompressed] = resolveDialogOverlayVariant({ - ...props, - opened, - }); + const [variantProps, otherPropsCompressed] = + resolveDialogOverlayVariant(props); const { children, closeOnClick, onClick, ...otherPropsExtracted } = otherPropsCompressed; + const internalRef = useRef(null); + + const { isMounted, isRendered } = useAnimatedMount(opened, internalRef); + const document = useDocument(); if (!document) return null; - return ReactDOM.createPortal( -
{ - if (closeOnClick) { - setContext((p) => ({ ...p, opened: false })); - } - onClick?.(e); - }} - > -
- {/* Layer for overflow positioning */} - {children} -
-
, - document.body, - ); + return isMounted + ? ReactDOM.createPortal( +
{ + internalRef.current = el; + if (typeof ref === "function") { + ref(el); + } else if (ref) { + ref.current = el; + } + }} + className={dialogOverlayVariant({ + ...variantProps, + opened: isRendered, + })} + onClick={(e) => { + if (closeOnClick) { + setContext((p) => ({ ...p, opened: false })); + } + onClick?.(e); + }} + > + {/* Layer for overflow positioning */} +
+ + {children} + +
+
, + document.body, + ) + : null; }, ); DialogOverlay.displayName = "DialogOverlay"; @@ -144,11 +168,10 @@ interface DialogContentProps const DialogContent = React.forwardRef( (props, ref) => { - const [{ opened, ids }] = useDialogContext(); - const [variantProps, otherPropsCompressed] = resolveDialogContentVariant({ - ...props, - opened, - }); + const [{ ids }] = useDialogContext(); + const [variantProps, otherPropsCompressed] = + resolveDialogContentVariant(props); + const { isRendered } = useInnerDialogContext(); const { children, onClick, ...otherPropsExtracted } = otherPropsCompressed; return (
( role="dialog" aria-labelledby={ids.title} aria-describedby={ids.description} - className={dialogContentVariant(variantProps)} + className={dialogContentVariant({ + ...variantProps, + opened: isRendered, + })} onClick={(e) => { e.stopPropagation(); onClick?.(e); diff --git a/packages/react/components/Dialog/Context.ts b/packages/react/components/Dialog/Context.ts index b28feb9..fe4c5c1 100644 --- a/packages/react/components/Dialog/Context.ts +++ b/packages/react/components/Dialog/Context.ts @@ -38,3 +38,23 @@ export const DialogContext = createContext< ]); export const useDialogContext = () => useContext(DialogContext); + +/** + * =================== + * InnerDialogContext + * =================== + */ + +export interface IInnerDialogContext { + isMounted: boolean; + isRendered: boolean; +} +export const initialInnerDialogContext: IInnerDialogContext = { + isMounted: false, + isRendered: false, +}; +export const InnerDialogContext = createContext( + initialInnerDialogContext, +); + +export const useInnerDialogContext = () => useContext(InnerDialogContext);