refactor: replace withSSD & ssdFallback with useDocument

This commit is contained in:
p-sw 2024-08-03 21:35:51 +09:00
parent 71751f7e22
commit 00ebabe8b3
7 changed files with 131 additions and 160 deletions

View File

@ -1,9 +1,4 @@
import { import { Slot, type VariantProps, useDocument, vcn } from "@pswui-lib";
ServerSideDocumentFallback,
Slot,
type VariantProps,
vcn,
} from "@pswui-lib";
import React, { type ReactNode, useId, useState } from "react"; import React, { type ReactNode, useId, useState } from "react";
import ReactDOM from "react-dom"; import ReactDOM from "react-dom";
@ -93,10 +88,10 @@ const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
const { children, closeOnClick, onClick, ...otherPropsExtracted } = const { children, closeOnClick, onClick, ...otherPropsExtracted } =
otherPropsCompressed; otherPropsCompressed;
return ( const document = useDocument();
<ServerSideDocumentFallback> if (!document) return null;
{() =>
ReactDOM.createPortal( return ReactDOM.createPortal(
<div <div
{...otherPropsExtracted} {...otherPropsExtracted}
id={ids.dialog} id={ids.dialog}
@ -119,9 +114,6 @@ const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
</div> </div>
</div>, </div>,
document.body, document.body,
)
}
</ServerSideDocumentFallback>
); );
}, },
); );

View File

@ -1,8 +1,8 @@
import { import {
type AsChild, type AsChild,
ServerSideDocumentFallback,
Slot, Slot,
type VariantProps, type VariantProps,
useDocument,
vcn, vcn,
} from "@pswui-lib"; } from "@pswui-lib";
import React, { import React, {
@ -125,10 +125,10 @@ const DrawerOverlay = forwardRef<HTMLDivElement, DrawerOverlayProps>(
: 1 : 1
})`; })`;
return ( const document = useDocument();
<ServerSideDocumentFallback> if (!document) return null;
{() =>
createPortal( return createPortal(
<Comp <Comp
{...restPropsExtracted} {...restPropsExtracted}
className={drawerOverlayVariant({ className={drawerOverlayVariant({
@ -144,9 +144,6 @@ const DrawerOverlay = forwardRef<HTMLDivElement, DrawerOverlayProps>(
ref={ref} ref={ref}
/>, />,
document.body, document.body,
)
}
</ServerSideDocumentFallback>
); );
}, },
); );

View File

@ -1,4 +1,4 @@
import { type VariantProps, vcn, withServerSideDocument } from "@pswui-lib"; import { type VariantProps, useDocument, vcn } from "@pswui-lib";
import React, { useEffect, useId, useRef } from "react"; import React, { useEffect, useId, useRef } from "react";
import ReactDOM from "react-dom"; import ReactDOM from "react-dom";
@ -145,8 +145,7 @@ interface ToasterProps
muteDuplicationWarning?: boolean; muteDuplicationWarning?: boolean;
} }
const Toaster = withServerSideDocument( const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
const id = useId(); const id = useId();
const [variantProps, otherPropsCompressed] = const [variantProps, otherPropsCompressed] =
resolveToasterVariantProps(props); resolveToasterVariantProps(props);
@ -169,6 +168,10 @@ const Toaster = withServerSideDocument(
}; };
}, [defaultOption]); }, [defaultOption]);
const document = useDocument();
if (!document) return null;
const toasterInstance = document.querySelector("div[data-toaster-root]"); const toasterInstance = document.querySelector("div[data-toaster-root]");
if (toasterInstance && id !== toasterInstance.id) { if (toasterInstance && id !== toasterInstance.id) {
if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) { if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) {
@ -208,8 +211,7 @@ const Toaster = withServerSideDocument(
)} )}
</> </>
); );
}), });
);
Toaster.displayName = "Toaster"; Toaster.displayName = "Toaster";
export { Toaster }; export { Toaster };

View File

@ -1,4 +1,3 @@
export * from "./vcn"; export * from "./vcn";
export * from "./Slot"; export * from "./Slot";
export * from "./ssrFallback"; export * from "./useDocument";
export * from "./withSSD";

View File

@ -1,22 +0,0 @@
import { type ReactNode, useEffect, useState } from "react";
/**
* This component allows components to use `document` as like they're always in the client side.
* Return null if there is no `document` (which represents it's server side) or initial render(to avoid hydration error).
*/
function ServerSideDocumentFallback({
children,
}: { children: () => ReactNode }) {
const [initialRender, setInitialRender] = useState<boolean>(true);
useEffect(() => {
setInitialRender(false);
}, []);
if (typeof document === "undefined" /* server side */ || initialRender)
return null;
return children();
}
export { ServerSideDocumentFallback };

View File

@ -0,0 +1,21 @@
"use client";
import { useEffect, useState } from "react";
/**
* This hook allows components to use `document` as like they're always in the client side.
* Return undefined if there is no `document` (which represents it's server side) or initial render(to avoid hydration error).
*/
function useDocument(): undefined | Document {
const [initialRender, setInitialState] = useState(true);
useEffect(() => {
setInitialState(false);
}, []);
if (typeof document === "undefined" || initialRender) return undefined;
return document;
}
export { useDocument };

View File

@ -1,18 +0,0 @@
import type { ComponentType } from "react";
import { ServerSideDocumentFallback } from "./ssrFallback";
function withServerSideDocument<P extends {}>(
Component: ComponentType<P>,
): ComponentType<P> {
const SSDocumentFallbackWrapper = (props: P) => {
return (
<ServerSideDocumentFallback>
{() => <Component {...props} />}
</ServerSideDocumentFallback>
);
};
return SSDocumentFallbackWrapper;
}
export { withServerSideDocument };