feat: add toaster duplication detection process
This commit is contained in:
parent
e3086340c1
commit
86879b4456
@ -1,4 +1,4 @@
|
|||||||
import React, { useEffect } from "react";
|
import React, { useEffect, useRef } from "react";
|
||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
import { VariantProps, vcn } from "../shared";
|
import { VariantProps, vcn } from "../shared";
|
||||||
|
|
||||||
@ -266,14 +266,17 @@ interface ToasterProps
|
|||||||
extends React.ComponentPropsWithoutRef<"div">,
|
extends React.ComponentPropsWithoutRef<"div">,
|
||||||
VariantProps<typeof toasterVariant> {
|
VariantProps<typeof toasterVariant> {
|
||||||
defaultOption?: Partial<ToastOption>;
|
defaultOption?: Partial<ToastOption>;
|
||||||
|
muteDuplicationWarning?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
||||||
const [variantProps, otherPropsCompressed] =
|
const [variantProps, otherPropsCompressed] =
|
||||||
resolveToasterVariantProps(props);
|
resolveToasterVariantProps(props);
|
||||||
const { defaultOption, ...otherPropsExtracted } = otherPropsCompressed;
|
const { defaultOption, muteDuplicationWarning, ...otherPropsExtracted } =
|
||||||
|
otherPropsCompressed;
|
||||||
|
|
||||||
const [toastList, setToastList] = React.useState<typeof toasts>(toasts);
|
const [toastList, setToastList] = React.useState<typeof toasts>(toasts);
|
||||||
|
const internalRef = useRef<HTMLDivElement | null>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const unsubscribe = subscribe(() => {
|
const unsubscribe = subscribe(() => {
|
||||||
@ -282,6 +285,20 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
return unsubscribe;
|
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(() => {
|
const option = React.useMemo(() => {
|
||||||
return {
|
return {
|
||||||
...defaultToastOption,
|
...defaultToastOption,
|
||||||
@ -295,7 +312,15 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
<div
|
<div
|
||||||
{...otherPropsExtracted}
|
{...otherPropsExtracted}
|
||||||
className={toasterVariant(variantProps)}
|
className={toasterVariant(variantProps)}
|
||||||
ref={ref}
|
ref={(el) => {
|
||||||
|
internalRef.current = el;
|
||||||
|
if (typeof ref === "function") {
|
||||||
|
ref(el);
|
||||||
|
} else if (ref) {
|
||||||
|
ref.current = el;
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
id="toaster"
|
||||||
>
|
>
|
||||||
{Object.entries(toastList).map(([id]) => (
|
{Object.entries(toastList).map(([id]) => (
|
||||||
<ToastTemplate
|
<ToastTemplate
|
||||||
|
Loading…
x
Reference in New Issue
Block a user