From cb51ee240b5584acc16f6fa2fd9fe6ada689a6fc Mon Sep 17 00:00:00 2001 From: Carl Whittaker Date: Tue, 18 Jun 2024 15:03:55 +0100 Subject: [PATCH] fix: `OakModal` blowing up when doing SSR `document.body` isn't available during SSR, so this would blow up. This commit adds a awful `useEffect` which flips an `isMounted` boolean which we can use to determine if we're rendering on the client while avoiding a server->client markup mismatch warning. *groan* --- .../molecules/OakModal/OakModal.test.tsx | 16 +- .../molecules/OakModal/OakModal.tsx | 16 +- .../__snapshots__/OakModal.test.tsx.snap | 148 ++++++------------ 3 files changed, 74 insertions(+), 106 deletions(-) diff --git a/src/components/molecules/OakModal/OakModal.test.tsx b/src/components/molecules/OakModal/OakModal.test.tsx index 8ab8fd98..c906cfdd 100644 --- a/src/components/molecules/OakModal/OakModal.test.tsx +++ b/src/components/molecules/OakModal/OakModal.test.tsx @@ -20,16 +20,26 @@ jest.mock("react-dom", () => { }); describe(OakModal, () => { - it("matches snapshot", () => { + it("does not render until mounted on the client", () => { const tree = create( {}} footerSlot="Modal footer"> Modal content , - ).toJSON(); + ); + + expect(tree.toJSON()).toBeNull(); + }); + + it("matches snapshot when mounted", async () => { + const result = renderWithTheme( + {}}> + Modal content + , + ); - expect(tree).toMatchSnapshot(); + expect(result.container).toMatchSnapshot(); }); it("calls onClose when the close button is clicked", () => { diff --git a/src/components/molecules/OakModal/OakModal.tsx b/src/components/molecules/OakModal/OakModal.tsx index 41f7722c..771afb26 100644 --- a/src/components/molecules/OakModal/OakModal.tsx +++ b/src/components/molecules/OakModal/OakModal.tsx @@ -1,6 +1,7 @@ import React, { HTMLAttributes, ReactNode, + useEffect, useLayoutEffect, useRef, useState, @@ -79,7 +80,7 @@ const logoSrc = `https://${process.env.NEXT_PUBLIC_OAK_ASSETS_HOST}/${process.en export const OakModal = ({ children, footerSlot, - domContainer = document.body, + domContainer, isOpen, onClose, ...rest @@ -109,6 +110,17 @@ export const OakModal = ({ }; }, [canaryElement]); + // `createPortal` is not supported in SSR so we can only render when mounted on the client + const [isMounted, setIsMounted] = useState(false); + + useEffect(() => { + setIsMounted(true); + }, []); + + if (!isMounted) { + return null; + } + return createPortal( )} , - domContainer, + domContainer ?? document.body, ); }; diff --git a/src/components/molecules/OakModal/__snapshots__/OakModal.test.tsx.snap b/src/components/molecules/OakModal/__snapshots__/OakModal.test.tsx.snap index 13b37454..fcd007d0 100644 --- a/src/components/molecules/OakModal/__snapshots__/OakModal.test.tsx.snap +++ b/src/components/molecules/OakModal/__snapshots__/OakModal.test.tsx.snap @@ -1,23 +1,7 @@ // Jest Snapshot v1, https://goo.gl/fbAQLP -exports[`OakModal matches snapshot 1`] = ` -[ -
, - .c0 { +exports[`OakModal matches snapshot when mounted 1`] = ` +.c0 { position: fixed; inset: 0rem; background: #222222; @@ -193,16 +177,21 @@ exports[`OakModal matches snapshot 1`] = ` letter-spacing: 0.0115rem; } -.c8 { - object-fit: contain; -} - .c20 { -webkit-filter: invert(10%) sepia(1%) saturate(236%) hue-rotate(314deg) brightness(95%) contrast(91%); filter: invert(10%) sepia(1%) saturate(236%) hue-rotate(314deg) brightness(95%) contrast(91%); object-fit: contain; } +.c8 { + background-image: url(); + background-color: #e7f6f5; + background-size: 4rem; + background-position: center; + background-repeat: no-repeat; + object-fit: contain; +} + .c12 { background: none; color: inherit; @@ -281,140 +270,97 @@ exports[`OakModal matches snapshot 1`] = ` transform: translateX(0); } -
+