diff --git a/packages/react/components/Toast.tsx b/packages/react/components/Toast.tsx index ce071e6..79edcc2 100644 --- a/packages/react/components/Toast.tsx +++ b/packages/react/components/Toast.tsx @@ -1,4 +1,4 @@ -import React, { useEffect } from "react"; +import React, { useEffect, useRef } from "react"; import ReactDOM from "react-dom"; import { VariantProps, vcn } from "../shared"; @@ -266,14 +266,17 @@ interface ToasterProps extends React.ComponentPropsWithoutRef<"div">, VariantProps { defaultOption?: Partial; + muteDuplicationWarning?: boolean; } const Toaster = React.forwardRef((props, ref) => { const [variantProps, otherPropsCompressed] = resolveToasterVariantProps(props); - const { defaultOption, ...otherPropsExtracted } = otherPropsCompressed; + const { defaultOption, muteDuplicationWarning, ...otherPropsExtracted } = + otherPropsCompressed; const [toastList, setToastList] = React.useState(toasts); + const internalRef = useRef(null); useEffect(() => { const unsubscribe = subscribe(() => { @@ -282,6 +285,20 @@ const Toaster = React.forwardRef((props, ref) => { return unsubscribe; }, []); + const toasterInstance = document.querySelector("#toaster"); + if ( + toasterInstance && + (internalRef.current === null || + !internalRef.current.isEqualNode(toasterInstance)) + ) { + if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) { + console.warn( + `Multiple Toaster instances detected. Only one Toaster is allowed.` + ); + } + return null; + } + const option = React.useMemo(() => { return { ...defaultToastOption, @@ -295,7 +312,15 @@ const Toaster = React.forwardRef((props, ref) => {
{ + internalRef.current = el; + if (typeof ref === "function") { + ref(el); + } else if (ref) { + ref.current = el; + } + }} + id="toaster" > {Object.entries(toastList).map(([id]) => (