feat: add dialog
This commit is contained in:
parent
08b111ab4f
commit
56ae87cd81
415
packages/react/components/Dialog.tsx
Normal file
415
packages/react/components/Dialog.tsx
Normal file
@ -0,0 +1,415 @@
|
|||||||
|
import React, {
|
||||||
|
Dispatch,
|
||||||
|
MouseEventHandler,
|
||||||
|
SetStateAction,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
|
import { 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<SetStateAction<DialogContext>>]
|
||||||
|
>([initialDialogContext, () => {}]);
|
||||||
|
|
||||||
|
const useDialogContext = () => React.useContext(DialogContext);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* DialogRoot
|
||||||
|
* =========================
|
||||||
|
*/
|
||||||
|
|
||||||
|
interface DialogRootProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DialogRoot = ({ children }: DialogRootProps) => {
|
||||||
|
const state = useState<DialogContext>(initialDialogContext);
|
||||||
|
return (
|
||||||
|
<DialogContext.Provider value={state}>{children}</DialogContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* DialogTrigger
|
||||||
|
* =========================
|
||||||
|
*/
|
||||||
|
|
||||||
|
interface DialogTriggerProps<T extends React.ReactNode> {
|
||||||
|
children: T extends any[] ? never : T;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DialogTrigger = <T extends React.ReactNode>({
|
||||||
|
children,
|
||||||
|
}: DialogTriggerProps<T>) => {
|
||||||
|
const [_, setState] = useDialogContext();
|
||||||
|
// const onClick = () => setState((p) => ({ ...p, opened: true }));
|
||||||
|
|
||||||
|
const child = React.Children.only(children) as React.ReactElement;
|
||||||
|
const onClick: MouseEventHandler<HTMLButtonElement> = (e) => {
|
||||||
|
child.props.onClick?.(e);
|
||||||
|
setState((p) => ({ ...p, opened: true }));
|
||||||
|
};
|
||||||
|
return <>{React.cloneElement(child, { onClick })}</>;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* 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<VariantProps<typeof dialogOverlayVariant>, "opened"> {
|
||||||
|
closeOnClick?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
|
||||||
|
(props, ref) => {
|
||||||
|
const [{ opened }, setContext] = useDialogContext();
|
||||||
|
const [variantProps, { children, closeOnClick, onClick, ...otherProps }] =
|
||||||
|
resolveDialogOverlayVariant({ ...props, opened });
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{ReactDOM.createPortal(
|
||||||
|
<div
|
||||||
|
{...otherProps}
|
||||||
|
ref={ref}
|
||||||
|
className={dialogOverlayVariant(variantProps)}
|
||||||
|
onClick={(e) => {
|
||||||
|
if (closeOnClick) {
|
||||||
|
setContext((p) => ({ ...p, opened: false }));
|
||||||
|
}
|
||||||
|
onClick?.(e);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>,
|
||||||
|
document.body
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* DialogContent
|
||||||
|
* =========================
|
||||||
|
*/
|
||||||
|
|
||||||
|
const [dialogContentVariant, resolveDialogContentVariant] = vcn({
|
||||||
|
base: "transition-transform duration-300 bg-white dark:bg-black border border-black/10 dark:border-white/10",
|
||||||
|
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<VariantProps<typeof dialogContentVariant>, "opened"> {}
|
||||||
|
|
||||||
|
const DialogContent = React.forwardRef<HTMLDivElement, DialogContent>(
|
||||||
|
(props, ref) => {
|
||||||
|
const [{ opened }] = useDialogContext();
|
||||||
|
const [variantProps, { children, ...otherProps }] =
|
||||||
|
resolveDialogContentVariant({ ...props, opened });
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
{...otherProps}
|
||||||
|
ref={ref}
|
||||||
|
className={dialogContentVariant(variantProps)}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* DialogClose
|
||||||
|
* =========================
|
||||||
|
*/
|
||||||
|
|
||||||
|
interface DialogCloseProps<T extends React.ReactNode> {
|
||||||
|
children: T extends any[] ? never : T;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DialogClose = <T extends React.ReactNode>({
|
||||||
|
children,
|
||||||
|
}: DialogCloseProps<T>) => {
|
||||||
|
const [_, setState] = useDialogContext();
|
||||||
|
// const onClick = () => setState((p) => ({ ...p, opened: false }));
|
||||||
|
|
||||||
|
const child = React.Children.only(children) as React.ReactElement;
|
||||||
|
|
||||||
|
const onClick: MouseEventHandler<HTMLButtonElement> = (e) => {
|
||||||
|
child.props.onClick?.(e);
|
||||||
|
setState((p) => ({ ...p, opened: false }));
|
||||||
|
};
|
||||||
|
|
||||||
|
return <>{React.cloneElement(child, { onClick })}</>;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* 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<typeof dialogHeaderVariant> {}
|
||||||
|
|
||||||
|
const DialogHeader = React.forwardRef<HTMLElement, DialogHeaderProps>(
|
||||||
|
(props, ref) => {
|
||||||
|
const [variantProps, { children, ...otherProps }] =
|
||||||
|
resolveDialogHeaderVariant(props);
|
||||||
|
return (
|
||||||
|
<header
|
||||||
|
{...otherProps}
|
||||||
|
ref={ref}
|
||||||
|
className={dialogHeaderVariant(variantProps)}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</header>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* 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<typeof dialogTitleVariant> {}
|
||||||
|
|
||||||
|
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<typeof dialogSubtitleVariant> {}
|
||||||
|
|
||||||
|
const DialogTitle = React.forwardRef<HTMLHeadingElement, DialogTitleProps>(
|
||||||
|
(props, ref) => {
|
||||||
|
const [variantProps, { children, ...otherProps }] =
|
||||||
|
resolveDialogTitleVariant(props);
|
||||||
|
return (
|
||||||
|
<h1
|
||||||
|
{...otherProps}
|
||||||
|
ref={ref}
|
||||||
|
className={dialogTitleVariant(variantProps)}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</h1>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const DialogSubtitle = React.forwardRef<
|
||||||
|
HTMLHeadingElement,
|
||||||
|
DialogSubtitleProps
|
||||||
|
>((props, ref) => {
|
||||||
|
const [variantProps, { children, ...otherProps }] =
|
||||||
|
resolveDialogSubtitleVariant(props);
|
||||||
|
return (
|
||||||
|
<h2
|
||||||
|
{...otherProps}
|
||||||
|
ref={ref}
|
||||||
|
className={dialogSubtitleVariant(variantProps)}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</h2>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* DialogFooter
|
||||||
|
* =========================
|
||||||
|
*/
|
||||||
|
|
||||||
|
const [dialogFooterVariant, resolveDialogFooterVariant] = vcn({
|
||||||
|
base: "flex flex-col-reverse sm:flex-row sm:justify-end",
|
||||||
|
variants: {
|
||||||
|
gap: {
|
||||||
|
sm: "gap-2",
|
||||||
|
md: "gap-4",
|
||||||
|
lg: "gap-6",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaults: {
|
||||||
|
gap: "md",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
interface DialogFooterProps
|
||||||
|
extends React.ComponentPropsWithoutRef<"footer">,
|
||||||
|
VariantProps<typeof dialogFooterVariant> {}
|
||||||
|
|
||||||
|
const DialogFooter = React.forwardRef<HTMLDivElement, DialogFooterProps>(
|
||||||
|
(props, ref) => {
|
||||||
|
const [variantProps, { children, ...otherProps }] =
|
||||||
|
resolveDialogFooterVariant(props);
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
{...otherProps}
|
||||||
|
ref={ref}
|
||||||
|
className={dialogFooterVariant(variantProps)}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
export {
|
||||||
|
useDialogContext,
|
||||||
|
DialogRoot,
|
||||||
|
DialogTrigger,
|
||||||
|
DialogOverlay,
|
||||||
|
DialogContent,
|
||||||
|
DialogClose,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
DialogSubtitle,
|
||||||
|
DialogFooter,
|
||||||
|
};
|
101
packages/react/stories/Dialog.stories.tsx
Normal file
101
packages/react/stories/Dialog.stories.tsx
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
import { Button } from "../components/Button";
|
||||||
|
import {
|
||||||
|
DialogClose,
|
||||||
|
DialogContent,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogOverlay,
|
||||||
|
DialogRoot,
|
||||||
|
DialogSubtitle,
|
||||||
|
DialogTitle,
|
||||||
|
DialogTrigger,
|
||||||
|
} from "../components/Dialog";
|
||||||
|
|
||||||
|
export default {
|
||||||
|
title: "React/Dialog",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Default = () => {
|
||||||
|
return (
|
||||||
|
<DialogRoot>
|
||||||
|
<DialogTrigger>
|
||||||
|
<Button
|
||||||
|
onClick={(e) => {
|
||||||
|
console.log(`Opened dialog: `, e.currentTarget);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Open Dialog
|
||||||
|
</Button>
|
||||||
|
</DialogTrigger>
|
||||||
|
<DialogOverlay>
|
||||||
|
<DialogContent size={"fullMd"}>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>Lorem Ipsum</DialogTitle>
|
||||||
|
<DialogSubtitle>This is a test dialog</DialogSubtitle>
|
||||||
|
</DialogHeader>
|
||||||
|
<p>
|
||||||
|
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed do
|
||||||
|
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim
|
||||||
|
ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut
|
||||||
|
aliquip ex ea commodo consequat.
|
||||||
|
</p>
|
||||||
|
<DialogFooter>
|
||||||
|
<DialogClose>
|
||||||
|
<Button
|
||||||
|
onClick={(e) => {
|
||||||
|
console.log(`Closed dialog: `, e.currentTarget);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</Button>
|
||||||
|
</DialogClose>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</DialogOverlay>
|
||||||
|
</DialogRoot>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WithCloseOnClick = () => {
|
||||||
|
return (
|
||||||
|
<DialogRoot>
|
||||||
|
<DialogTrigger>
|
||||||
|
<Button
|
||||||
|
onClick={(e) => {
|
||||||
|
console.log(`Opened dialog: `, e.currentTarget);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Open Dialog
|
||||||
|
</Button>
|
||||||
|
</DialogTrigger>
|
||||||
|
<DialogOverlay
|
||||||
|
closeOnClick
|
||||||
|
onClick={(e) => {
|
||||||
|
console.log(`Closed dialog on overlay click: `, e.currentTarget);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<DialogContent size={"fullMd"}>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>Lorem Ipsum</DialogTitle>
|
||||||
|
<DialogSubtitle>This is a test dialog</DialogSubtitle>
|
||||||
|
</DialogHeader>
|
||||||
|
<p>
|
||||||
|
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed do
|
||||||
|
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim
|
||||||
|
ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut
|
||||||
|
aliquip ex ea commodo consequat.
|
||||||
|
</p>
|
||||||
|
<DialogClose>
|
||||||
|
<Button
|
||||||
|
onClick={(e) => {
|
||||||
|
console.log(`Closed dialog: `, e.currentTarget);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Close Dialog
|
||||||
|
</Button>
|
||||||
|
</DialogClose>
|
||||||
|
</DialogContent>
|
||||||
|
</DialogOverlay>
|
||||||
|
</DialogRoot>
|
||||||
|
);
|
||||||
|
};
|
Loading…
x
Reference in New Issue
Block a user