Skip to content

Commit 8b466a9

Browse files
authored
fix(nextjs,clerk-js,types): Stop Clerk component flickering when used in App Router (clerk#2765)
1 parent 6ac9e71 commit 8b466a9

File tree

8 files changed

+117
-49
lines changed

8 files changed

+117
-49
lines changed

.changeset/olive-files-listen.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
'@clerk/clerk-js': patch
3+
'@clerk/nextjs': patch
4+
'@clerk/types': patch
5+
---
6+
7+
Prevent Clerk component flickering when mounted in a Next.js app using App Router

packages/clerk-js/src/core/clerk.redirects.test.ts

+14-14
Original file line numberDiff line numberDiff line change
@@ -126,56 +126,56 @@ describe('Clerk singleton - Redirects', () => {
126126
await clerkForProductionInstance.redirectToSignIn({ redirectUrl: 'https://www.example.com/' });
127127
await clerkForDevelopmentInstance.redirectToSignIn({ redirectUrl: 'https://www.example.com/' });
128128

129-
expect(mockNavigate).toHaveBeenNthCalledWith(1, '/sign-in#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
130-
expect(mockNavigate).toHaveBeenNthCalledWith(2, '/sign-in#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
129+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-in#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
130+
expect(mockNavigate.mock.calls[1][0]).toBe('/sign-in#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
131131
});
132132

133133
it('redirects to signUpUrl', async () => {
134134
await clerkForProductionInstance.redirectToSignUp({ redirectUrl: 'https://www.example.com/' });
135135
await clerkForDevelopmentInstance.redirectToSignUp({ redirectUrl: 'https://www.example.com/' });
136136

137-
expect(mockNavigate).toHaveBeenNthCalledWith(1, '/sign-up#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
138-
expect(mockNavigate).toHaveBeenNthCalledWith(2, '/sign-up#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
137+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-up#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
138+
expect(mockNavigate.mock.calls[1][0]).toBe('/sign-up#/?redirect_url=https%3A%2F%2Fwww.example.com%2F');
139139
});
140140

141141
it('redirects to userProfileUrl', async () => {
142142
await clerkForProductionInstance.redirectToUserProfile();
143143
await clerkForDevelopmentInstance.redirectToUserProfile();
144144

145-
expect(mockNavigate).toHaveBeenNthCalledWith(1, '/user-profile');
146-
expect(mockNavigate).toHaveBeenNthCalledWith(2, '/user-profile');
145+
expect(mockNavigate.mock.calls[0][0]).toBe('/user-profile');
146+
expect(mockNavigate.mock.calls[1][0]).toBe('/user-profile');
147147
});
148148

149149
it('redirects to afterSignUp', async () => {
150150
await clerkForProductionInstance.redirectToAfterSignUp();
151151
await clerkForDevelopmentInstance.redirectToAfterSignUp();
152152

153-
expect(mockNavigate).toHaveBeenNthCalledWith(1, '/');
154-
expect(mockNavigate).toHaveBeenNthCalledWith(2, '/');
153+
expect(mockNavigate.mock.calls[0][0]).toBe('/');
154+
expect(mockNavigate.mock.calls[1][0]).toBe('/');
155155
});
156156

157157
it('redirects to afterSignIn', async () => {
158158
await clerkForProductionInstance.redirectToAfterSignIn();
159159
await clerkForDevelopmentInstance.redirectToAfterSignIn();
160160

161-
expect(mockNavigate).toHaveBeenNthCalledWith(1, '/');
162-
expect(mockNavigate).toHaveBeenNthCalledWith(2, '/');
161+
expect(mockNavigate.mock.calls[0][0]).toBe('/');
162+
expect(mockNavigate.mock.calls[1][0]).toBe('/');
163163
});
164164

165165
it('redirects to create organization', async () => {
166166
await clerkForProductionInstance.redirectToCreateOrganization();
167167
await clerkForDevelopmentInstance.redirectToCreateOrganization();
168168

169-
expect(mockNavigate).toHaveBeenNthCalledWith(1, '/create-organization');
170-
expect(mockNavigate).toHaveBeenNthCalledWith(2, '/create-organization');
169+
expect(mockNavigate.mock.calls[0][0]).toBe('/create-organization');
170+
expect(mockNavigate.mock.calls[1][0]).toBe('/create-organization');
171171
});
172172

173173
it('redirects to organization profile', async () => {
174174
await clerkForProductionInstance.redirectToOrganizationProfile();
175175
await clerkForDevelopmentInstance.redirectToOrganizationProfile();
176176

177-
expect(mockNavigate).toHaveBeenNthCalledWith(1, '/organization-profile');
178-
expect(mockNavigate).toHaveBeenNthCalledWith(2, '/organization-profile');
177+
expect(mockNavigate.mock.calls[0][0]).toBe('/organization-profile');
178+
expect(mockNavigate.mock.calls[1][0]).toBe('/organization-profile');
179179
});
180180
});
181181

packages/clerk-js/src/core/clerk.test.ts

+18-18
Original file line numberDiff line numberDiff line change
@@ -571,7 +571,7 @@ describe('Clerk singleton', () => {
571571
const res = sut.navigate(toUrl);
572572
expect(res.then).toBeDefined();
573573
expect(mockHref).not.toHaveBeenCalled();
574-
expect(mockNavigate).toHaveBeenCalledWith('/path#hash');
574+
expect(mockNavigate.mock.calls[0][0]).toBe('/path#hash');
575575
expect(logSpy).not.toBeCalled();
576576
});
577577

@@ -591,7 +591,7 @@ describe('Clerk singleton', () => {
591591
const res = sut.navigate(toUrl);
592592
expect(res.then).toBeDefined();
593593
expect(mockHref).not.toHaveBeenCalled();
594-
expect(mockNavigate).toHaveBeenCalledWith('/path#hash');
594+
expect(mockNavigate.mock.calls[0][0]).toBe('/path#hash');
595595

596596
expect(logSpy).toBeCalledTimes(1);
597597
expect(logSpy).toBeCalledWith(`Clerk is navigating to: ${toUrl}`);
@@ -721,7 +721,7 @@ describe('Clerk singleton', () => {
721721

722722
await waitFor(() => {
723723
expect(mockSignUpCreate).toHaveBeenCalledWith({ transfer: true });
724-
expect(mockNavigate).toHaveBeenCalledWith('/sign-up#/continue');
724+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-up#/continue');
725725
});
726726
});
727727

@@ -934,7 +934,7 @@ describe('Clerk singleton', () => {
934934
await sut.handleRedirectCallback();
935935

936936
await waitFor(() => {
937-
expect(mockNavigate).toHaveBeenCalledWith('/sign-in#/factor-two');
937+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-in#/factor-two');
938938
});
939939
});
940940

@@ -977,7 +977,7 @@ describe('Clerk singleton', () => {
977977
});
978978

979979
await waitFor(() => {
980-
expect(mockNavigate).toHaveBeenCalledWith('/custom-2fa');
980+
expect(mockNavigate.mock.calls[0][0]).toBe('/custom-2fa');
981981
});
982982
});
983983

@@ -1032,7 +1032,7 @@ describe('Clerk singleton', () => {
10321032
});
10331033

10341034
await waitFor(() => {
1035-
expect(mockNavigate).toHaveBeenCalledWith('/custom-sign-in');
1035+
expect(mockNavigate.mock.calls[0][0]).toBe('/custom-sign-in');
10361036
});
10371037
});
10381038

@@ -1088,7 +1088,7 @@ describe('Clerk singleton', () => {
10881088
} as any);
10891089

10901090
await waitFor(() => {
1091-
expect(mockNavigate).toHaveBeenCalledWith('/custom-sign-in');
1091+
expect(mockNavigate.mock.calls[0][0]).toBe('/custom-sign-in');
10921092
});
10931093
});
10941094

@@ -1136,7 +1136,7 @@ describe('Clerk singleton', () => {
11361136
await sut.handleRedirectCallback();
11371137

11381138
await waitFor(() => {
1139-
expect(mockNavigate).toHaveBeenCalledWith('/sign-up');
1139+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-up');
11401140
});
11411141
});
11421142

@@ -1184,7 +1184,7 @@ describe('Clerk singleton', () => {
11841184
await sut.handleRedirectCallback();
11851185

11861186
await waitFor(() => {
1187-
expect(mockNavigate).toHaveBeenCalledWith('/sign-up');
1187+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-up');
11881188
});
11891189
});
11901190

@@ -1226,7 +1226,7 @@ describe('Clerk singleton', () => {
12261226
await sut.handleRedirectCallback();
12271227

12281228
await waitFor(() => {
1229-
expect(mockNavigate).toHaveBeenCalledWith('/sign-up#/continue');
1229+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-up#/continue');
12301230
});
12311231
});
12321232

@@ -1273,7 +1273,7 @@ describe('Clerk singleton', () => {
12731273
await sut.handleRedirectCallback();
12741274

12751275
await waitFor(() => {
1276-
expect(mockNavigate).toHaveBeenCalledWith('/sign-up#/verify-email-address');
1276+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-up#/verify-email-address');
12771277
});
12781278
});
12791279

@@ -1305,7 +1305,7 @@ describe('Clerk singleton', () => {
13051305
await sut.handleRedirectCallback();
13061306

13071307
await waitFor(() => {
1308-
expect(mockNavigate).toHaveBeenCalledWith('/sign-in#/factor-one');
1308+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-in#/factor-one');
13091309
});
13101310
});
13111311

@@ -1353,7 +1353,7 @@ describe('Clerk singleton', () => {
13531353
await sut.handleRedirectCallback();
13541354

13551355
await waitFor(() => {
1356-
expect(mockNavigate).toHaveBeenCalledWith('/sign-in#/factor-one');
1356+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-in#/factor-one');
13571357
});
13581358
});
13591359

@@ -1412,7 +1412,7 @@ describe('Clerk singleton', () => {
14121412
await sut.handleRedirectCallback();
14131413

14141414
await waitFor(() => {
1415-
expect(mockNavigate).toHaveBeenCalledWith('/sign-up');
1415+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-up');
14161416
});
14171417
});
14181418

@@ -1463,7 +1463,7 @@ describe('Clerk singleton', () => {
14631463
await sut.handleRedirectCallback();
14641464

14651465
await waitFor(() => {
1466-
expect(mockNavigate).toHaveBeenCalledWith('/sign-in');
1466+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-in');
14671467
});
14681468
});
14691469

@@ -1501,7 +1501,7 @@ describe('Clerk singleton', () => {
15011501
await sut.handleRedirectCallback();
15021502

15031503
await waitFor(() => {
1504-
expect(mockNavigate).toHaveBeenCalledWith('/sign-in#/reset-password');
1504+
expect(mockNavigate.mock.calls[0][0]).toBe('/sign-in#/reset-password');
15051505
});
15061506
});
15071507
});
@@ -1568,7 +1568,7 @@ describe('Clerk singleton', () => {
15681568

15691569
await waitFor(() => {
15701570
expect(mockSetActive).not.toHaveBeenCalled();
1571-
expect(mockNavigate).toHaveBeenCalledWith(redirectUrl);
1571+
expect(mockNavigate.mock.calls[0][0]).toBe(redirectUrl);
15721572
});
15731573
});
15741574

@@ -1628,7 +1628,7 @@ describe('Clerk singleton', () => {
16281628

16291629
await waitFor(() => {
16301630
expect(mockSetActive).not.toHaveBeenCalled();
1631-
expect(mockNavigate).toHaveBeenCalledWith(redirectUrl);
1631+
expect(mockNavigate.mock.calls[0][0]).toBe(redirectUrl);
16321632
});
16331633
});
16341634

packages/clerk-js/src/core/clerk.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -726,8 +726,9 @@ export class Clerk implements ClerkInterface {
726726
return;
727727
}
728728

729+
const metadata = options?.metadata ? { __internal_metadata: options?.metadata } : undefined;
729730
// React router only wants the path, search or hash portion.
730-
return await customNavigate(stripOrigin(toURL));
731+
return await customNavigate(stripOrigin(toURL), metadata);
731732
};
732733

733734
public buildUrlWithAuth(to: string): string {

packages/clerk-js/src/ui/router/BaseRouter.tsx

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import { useClerk } from '@clerk/shared/react';
2+
import type { NavigateOptions } from '@clerk/types';
23
import qs from 'qs';
34
import React from 'react';
45

@@ -14,7 +15,7 @@ interface BaseRouterProps {
1415
startPath: string;
1516
getPath: () => string;
1617
getQueryString: () => string;
17-
internalNavigate: (toURL: URL) => Promise<any> | any;
18+
internalNavigate: (toURL: URL, options?: NavigateOptions) => Promise<any> | any;
1819
onExternalNavigate?: () => any;
1920
refreshEvents?: Array<keyof WindowEventMap>;
2021
preservedParams?: string[];
@@ -114,7 +115,7 @@ export const BaseRouter = ({
114115
});
115116
toURL.search = qs.stringify(toQueryParams);
116117
}
117-
const internalNavRes = await internalNavigate(toURL);
118+
const internalNavRes = await internalNavigate(toURL, { metadata: { navigationType: 'internal' } });
118119
setRouteParts({ path: toURL.pathname, queryString: toURL.search });
119120
return internalNavRes;
120121
};

packages/nextjs/src/app-router/client/useAwaitableNavigate.ts

+25-8
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
import { useRouter } from 'next/navigation';
44
import { useEffect, useRef, useTransition } from 'react';
55

6-
type NavigateFunction = ReturnType<typeof useRouter>['push'];
6+
import type { NextClerkProviderProps } from '../../types';
77

88
/**
99
* Creates an "awaitable" navigation function that will do its best effort to wait for Next.js to finish its route transition.
@@ -12,31 +12,48 @@ type NavigateFunction = ReturnType<typeof useRouter>['push'];
1212
* `isPending` to flush the stored promises and ensure the navigates "resolve".
1313
*/
1414
export const useAwaitableNavigate = () => {
15-
// eslint-disable-next-line @typescript-eslint/unbound-method
16-
const { push } = useRouter();
15+
const router = useRouter();
1716
const [isPending, startTransition] = useTransition();
18-
const clerkNavRef = useRef<(...args: Parameters<NavigateFunction>) => Promise<void>>();
17+
const clerkNavRef = useRef<NonNullable<NextClerkProviderProps['routerPush']>>();
1918
const clerkNavPromiseBuffer = useRef<(() => void)[]>([]);
2019

2120
// Set the navigation function reference only once
2221
if (!clerkNavRef.current) {
23-
clerkNavRef.current = (to, opts) => {
22+
clerkNavRef.current = ((to, opts) => {
2423
return new Promise<void>(res => {
2524
clerkNavPromiseBuffer.current.push(res);
2625
startTransition(() => {
27-
push(to, opts);
26+
// If the navigation is internal, we should use the history API to navigate
27+
// as this is the way to perform a shallow navigation in Next.js App Router
28+
// without unmounting/remounting the page or fetching data from the server.
29+
if (opts?.__internal_metadata?.navigationType === 'internal') {
30+
// In 14.1.0, useSearchParams becomes reactive to shallow updates,
31+
// but only if passing `null` as the history state.
32+
// Older versions need to maintain the history state for push to work,
33+
// without affecting how the Next router works.
34+
const state = ((window as any).next?.version ?? '') < '14.1.0' ? history.state : null;
35+
window.history.pushState(state, '', to);
36+
} else {
37+
// If the navigation is external (usually when navigating away from the component but still within the app),
38+
// we should use the Next.js router to navigate as it will handle updating the URL and also
39+
// fetching the new page if necessary.
40+
router.push(to);
41+
}
2842
});
2943
});
30-
};
44+
}) as NextClerkProviderProps['routerPush'];
3145
}
3246

3347
// Handle flushing the promise buffer when pending is false. If pending is false and there are promises in the buffer we should be able to safely flush them.
3448
useEffect(() => {
35-
if (isPending) return;
49+
if (isPending) {
50+
return;
51+
}
3652

3753
if (clerkNavPromiseBuffer?.current?.length) {
3854
clerkNavPromiseBuffer.current.forEach(resolve => resolve());
3955
}
56+
4057
clerkNavPromiseBuffer.current = [];
4158
}, [isPending]);
4259

packages/nextjs/src/client-boundary/usePathnameWithoutCatchAll.tsx

+16-4
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,23 @@
11
import { useRouter } from 'next/compat/router';
2+
import React from 'react';
23

34
export const usePathnameWithoutWithCatchAll = () => {
5+
const pathRef = React.useRef<string>();
46
// The compat version of useRouter returns null instead of throwing an error
57
// when used inside app router instead of pages router
68
// we use it to detect if the component is used inside pages or app router
79
// so we can use the correct algorithm to get the path
810
const pagesRouter = useRouter();
911

1012
if (pagesRouter) {
11-
// in pages router things are simpler as the pathname includes the catch all route
12-
// which starts with [[... and we can just remove it
13-
return pagesRouter.pathname.replace(/\/\[\[\.\.\..*/, '');
13+
if (pathRef.current) {
14+
return pathRef.current;
15+
} else {
16+
// in pages router things are simpler as the pathname includes the catch all route
17+
// which starts with [[... and we can just remove it
18+
pathRef.current = pagesRouter.pathname.replace(/\/\[\[\.\.\..*/, '');
19+
return pathRef.current;
20+
}
1421
}
1522

1623
// require is used to avoid importing next/navigation when the pages router is used,
@@ -38,5 +45,10 @@ export const usePathnameWithoutWithCatchAll = () => {
3845
.flat(Infinity);
3946
// so we end up with the pathname where the components are mounted at
4047
// eg /user/123/profile/security will return /user/123/profile as the path
41-
return `/${pathParts.slice(0, pathParts.length - catchAllParams.length).join('/')}`;
48+
if (pathRef.current) {
49+
return pathRef.current;
50+
} else {
51+
pathRef.current = `/${pathParts.slice(0, pathParts.length - catchAllParams.length).join('/')}`;
52+
return pathRef.current;
53+
}
4254
};

0 commit comments

Comments
 (0)