diff --git a/.changeset/serious-stingrays-learn.md b/.changeset/serious-stingrays-learn.md new file mode 100644 index 0000000000..ff11afd3bf --- /dev/null +++ b/.changeset/serious-stingrays-learn.md @@ -0,0 +1,10 @@ +--- +'@clerk/clerk-react': minor +--- + +Adds support for a `fallback` prop on Clerk's components. This allows rendering of a placeholder element while Clerk's components are mounting. Use this to help mitigate mitigate layout shift when using Clerk's components. Example usage: + + +```tsx +} /> +``` diff --git a/integration/templates/next-app-router/src/app/sign-in/[[...catchall]]/page.tsx b/integration/templates/next-app-router/src/app/sign-in/[[...catchall]]/page.tsx index 3b4342ff8a..d193e28a46 100644 --- a/integration/templates/next-app-router/src/app/sign-in/[[...catchall]]/page.tsx +++ b/integration/templates/next-app-router/src/app/sign-in/[[...catchall]]/page.tsx @@ -7,6 +7,7 @@ export default function Page() { routing={'path'} path={'/sign-in'} signUpUrl={'/sign-up'} + fallback={<>Loading sign in} /> ); diff --git a/integration/templates/react-vite/src/sign-in/index.tsx b/integration/templates/react-vite/src/sign-in/index.tsx index 39caef7d00..7ec2593036 100644 --- a/integration/templates/react-vite/src/sign-in/index.tsx +++ b/integration/templates/react-vite/src/sign-in/index.tsx @@ -6,6 +6,7 @@ export default function Page() { Loading sign in} /> ); diff --git a/integration/tests/sign-in-flow.test.ts b/integration/tests/sign-in-flow.test.ts index e04b5e0012..290758ba62 100644 --- a/integration/tests/sign-in-flow.test.ts +++ b/integration/tests/sign-in-flow.test.ts @@ -23,6 +23,12 @@ testAgainstRunningApps({ withEnv: [appConfigs.envs.withEmailCodes] })('sign in f await app.teardown(); }); + test('sign in supports fallback', async ({ page, context }) => { + const u = createTestUtils({ app, page, context }); + await u.page.goToRelative('/sign-in'); + await expect(u.page.getByText('Loading sign in')).toBeVisible(); + }); + test('sign in with email and password', async ({ page, context }) => { const u = createTestUtils({ app, page, context }); await u.po.signIn.goTo(); diff --git a/packages/nextjs/src/client-boundary/uiComponents.tsx b/packages/nextjs/src/client-boundary/uiComponents.tsx index a4dde9da72..054e54a428 100644 --- a/packages/nextjs/src/client-boundary/uiComponents.tsx +++ b/packages/nextjs/src/client-boundary/uiComponents.tsx @@ -6,7 +6,7 @@ import { SignUp as BaseSignUp, UserProfile as BaseUserProfile, } from '@clerk/clerk-react'; -import type { OrganizationProfileProps, SignInProps, SignUpProps, UserProfileProps } from '@clerk/types'; +import type { ComponentProps } from 'react'; import React from 'react'; import { useEnforceCorrectRoutingProps } from './hooks/useEnforceRoutingProps'; @@ -29,7 +29,7 @@ export { // Also the `typeof BaseUserProfile` is used to resolve the following error: // "The inferred type of 'UserProfile' cannot be named without a reference to ..." export const UserProfile: typeof BaseUserProfile = Object.assign( - (props: UserProfileProps) => { + (props: ComponentProps) => { return ; }, { ...BaseUserProfile }, @@ -40,16 +40,16 @@ export const UserProfile: typeof BaseUserProfile = Object.assign( // Also the `typeof BaseOrganizationProfile` is used to resolved the following error: // "The inferred type of 'OrganizationProfile' cannot be named without a reference to ..." export const OrganizationProfile: typeof BaseOrganizationProfile = Object.assign( - (props: OrganizationProfileProps) => { + (props: ComponentProps) => { return ; }, { ...BaseOrganizationProfile }, ); -export const SignIn = (props: SignInProps) => { +export const SignIn = (props: ComponentProps) => { return ; }; -export const SignUp = (props: SignUpProps) => { +export const SignUp = (props: ComponentProps) => { return ; }; diff --git a/packages/react/src/components/ClerkHostRenderer.tsx b/packages/react/src/components/ClerkHostRenderer.tsx new file mode 100644 index 0000000000..19b8187718 --- /dev/null +++ b/packages/react/src/components/ClerkHostRenderer.tsx @@ -0,0 +1,118 @@ +import { without } from '@clerk/shared/object'; +import { isDeeplyEqual } from '@clerk/shared/react'; +import type { PropsWithChildren } from 'react'; +import React from 'react'; + +import type { MountProps, OpenProps } from '../types'; + +const isMountProps = (props: any): props is MountProps => { + return 'mount' in props; +}; + +const isOpenProps = (props: any): props is OpenProps => { + return 'open' in props; +}; +// README: should be a class pure component in order for mount and unmount +// lifecycle props to be invoked correctly. Replacing the class component with a +// functional component wrapped with a React.memo is not identical to the original +// class implementation due to React intricacies such as the useEffect’s cleanup +// seems to run AFTER unmount, while componentWillUnmount runs BEFORE. + +// More information can be found at https://clerk.slack.com/archives/C015S0BGH8R/p1624891993016300 + +// The function Portal implementation is commented out for future reference. + +// const Portal = React.memo(({ props, mount, unmount }: MountProps) => { +// const portalRef = React.createRef(); + +// useEffect(() => { +// if (portalRef.current) { +// mount(portalRef.current, props); +// } +// return () => { +// if (portalRef.current) { +// unmount(portalRef.current); +// } +// }; +// }, []); + +// return
; +// }); + +// Portal.displayName = 'ClerkPortal'; + +/** + * Used to orchestrate mounting of Clerk components in a host React application. + * Components are rendered into a specific DOM node using mount/unmount methods provided by the Clerk class. + */ +export class ClerkHostRenderer extends React.PureComponent< + PropsWithChildren< + (MountProps | OpenProps) & { + component?: string; + hideRootHtmlElement?: boolean; + rootProps?: JSX.IntrinsicElements['div']; + } + > +> { + private rootRef = React.createRef(); + + componentDidUpdate(_prevProps: Readonly) { + if (!isMountProps(_prevProps) || !isMountProps(this.props)) { + return; + } + + // Remove children and customPages from props before comparing + // children might hold circular references which deepEqual can't handle + // and the implementation of customPages or customMenuItems relies on props getting new references + const prevProps = without(_prevProps.props, 'customPages', 'customMenuItems', 'children'); + const newProps = without(this.props.props, 'customPages', 'customMenuItems', 'children'); + // instead, we simply use the length of customPages to determine if it changed or not + const customPagesChanged = prevProps.customPages?.length !== newProps.customPages?.length; + const customMenuItemsChanged = prevProps.customMenuItems?.length !== newProps.customMenuItems?.length; + + if (!isDeeplyEqual(prevProps, newProps) || customPagesChanged || customMenuItemsChanged) { + if (this.rootRef.current) { + this.props.updateProps({ node: this.rootRef.current, props: this.props.props }); + } + } + } + + componentDidMount() { + if (this.rootRef.current) { + if (isMountProps(this.props)) { + this.props.mount(this.rootRef.current, this.props.props); + } + + if (isOpenProps(this.props)) { + this.props.open(this.props.props); + } + } + } + + componentWillUnmount() { + if (this.rootRef.current) { + if (isMountProps(this.props)) { + this.props.unmount(this.rootRef.current); + } + if (isOpenProps(this.props)) { + this.props.close(); + } + } + } + + render() { + const { hideRootHtmlElement = false } = this.props; + const rootAttributes = { + ref: this.rootRef, + ...this.props.rootProps, + ...(this.props.component && { 'data-clerk-component': this.props.component }), + }; + + return ( + <> + {!hideRootHtmlElement &&
} + {this.props.children} + + ); + } +} diff --git a/packages/react/src/components/uiComponents.tsx b/packages/react/src/components/uiComponents.tsx index a0d2a3263d..094f060707 100644 --- a/packages/react/src/components/uiComponents.tsx +++ b/packages/react/src/components/uiComponents.tsx @@ -1,5 +1,3 @@ -import { without } from '@clerk/shared/object'; -import { isDeeplyEqual } from '@clerk/shared/react'; import { logErrorInDevMode } from '@clerk/shared/utils'; import type { CreateOrganizationProps, @@ -14,7 +12,7 @@ import type { WaitlistProps, Without, } from '@clerk/types'; -import type { PropsWithChildren } from 'react'; +import type { PropsWithChildren, ReactElement } from 'react'; import React, { createContext, createElement, useContext } from 'react'; import { @@ -29,7 +27,6 @@ import { import type { CustomPortalsRendererProps, MountProps, - OpenProps, OrganizationProfileLinkProps, OrganizationProfilePageProps, UserButtonActionProps, @@ -44,8 +41,17 @@ import { useUserButtonCustomMenuItems, useUserProfileCustomPages, } from '../utils'; +import { useWaitForComponentMount } from '../utils/useWaitForComponentMount'; +import { ClerkHostRenderer } from './ClerkHostRenderer'; import { withClerk } from './withClerk'; +type FallbackProp = { + /** + * An optional element to render while the component is mounting. + */ + fallback?: ReactElement; +}; + type UserProfileExportType = typeof _UserProfile & { Page: typeof UserProfilePage; Link: typeof UserProfileLink; @@ -59,8 +65,7 @@ type UserButtonExportType = typeof _UserButton & { Link: typeof MenuLink; /** * The `` component can be used in conjunction with `asProvider` in order to control rendering - * of the `` without affecting its configuration or any custom pages - * that could be mounted + * of the `` without affecting its configuration or any custom pages that could be mounted * @experimental This API is experimental and may change at any moment. */ __experimental_Outlet: typeof UserButtonOutlet; @@ -89,8 +94,7 @@ type OrganizationSwitcherExportType = typeof _OrganizationSwitcher & { OrganizationProfileLink: typeof OrganizationProfileLink; /** * The `` component can be used in conjunction with `asProvider` in order to control rendering - * of the `` without affecting its configuration or any custom pages - * that could be mounted + * of the `` without affecting its configuration or any custom pages that could be mounted * @experimental This API is experimental and may change at any moment. */ __experimental_Outlet: typeof OrganizationSwitcherOutlet; @@ -109,103 +113,6 @@ type OrganizationSwitcherPropsWithoutCustomPages = Without< __experimental_asProvider?: boolean; }; -const isMountProps = (props: any): props is MountProps => { - return 'mount' in props; -}; - -const isOpenProps = (props: any): props is OpenProps => { - return 'open' in props; -}; - -// README: should be a class pure component in order for mount and unmount -// lifecycle props to be invoked correctly. Replacing the class component with a -// functional component wrapped with a React.memo is not identical to the original -// class implementation due to React intricacies such as the useEffect’s cleanup -// seems to run AFTER unmount, while componentWillUnmount runs BEFORE. - -// More information can be found at https://clerk.slack.com/archives/C015S0BGH8R/p1624891993016300 - -// The function Portal implementation is commented out for future reference. - -// const Portal = React.memo(({ props, mount, unmount }: MountProps) => { -// const portalRef = React.createRef(); - -// useEffect(() => { -// if (portalRef.current) { -// mount(portalRef.current, props); -// } -// return () => { -// if (portalRef.current) { -// unmount(portalRef.current); -// } -// }; -// }, []); - -// return
; -// }); - -// Portal.displayName = 'ClerkPortal'; - -class Portal extends React.PureComponent< - PropsWithChildren<(MountProps | OpenProps) & { hideRootHtmlElement?: boolean }> -> { - private portalRef = React.createRef(); - - componentDidUpdate(_prevProps: Readonly) { - if (!isMountProps(_prevProps) || !isMountProps(this.props)) { - return; - } - - // Remove children and customPages from props before comparing - // children might hold circular references which deepEqual can't handle - // and the implementation of customPages or customMenuItems relies on props getting new references - const prevProps = without(_prevProps.props, 'customPages', 'customMenuItems', 'children'); - const newProps = without(this.props.props, 'customPages', 'customMenuItems', 'children'); - // instead, we simply use the length of customPages to determine if it changed or not - const customPagesChanged = prevProps.customPages?.length !== newProps.customPages?.length; - const customMenuItemsChanged = prevProps.customMenuItems?.length !== newProps.customMenuItems?.length; - - if (!isDeeplyEqual(prevProps, newProps) || customPagesChanged || customMenuItemsChanged) { - if (this.portalRef.current) { - this.props.updateProps({ node: this.portalRef.current, props: this.props.props }); - } - } - } - - componentDidMount() { - if (this.portalRef.current) { - if (isMountProps(this.props)) { - this.props.mount(this.portalRef.current, this.props.props); - } - - if (isOpenProps(this.props)) { - this.props.open(this.props.props); - } - } - } - - componentWillUnmount() { - if (this.portalRef.current) { - if (isMountProps(this.props)) { - this.props.unmount(this.portalRef.current); - } - if (isOpenProps(this.props)) { - this.props.close(); - } - } - } - - render() { - const { hideRootHtmlElement = false } = this.props; - return ( - <> - {!hideRootHtmlElement &&
} - {this.props.children} - - ); - } -} - const CustomPortalsRenderer = (props: CustomPortalsRendererProps) => { return ( <> @@ -215,27 +122,61 @@ const CustomPortalsRenderer = (props: CustomPortalsRendererProps) => { ); }; -export const SignIn = withClerk(({ clerk, ...props }: WithClerkProp) => { - return ( - - ); -}, 'SignIn'); +export const SignIn = withClerk( + ({ clerk, component, fallback, ...props }: WithClerkProp) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; -export const SignUp = withClerk(({ clerk, ...props }: WithClerkProp) => { - return ( - - ); -}, 'SignUp'); + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + + return ( + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + )} + + ); + }, + { component: 'SignIn', renderWhileLoading: true }, +); + +export const SignUp = withClerk( + ({ clerk, component, fallback, ...props }: WithClerkProp) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + + return ( + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + )} + + ); + }, + { component: 'SignUp', renderWhileLoading: true }, +); export function UserProfilePage({ children }: PropsWithChildren) { logErrorInDevMode(userProfilePageRenderedError); @@ -248,20 +189,37 @@ export function UserProfileLink({ children }: PropsWithChildren>>) => { + ({ + clerk, + component, + fallback, + ...props + }: WithClerkProp> & FallbackProp>) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + const { customPages, customPagesPortals } = useUserProfileCustomPages(props.children); return ( - - - + <> + {shouldShowFallback && fallback} + + + + ); }, - 'UserProfile', + { component: 'UserProfile', renderWhileLoading: true }, ); export const UserProfile: UserProfileExportType = Object.assign(_UserProfile, { @@ -276,7 +234,19 @@ const UserButtonContext = createContext({ }); const _UserButton = withClerk( - ({ clerk, ...props }: WithClerkProp>) => { + ({ + clerk, + component, + fallback, + ...props + }: WithClerkProp & FallbackProp>) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + const { customPages, customPagesPortals } = useUserProfileCustomPages(props.children, { allowForAnyChildren: !!props.__experimental_asProvider, }); @@ -297,18 +267,23 @@ const _UserButton = withClerk( return ( - - {/*This mimics the previous behaviour before asProvider existed*/} - {props.__experimental_asProvider ? sanitizedChildren : null} - - + {shouldShowFallback && fallback} + {clerk.loaded && ( + + {/*This mimics the previous behaviour before asProvider existed*/} + {props.__experimental_asProvider ? sanitizedChildren : null} + + + )} ); }, - 'UserButton', + { component: 'UserButton', renderWhileLoading: true }, ); export function MenuItems({ children }: PropsWithChildren) { @@ -337,7 +312,7 @@ export function UserButtonOutlet(outletProps: Without; + return ; } export const UserButton: UserButtonExportType = Object.assign(_UserButton, { @@ -360,20 +335,39 @@ export function OrganizationProfileLink({ children }: PropsWithChildren>>) => { + ({ + clerk, + component, + fallback, + ...props + }: WithClerkProp> & FallbackProp>) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + const { customPages, customPagesPortals } = useOrganizationProfileCustomPages(props.children); return ( - - - + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + + + )} + ); }, - 'OrganizationProfile', + { component: 'OrganizationProfile', renderWhileLoading: true }, ); export const OrganizationProfile: OrganizationProfileExportType = Object.assign(_OrganizationProfile, { @@ -381,16 +375,33 @@ export const OrganizationProfile: OrganizationProfileExportType = Object.assign( Link: OrganizationProfileLink, }); -export const CreateOrganization = withClerk(({ clerk, ...props }: WithClerkProp) => { - return ( - - ); -}, 'CreateOrganization'); +export const CreateOrganization = withClerk( + ({ clerk, component, fallback, ...props }: WithClerkProp) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + + return ( + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + )} + + ); + }, + { component: 'CreateOrganization', renderWhileLoading: true }, +); const OrganizationSwitcherContext = createContext({ mount: () => {}, @@ -399,7 +410,19 @@ const OrganizationSwitcherContext = createContext({ }); const _OrganizationSwitcher = withClerk( - ({ clerk, ...props }: WithClerkProp>) => { + ({ + clerk, + component, + fallback, + ...props + }: WithClerkProp & FallbackProp>) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + const { customPages, customPagesPortals } = useOrganizationProfileCustomPages(props.children, { allowForAnyChildren: !!props.__experimental_asProvider, }); @@ -411,6 +434,8 @@ const _OrganizationSwitcher = withClerk( unmount: clerk.unmountOrganizationSwitcher, updateProps: (clerk as any).__unstable__updateProps, props: { ...props, organizationProfileProps }, + rootProps: rendererRootProps, + component, }; /** @@ -420,18 +445,23 @@ const _OrganizationSwitcher = withClerk( return ( - - {/*This mimics the previous behaviour before asProvider existed*/} - {props.__experimental_asProvider ? sanitizedChildren : null} - - + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + {/*This mimics the previous behaviour before asProvider existed*/} + {props.__experimental_asProvider ? sanitizedChildren : null} + + + )} + ); }, - 'OrganizationSwitcher', + { component: 'OrganizationSwitcher', renderWhileLoading: true }, ); export function OrganizationSwitcherOutlet( @@ -447,7 +477,7 @@ export function OrganizationSwitcherOutlet( }, } satisfies MountProps; - return ; + return ; } export const OrganizationSwitcher: OrganizationSwitcherExportType = Object.assign(_OrganizationSwitcher, { @@ -456,34 +486,86 @@ export const OrganizationSwitcher: OrganizationSwitcherExportType = Object.assig __experimental_Outlet: OrganizationSwitcherOutlet, }); -export const OrganizationList = withClerk(({ clerk, ...props }: WithClerkProp) => { - return ( - - ); -}, 'OrganizationList'); +export const OrganizationList = withClerk( + ({ clerk, component, fallback, ...props }: WithClerkProp) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; -export const GoogleOneTap = withClerk(({ clerk, ...props }: WithClerkProp) => { - return ( - - ); -}, 'GoogleOneTap'); + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; -export const Waitlist = withClerk(({ clerk, ...props }: WithClerkProp) => { - return ( - - ); -}, 'Waitlist'); + return ( + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + )} + + ); + }, + { component: 'OrganizationList', renderWhileLoading: true }, +); + +export const GoogleOneTap = withClerk( + ({ clerk, component, fallback, ...props }: WithClerkProp) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + + return ( + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + )} + + ); + }, + { component: 'GoogleOneTap', renderWhileLoading: true }, +); + +export const Waitlist = withClerk( + ({ clerk, component, fallback, ...props }: WithClerkProp) => { + const mountingStatus = useWaitForComponentMount(component); + const shouldShowFallback = mountingStatus === 'rendering' || !clerk.loaded; + + const rendererRootProps = { + ...(shouldShowFallback && fallback && { style: { display: 'none' } }), + }; + + return ( + <> + {shouldShowFallback && fallback} + {clerk.loaded && ( + + )} + + ); + }, + { component: 'Waitlist', renderWhileLoading: true }, +); diff --git a/packages/react/src/components/withClerk.tsx b/packages/react/src/components/withClerk.tsx index 54ff7be47c..a78a1c82c8 100644 --- a/packages/react/src/components/withClerk.tsx +++ b/packages/react/src/components/withClerk.tsx @@ -6,24 +6,30 @@ import { errorThrower } from '../errors/errorThrower'; import { hocChildrenNotAFunctionError } from '../errors/messages'; import { useAssertWrappedByClerkProvider } from '../hooks/useAssertWrappedByClerkProvider'; -export const withClerk =

( +export const withClerk =

( Component: React.ComponentType

, - displayName?: string, + displayNameOrOptions?: string | { component: string; renderWhileLoading?: boolean }, ) => { - displayName = displayName || Component.displayName || Component.name || 'Component'; + const passedDisplayedName = + typeof displayNameOrOptions === 'string' ? displayNameOrOptions : displayNameOrOptions?.component; + const displayName = passedDisplayedName || Component.displayName || Component.name || 'Component'; Component.displayName = displayName; + + const options = typeof displayNameOrOptions === 'string' ? undefined : displayNameOrOptions; + const HOC = (props: Without) => { useAssertWrappedByClerkProvider(displayName || 'withClerk'); const clerk = useIsomorphicClerkContext(); - if (!clerk.loaded) { + if (!clerk.loaded && !options?.renderWhileLoading) { return null; } return ( ); diff --git a/packages/react/src/types.ts b/packages/react/src/types.ts index eb1326d8d7..0c732dbc1c 100644 --- a/packages/react/src/types.ts +++ b/packages/react/src/types.ts @@ -70,7 +70,7 @@ export interface HeadlessBrowserClerkConstructor { new (publishableKey: string, options?: DomainOrProxyUrl): HeadlessBrowserClerk; } -export type WithClerkProp = T & { clerk: LoadedClerk }; +export type WithClerkProp = T & { clerk: LoadedClerk; component?: string }; export interface CustomPortalsRendererProps { customPagesPortals?: any[]; diff --git a/packages/react/src/utils/useWaitForComponentMount.ts b/packages/react/src/utils/useWaitForComponentMount.ts new file mode 100644 index 0000000000..dc6019b65f --- /dev/null +++ b/packages/react/src/utils/useWaitForComponentMount.ts @@ -0,0 +1,80 @@ +import { useEffect, useRef, useState } from 'react'; + +/** + * Used to detect when a Clerk component has been added to the DOM. + */ +function waitForElementChildren(options: { selector?: string; root?: HTMLElement | null; timeout?: number }) { + const { root = document?.body, selector, timeout = 0 } = options; + + return new Promise((resolve, reject) => { + if (!root) { + reject(new Error('No root element provided')); + return; + } + + let elementToWatch: HTMLElement | null = root; + if (selector) { + elementToWatch = root?.querySelector(selector); + } + + // Check if the element already has child nodes + const isElementAlreadyPresent = elementToWatch?.childElementCount && elementToWatch.childElementCount > 0; + if (isElementAlreadyPresent) { + resolve(); + return; + } + + // Set up a MutationObserver to detect when the element has children + const observer = new MutationObserver(mutationsList => { + for (const mutation of mutationsList) { + if (mutation.type === 'childList') { + if (!elementToWatch && selector) { + elementToWatch = root?.querySelector(selector); + } + + if (elementToWatch?.childElementCount && elementToWatch.childElementCount > 0) { + observer.disconnect(); + resolve(); + return; + } + } + } + }); + + observer.observe(root, { childList: true, subtree: true }); + + // Set up an optional timeout to reject the promise if the element never gets child nodes + if (timeout > 0) { + setTimeout(() => { + observer.disconnect(); + reject(new Error(`Timeout waiting for element children`)); + }, timeout); + } + }); +} + +/** + * Detect when a Clerk component has mounted by watching DOM updates to an element with a `data-clerk-component="${component}"` property. + */ +export function useWaitForComponentMount(component?: string) { + const watcherRef = useRef>(); + const [status, setStatus] = useState<'rendering' | 'rendered' | 'error'>('rendering'); + + useEffect(() => { + if (!component) { + throw new Error('Clerk: no component name provided, unable to detect mount.'); + } + + if (typeof window !== 'undefined' && !watcherRef.current) { + watcherRef.current = waitForElementChildren({ selector: `[data-clerk-component="${component}"]` }) + .then(() => { + setStatus('rendered'); + }) + .catch(() => { + setStatus('error'); + }); + } + }, [component]); + + return status; +}