fix: use id strategy to check duplication

This commit is contained in:
p-sw 2024-06-03 20:50:04 +09:00
parent 86879b4456
commit 38782bda7e

View File

@ -1,4 +1,4 @@
import React, { useEffect, useRef } from "react"; import React, { useEffect, useId, useRef } from "react";
import ReactDOM from "react-dom"; import ReactDOM from "react-dom";
import { VariantProps, vcn } from "../shared"; import { VariantProps, vcn } from "../shared";
@ -270,6 +270,7 @@ interface ToasterProps
} }
const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => { const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
const id = useId();
const [variantProps, otherPropsCompressed] = const [variantProps, otherPropsCompressed] =
resolveToasterVariantProps(props); resolveToasterVariantProps(props);
const { defaultOption, muteDuplicationWarning, ...otherPropsExtracted } = const { defaultOption, muteDuplicationWarning, ...otherPropsExtracted } =
@ -285,12 +286,15 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
return unsubscribe; return unsubscribe;
}, []); }, []);
const toasterInstance = document.querySelector("#toaster"); const option = React.useMemo(() => {
if ( return {
toasterInstance && ...defaultToastOption,
(internalRef.current === null || ...defaultOption,
!internalRef.current.isEqualNode(toasterInstance)) };
) { }, [defaultOption]);
const toasterInstance = document.querySelector("div[data-toaster-root]");
if (toasterInstance && id !== toasterInstance.id) {
if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) { if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) {
console.warn( console.warn(
`Multiple Toaster instances detected. Only one Toaster is allowed.` `Multiple Toaster instances detected. Only one Toaster is allowed.`
@ -299,18 +303,12 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
return null; return null;
} }
const option = React.useMemo(() => {
return {
...defaultToastOption,
...defaultOption,
};
}, [defaultOption]);
return ( return (
<> <>
{ReactDOM.createPortal( {ReactDOM.createPortal(
<div <div
{...otherPropsExtracted} {...otherPropsExtracted}
data-toaster-root
className={toasterVariant(variantProps)} className={toasterVariant(variantProps)}
ref={(el) => { ref={(el) => {
internalRef.current = el; internalRef.current = el;
@ -320,7 +318,7 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
ref.current = el; ref.current = el;
} }
}} }}
id="toaster" id={id}
> >
{Object.entries(toastList).map(([id]) => ( {Object.entries(toastList).map(([id]) => (
<ToastTemplate <ToastTemplate