import React, { useEffect, useId, useRef } from "react"; import ReactDOM from "react-dom"; import { VariantProps, vcn } from "@pswui-lib"; import { toastVariant } from "./Variant"; import { ToastOption, toasts, subscribeSingle, getSingleSnapshot, notifySingle, close, notify, defaultToastOption, subscribe, getSnapshot, } from "./Store"; 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)()); }); }, [id]); const toastData = { ...globalOption, ...toast, }; React.useEffect(() => { if (toastData.life === "born") { requestAnimationFrame(() => { // To make sure that the toast is rendered as "born" state // and then change to "normal" state toasts[id] = { ...toasts[id], life: "normal", }; notifySingle(id); }); } if (toastData.life === "normal" && toastData.closeTimeout !== null) { const timeout = setTimeout(() => { close(id); }, toastData.closeTimeout); return () => clearTimeout(timeout); } if (toastData.life === "dead") { let transitionDuration: { value: number; unit: string; } | null; if (!ref.current) { transitionDuration = null; } else if (ref.current.computedStyleMap !== undefined) { transitionDuration = ref.current .computedStyleMap() .get("transition-duration") as { value: number; unit: string }; } else { const style = /(\d+(\.\d+)?)(.+)/.exec( window.getComputedStyle(ref.current).transitionDuration, ); transitionDuration = style ? { value: parseFloat(style[1] ?? "0"), unit: style[3] ?? style[2] ?? "s", } : null; } if (!transitionDuration) { delete toasts[id]; notify(); return; } const calculatedTransitionDuration = transitionDuration.value * ({ s: 1000, ms: 1, }[transitionDuration.unit] ?? 1); const timeout = setTimeout(() => { delete toasts[id]; notify(); }, calculatedTransitionDuration); return () => clearTimeout(timeout); } }, [id, toastData.life, toastData.closeTimeout, toastData.closeButton]); return (
{toastData.closeButton && ( )}
{toastData.title}
{toastData.description}
); }; const [toasterVariant, resolveToasterVariantProps] = vcn({ base: "fixed p-4 flex flex-col gap-4 top-0 right-0 w-full md:max-w-md md:bottom-0 md:top-auto pointer-events-none z-40", variants: {}, defaults: {}, }); interface ToasterProps extends React.ComponentPropsWithoutRef<"div">, VariantProps { defaultOption?: Partial; muteDuplicationWarning?: boolean; } const Toaster = React.forwardRef((props, ref) => { const id = useId(); const [variantProps, otherPropsCompressed] = resolveToasterVariantProps(props); const { defaultOption, muteDuplicationWarning, ...otherPropsExtracted } = otherPropsCompressed; const [toastList, setToastList] = React.useState(toasts); const internalRef = useRef(null); useEffect(() => { return subscribe(() => { setToastList(getSnapshot()); }); }, []); const option = React.useMemo(() => { return { ...defaultToastOption, ...defaultOption, }; }, [defaultOption]); const toasterInstance = document.querySelector("div[data-toaster-root]"); if (toasterInstance && id !== toasterInstance.id) { if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) { console.warn( `Multiple Toaster instances detected. Only one Toaster is allowed.`, ); } return null; } return ( <> {ReactDOM.createPortal(
{ internalRef.current = el; if (typeof ref === "function") { ref(el); } else if (ref) { ref.current = el; } }} id={id} > {Object.entries(toastList).map(([id]) => ( ))}
, document.body, )} ); }); export { Toaster };