diff --git a/packages/react/components/Toast.tsx b/packages/react/components/Toast.tsx new file mode 100644 index 0000000..8db53f6 --- /dev/null +++ b/packages/react/components/Toast.tsx @@ -0,0 +1,272 @@ +import React, { useEffect } from "react"; +import ReactDOM from "react-dom"; +import { VariantProps, vcn } from "../shared"; + +interface ToastOption { + closeButton: boolean; + closeTimeout: number | null; +} + +const defaultToastOption: ToastOption = { + closeButton: true, + closeTimeout: 3000, +}; + +const [toastVariant] = vcn({ + base: "flex flex-col gap-2 border p-4 rounded-lg pr-8 pointer-events-auto bg-white dark:bg-black relative transition-all duration-150", + variants: { + status: { + default: "border-black/10 dark:border-white/20", + error: "border-red-500/50", + success: "border-green-500/50", + warning: "border-yellow-500/50", + loading: "border-black/50 dark:border-white/50 animate-pulse", + }, + life: { + born: "translate-y-1/2 scale-50", + normal: "translate-y-0 scale-100 ease-out", + dead: "translate-y-1/2 scale-50 ease-in", + }, + }, + defaults: { + status: "default", + life: "born", + }, +}); + +interface ToastBody extends Omit, "preset"> { + title: string; + description: string; +} + +let index = 0; +let toasts: Record< + `${number}`, + ToastBody & + Partial & { subscribers: (() => void)[]; version: number } +> = {}; +let subscribers: (() => void)[] = []; + +/** + * ==== + * Controls + * ==== + */ + +function subscribe(callback: () => void) { + subscribers.push(callback); + return () => { + subscribers = subscribers.filter((subscriber) => subscriber !== callback); + }; +} + +function getSnapshot() { + return { ...toasts }; +} + +function subscribeSingle(id: `${number}`) { + return (callback: () => void) => { + toasts[id].subscribers.push(callback); + return () => { + toasts[id].subscribers = toasts[id].subscribers.filter( + (subscriber) => subscriber !== callback + ); + }; + }; +} + +function getSingleSnapshot(id: `${number}`) { + return () => { + return { + ...toasts[id], + }; + }; +} + +function notify() { + subscribers.forEach((subscriber) => subscriber()); +} + +function notifySingle(id: `${number}`) { + toasts[id].subscribers.forEach((subscriber) => subscriber()); +} + +function close(id: `${number}`) { + toasts[id] = { + ...toasts[id], + life: "dead", + version: toasts[id].version + 1, + }; + notifySingle(id); +} + +function update( + id: `${number}`, + toast: Partial & Partial> +) { + toasts[id] = { + ...toasts[id], + ...toast, + version: toasts[id].version + 1, + }; + notifySingle(id); +} + +function addToast(toast: Omit & Partial) { + const id: `${number}` = `${index}`; + toasts[id] = { + ...toast, + subscribers: [], + life: "born", + version: 0, + }; + index += 1; + notify(); + + return { + update: (toast: Partial & Partial>) => + update(id, toast), + close: () => close(id), + }; +} + +function useToast() { + return { + toast: addToast, + update, + close, + }; +} + +const ToastTemplate = ({ + id, + globalOption, +}: { + id: `${number}`; + globalOption: ToastOption; +}) => { + const [toast, setToast] = React.useState<(typeof toasts)[`${number}`]>( + toasts[id] + ); + const ref = React.useRef(null); + + useEffect(() => { + subscribeSingle(id)(() => { + setToast(getSingleSnapshot(id)()); + }); + }, []); + + const toastData = { + ...globalOption, + ...toast, + }; + + React.useEffect(() => { + if (toastData.life === "born") { + toasts[id] = { + ...toasts[id], + life: "normal", + version: toasts[id].version + 1, + }; + notifySingle(id); + } + if (toastData.life === "normal" && toastData.closeTimeout !== null) { + const timeout = setTimeout(() => { + close(id); + }, toastData.closeTimeout); + return () => clearTimeout(timeout); + } + if (toastData.life === "dead") { + const transitionDuration = ref.current + ?.computedStyleMap() + ?.get("transition-duration") as { + value: number; + unit: string; + }; + if (!transitionDuration) { + delete toasts[id]; + notify(); + return; + } + const calculatedTransitionDuration = + transitionDuration.value * + ({ + s: 1000, + ms: 1, + }[transitionDuration.unit] ?? 1); + setTimeout(() => { + delete toasts[id]; + notify(); + }, calculatedTransitionDuration); + } + }, [toastData.version]); + + return ( +
+ {toastData.closeButton && ( + + )} +
{toastData.title}
+
{toastData.description}
+
+ ); +}; + +const Toaster = ({ + defaultOption, +}: { + defaultOption?: Partial; +}) => { + const [toastList, setToastList] = React.useState(toasts); + + useEffect(() => { + const unsubscribe = subscribe(() => { + setToastList(getSnapshot()); + }); + return unsubscribe; + }, []); + + const option = React.useMemo(() => { + return { + ...defaultToastOption, + ...defaultOption, + }; + }, [defaultOption]); + + return ( + <> + {ReactDOM.createPortal( +
+ {Object.entries(toastList).map(([id]) => ( + + ))} +
, + document.body + )} + + ); +}; + +export { Toaster, useToast }; diff --git a/packages/react/stories/Toast.stories.tsx b/packages/react/stories/Toast.stories.tsx new file mode 100644 index 0000000..601692f --- /dev/null +++ b/packages/react/stories/Toast.stories.tsx @@ -0,0 +1,113 @@ +import { Button } from "../components/Button"; +import { Toaster, useToast } from "../components/Toast"; + +export default { + title: "React/Toast", + tags: ["!autodocs"], + decorators: [ + (Story: any) => ( + <> + + {Story()} + + ), + ], +}; + +export const Default = () => { + const { toast } = useToast(); + + return ( + <> + + + ); +}; + +const fetchAndWaitForSuccess = (): Promise => { + return new Promise((resolve) => + setTimeout(() => resolve("LoremSuccess"), 3000) + ); +}; +const fetchAndWaitForError = (): Promise => { + return new Promise((_, reject) => + setTimeout(() => reject("LoremError"), 3000) + ); +}; + +export const PromiseWaitSuccess = () => { + const { toast } = useToast(); + + return ( + <> + + + ); +}; + +export const PromiseWaitError = () => { + const { toast } = useToast(); + + return ( + <> + + + ); +};