feat: unmount dialog from dom on closed

This commit is contained in:
p-sw 2024-08-07 18:58:53 +09:00
parent 8728df1f60
commit 4b6724ed90
2 changed files with 82 additions and 36 deletions

View File

@ -1,12 +1,20 @@
import { Slot, type VariantProps, useDocument, vcn } from "@pswui-lib"; import {
import React, { type ReactNode, useId, useState } from "react"; Slot,
type VariantProps,
useAnimatedMount,
useDocument,
vcn,
} from "@pswui-lib";
import React, { type ReactNode, useId, useRef, useState } from "react";
import ReactDOM from "react-dom"; import ReactDOM from "react-dom";
import { import {
DialogContext, DialogContext,
type IDialogContext, type IDialogContext,
InnerDialogContext,
initialDialogContext, initialDialogContext,
useDialogContext, useDialogContext,
useInnerDialogContext,
} from "./Context"; } from "./Context";
/** /**
@ -81,22 +89,35 @@ interface DialogOverlay
const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>( const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
(props, ref) => { (props, ref) => {
const [{ opened, ids }, setContext] = useDialogContext(); const [{ opened, ids }, setContext] = useDialogContext();
const [variantProps, otherPropsCompressed] = resolveDialogOverlayVariant({ const [variantProps, otherPropsCompressed] =
...props, resolveDialogOverlayVariant(props);
opened,
});
const { children, closeOnClick, onClick, ...otherPropsExtracted } = const { children, closeOnClick, onClick, ...otherPropsExtracted } =
otherPropsCompressed; otherPropsCompressed;
const internalRef = useRef<HTMLDivElement | null>(null);
const { isMounted, isRendered } = useAnimatedMount(opened, internalRef);
const document = useDocument(); const document = useDocument();
if (!document) return null; if (!document) return null;
return ReactDOM.createPortal( return isMounted
? ReactDOM.createPortal(
<div <div
{...otherPropsExtracted} {...otherPropsExtracted}
id={ids.dialog} id={ids.dialog}
ref={ref} ref={(el) => {
className={dialogOverlayVariant(variantProps)} internalRef.current = el;
if (typeof ref === "function") {
ref(el);
} else if (ref) {
ref.current = el;
}
}}
className={dialogOverlayVariant({
...variantProps,
opened: isRendered,
})}
onClick={(e) => { onClick={(e) => {
if (closeOnClick) { if (closeOnClick) {
setContext((p) => ({ ...p, opened: false })); setContext((p) => ({ ...p, opened: false }));
@ -104,17 +125,20 @@ const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
onClick?.(e); onClick?.(e);
}} }}
> >
{/* Layer for overflow positioning */}
<div <div
className={ className={
"w-screen max-w-full min-h-screen flex flex-col justify-center items-center" "w-screen max-w-full min-h-screen flex flex-col justify-center items-center"
} }
> >
{/* Layer for overflow positioning */} <InnerDialogContext.Provider value={{ isMounted, isRendered }}>
{children} {children}
</InnerDialogContext.Provider>
</div> </div>
</div>, </div>,
document.body, document.body,
); )
: null;
}, },
); );
DialogOverlay.displayName = "DialogOverlay"; DialogOverlay.displayName = "DialogOverlay";
@ -144,11 +168,10 @@ interface DialogContentProps
const DialogContent = React.forwardRef<HTMLDivElement, DialogContentProps>( const DialogContent = React.forwardRef<HTMLDivElement, DialogContentProps>(
(props, ref) => { (props, ref) => {
const [{ opened, ids }] = useDialogContext(); const [{ ids }] = useDialogContext();
const [variantProps, otherPropsCompressed] = resolveDialogContentVariant({ const [variantProps, otherPropsCompressed] =
...props, resolveDialogContentVariant(props);
opened, const { isRendered } = useInnerDialogContext();
});
const { children, onClick, ...otherPropsExtracted } = otherPropsCompressed; const { children, onClick, ...otherPropsExtracted } = otherPropsCompressed;
return ( return (
<div <div
@ -157,7 +180,10 @@ const DialogContent = React.forwardRef<HTMLDivElement, DialogContentProps>(
role="dialog" role="dialog"
aria-labelledby={ids.title} aria-labelledby={ids.title}
aria-describedby={ids.description} aria-describedby={ids.description}
className={dialogContentVariant(variantProps)} className={dialogContentVariant({
...variantProps,
opened: isRendered,
})}
onClick={(e) => { onClick={(e) => {
e.stopPropagation(); e.stopPropagation();
onClick?.(e); onClick?.(e);

View File

@ -38,3 +38,23 @@ export const DialogContext = createContext<
]); ]);
export const useDialogContext = () => useContext(DialogContext); 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<IInnerDialogContext>(
initialInnerDialogContext,
);
export const useInnerDialogContext = () => useContext(InnerDialogContext);