Skip to content

Commit 7524943

Browse files
authoredMar 18, 2025··
chore(shared,clerk-react,types): Improve Typedoc (#5372)
1 parent 83a2a0e commit 7524943

28 files changed

+718
-48
lines changed
 

‎.changeset/green-donuts-press.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
---
2+
'@clerk/shared': patch
3+
'@clerk/clerk-react': patch
4+
'@clerk/types': patch
5+
---
6+
7+
Improve JSDoc documentation

‎.typedoc/custom-plugin.mjs

+71-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,43 @@
11
// @ts-check
2-
import { MarkdownRendererEvent } from 'typedoc-plugin-markdown';
2+
import { MarkdownPageEvent, MarkdownRendererEvent } from 'typedoc-plugin-markdown';
3+
4+
/**
5+
* A list of files where we want to remove any headings
6+
*/
7+
const FILES_WITHOUT_HEADINGS = [
8+
'use-organization-return.mdx',
9+
'use-organization-params.mdx',
10+
'paginated-resources.mdx',
11+
'pages-or-infinite-options.mdx',
12+
'pages-or-infinite-options.mdx',
13+
'paginated-hook-config.mdx',
14+
'use-organization-list-return.mdx',
15+
'use-organization-list-params.mdx',
16+
];
17+
18+
/**
19+
* An array of tuples where the first element is the file name and the second element is the new path.
20+
*/
21+
const LINK_REPLACEMENTS = [
22+
['clerk-paginated-response', '/docs/references/javascript/types/clerk-paginated-response'],
23+
['paginated-resources', '#paginated-resources'],
24+
];
25+
26+
/**
27+
* Inside the generated MDX files are links to other generated MDX files. These relative links need to be replaced with absolute links to pages that exist on clerk.com.
28+
* For example, `[Foobar](../../foo/bar.mdx)` needs to be replaced with `[Foobar](/docs/foo/bar)`.
29+
* It also shouldn't matter how level deep the relative link is.
30+
*
31+
* This function returns an array of `{ pattern: string, replace: string }` to pass into the `typedoc-plugin-replace-text` plugin.
32+
*/
33+
function getRelativeLinkReplacements() {
34+
return LINK_REPLACEMENTS.map(([fileName, newPath]) => {
35+
return {
36+
pattern: new RegExp(`\\((?:\\.{1,2}\\/)+.*?${fileName}\\.mdx\\)`, 'g'),
37+
replace: `(${newPath})`,
38+
};
39+
});
40+
}
341

442
/**
543
* @param {string} str
@@ -13,8 +51,9 @@ function toKebabCase(str) {
1351
*/
1452
export function load(app) {
1553
app.renderer.on(MarkdownRendererEvent.BEGIN, output => {
16-
// Do not output README.mdx files
54+
// Modify the output object
1755
output.urls = output.urls
56+
// Do not output README.mdx files
1857
?.filter(e => !e.url.endsWith('README.mdx'))
1958
.map(e => {
2059
// Convert URLs (by default camelCase) to kebab-case
@@ -23,7 +62,37 @@ export function load(app) {
2362
e.url = kebabUrl;
2463
e.model.url = kebabUrl;
2564

65+
/**
66+
* For the `@clerk/shared` package it outputs the hooks as for example: shared/react/hooks/use-clerk/functions/use-clerk.mdx.
67+
* It also places the interfaces as shared/react/hooks/use-organization/interfaces/use-organization-return.mdx
68+
* Group all those .mdx files under shared/react/hooks
69+
*/
70+
if (e.url.includes('shared/react/hooks')) {
71+
e.url = e.url.replace(/\/[^/]+\/(functions|interfaces)\//, '/');
72+
e.model.url = e.url;
73+
}
74+
2675
return e;
2776
});
2877
});
78+
79+
app.renderer.on(MarkdownPageEvent.END, output => {
80+
const fileName = output.url.split('/').pop();
81+
const linkReplacements = getRelativeLinkReplacements();
82+
83+
for (const { pattern, replace } of linkReplacements) {
84+
if (output.contents) {
85+
output.contents = output.contents.replace(pattern, replace);
86+
}
87+
}
88+
89+
if (fileName) {
90+
if (FILES_WITHOUT_HEADINGS.includes(fileName)) {
91+
if (output.contents) {
92+
// Remove any headings from the file, irrespective of the level
93+
output.contents = output.contents.replace(/^#+\s.+/gm, '');
94+
}
95+
}
96+
}
97+
});
2998
}

‎.typedoc/custom-theme.mjs

+110-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
// @ts-check
2-
import { ReflectionKind } from 'typedoc';
2+
import { ArrayType, IntersectionType, ReflectionKind, ReflectionType, UnionType } from 'typedoc';
33
import { MarkdownTheme, MarkdownThemeContext } from 'typedoc-plugin-markdown';
44

55
/**
@@ -36,7 +36,7 @@ class ClerkMarkdownThemeContext extends MarkdownThemeContext {
3636
this.partials = {
3737
...superPartials,
3838
/**
39-
* Copied from default theme / source code. This hides the return type from the output
39+
* Copied from default theme / source code. This hides the return type heading over the table from the output
4040
* https://github.com/typedoc2md/typedoc-plugin-markdown/blob/179a54c502b318cd4f3951e5e8b90f7f7a4752d8/packages/typedoc-plugin-markdown/src/theme/context/partials/member.signatureReturns.ts
4141
* @param {import('typedoc').SignatureReflection} model
4242
* @param {{ headingLevel: number }} options
@@ -235,6 +235,114 @@ class ClerkMarkdownThemeContext extends MarkdownThemeContext {
235235

236236
md.push(this.partials.body(model, { headingLevel: options.headingLevel }));
237237

238+
return md.join('\n\n');
239+
},
240+
/**
241+
* Copied from default theme / source code. This hides the "Type parameters" section and the declaration title from the output
242+
* https://github.com/typedoc2md/typedoc-plugin-markdown/blob/e798507a3c04f9ddf7710baf4cc7836053e438ff/packages/typedoc-plugin-markdown/src/theme/context/partials/member.declaration.ts
243+
* @param {import('typedoc').DeclarationReflection} model
244+
* @param {{ headingLevel: number, nested?: boolean }} options
245+
*/
246+
declaration: (model, options = { headingLevel: 2, nested: false }) => {
247+
const md = [];
248+
249+
const opts = {
250+
nested: false,
251+
...options,
252+
};
253+
254+
if (!opts.nested && model.sources && !this.options.getValue('disableSources')) {
255+
md.push(this.partials.sources(model));
256+
}
257+
258+
if (model?.documents) {
259+
md.push(
260+
this.partials.documents(model, {
261+
headingLevel: options.headingLevel,
262+
}),
263+
);
264+
}
265+
266+
/**
267+
* @type any
268+
*/
269+
const modelType = model.type;
270+
/**
271+
* @type {import('typedoc').DeclarationReflection}
272+
*/
273+
let typeDeclaration = modelType?.declaration;
274+
275+
if (model.type instanceof ArrayType && model.type?.elementType instanceof ReflectionType) {
276+
typeDeclaration = model.type?.elementType?.declaration;
277+
}
278+
279+
const hasTypeDeclaration =
280+
Boolean(typeDeclaration) ||
281+
(model.type instanceof UnionType && model.type?.types.some(type => type instanceof ReflectionType));
282+
283+
if (model.comment) {
284+
md.push(
285+
this.partials.comment(model.comment, {
286+
headingLevel: opts.headingLevel,
287+
showSummary: true,
288+
showTags: false,
289+
}),
290+
);
291+
}
292+
293+
if (model.type instanceof IntersectionType) {
294+
model.type?.types?.forEach(intersectionType => {
295+
if (intersectionType instanceof ReflectionType && !intersectionType.declaration.signatures) {
296+
if (intersectionType.declaration.children) {
297+
md.push(heading(opts.headingLevel, this.i18n.theme_type_declaration()));
298+
299+
md.push(
300+
this.partials.typeDeclaration(intersectionType.declaration, {
301+
headingLevel: opts.headingLevel,
302+
}),
303+
);
304+
}
305+
}
306+
});
307+
}
308+
309+
if (hasTypeDeclaration) {
310+
if (model.type instanceof UnionType) {
311+
if (this.helpers.hasUsefulTypeDetails(model.type)) {
312+
md.push(heading(opts.headingLevel, this.i18n.theme_type_declaration()));
313+
314+
model.type.types.forEach(type => {
315+
if (type instanceof ReflectionType) {
316+
md.push(this.partials.someType(type, { forceCollapse: true }));
317+
md.push(this.partials.typeDeclarationContainer(model, type.declaration, options));
318+
} else {
319+
md.push(`${this.partials.someType(type)}`);
320+
}
321+
});
322+
}
323+
} else {
324+
const useHeading =
325+
typeDeclaration?.children?.length &&
326+
(model.kind !== ReflectionKind.Property || this.helpers.useTableFormat('properties'));
327+
if (useHeading) {
328+
md.push(heading(opts.headingLevel, this.i18n.theme_type_declaration()));
329+
}
330+
md.push(this.partials.typeDeclarationContainer(model, typeDeclaration, options));
331+
}
332+
}
333+
if (model.comment) {
334+
md.push(
335+
this.partials.comment(model.comment, {
336+
headingLevel: opts.headingLevel,
337+
showSummary: false,
338+
showTags: true,
339+
showReturns: true,
340+
}),
341+
);
342+
}
343+
344+
md.push(this.partials.inheritance(model, { headingLevel: opts.headingLevel }));
345+
238346
return md.join('\n\n');
239347
},
240348
};

‎.typedoc/typedoc-prettier-config.json

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"tabWidth": 2,
3+
"semi": false,
4+
"singleQuote": true,
5+
"printWidth": 120,
6+
"useTabs": false,
7+
"bracketSpacing": true
8+
}

‎package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,7 @@
4848
"test:integration:tanstack-start": "E2E_APP_ID=tanstack.start pnpm test:integration:base --grep @tanstack-start",
4949
"test:integration:vue": "E2E_APP_ID=vue.vite pnpm test:integration:base --grep @vue",
5050
"turbo:clean": "turbo daemon clean",
51-
"typedoc:generate": "typedoc --tsconfig tsconfig.typedoc.json",
51+
"typedoc:generate": "pnpm build:declarations && typedoc --tsconfig tsconfig.typedoc.json",
5252
"version-packages": "changeset version && pnpm install --lockfile-only --engine-strict=false",
5353
"version-packages:canary": "./scripts/canary.mjs",
5454
"version-packages:snapshot": "./scripts/snapshot.mjs",

‎packages/react/.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
11
/*/
22
!/src/
3+
!/docs/

‎packages/react/docs/use-auth.md

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
<!-- #region nextjs-01 -->
2+
3+
```tsx {{ filename: 'app/external-data/page.tsx' }}
4+
'use client';
5+
6+
import { useAuth } from '@clerk/nextjs';
7+
8+
export default function ExternalDataPage() {
9+
const { userId, sessionId, getToken, isLoaded, isSignedIn } = useAuth();
10+
11+
const fetchExternalData = async () => {
12+
const token = await getToken();
13+
14+
// Fetch data from an external API
15+
const response = await fetch('https://api.example.com/data', {
16+
headers: {
17+
Authorization: `Bearer ${token}`,
18+
},
19+
});
20+
21+
return response.json();
22+
};
23+
24+
if (!isLoaded) {
25+
return <div>Loading...</div>;
26+
}
27+
28+
if (!isSignedIn) {
29+
return <div>Sign in to view this page</div>;
30+
}
31+
32+
return (
33+
<div>
34+
<p>
35+
Hello, {userId}! Your current active session is {sessionId}.
36+
</p>
37+
<button onClick={fetchExternalData}>Fetch Data</button>
38+
</div>
39+
);
40+
}
41+
```
42+
43+
<!-- #endregion nextjs-01 -->

‎packages/react/docs/use-sign-in.md

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
<!-- #region nextjs-01 -->
2+
3+
```tsx {{ filename: 'app/sign-in/page.tsx' }}
4+
'use client';
5+
6+
import { useSignIn } from '@clerk/nextjs';
7+
8+
export default function SignInPage() {
9+
const { isLoaded, signIn } = useSignIn();
10+
11+
if (!isLoaded) {
12+
// Handle loading state
13+
return null;
14+
}
15+
16+
return <div>The current sign-in attempt status is {signIn?.status}.</div>;
17+
}
18+
```
19+
20+
<!-- #endregion nextjs-01 -->

‎packages/react/docs/use-sign-up.md

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
<!-- #region nextjs-01 -->
2+
3+
```tsx {{ filename: 'app/sign-up/page.tsx' }}
4+
'use client';
5+
6+
import { useSignUp } from '@clerk/nextjs';
7+
8+
export default function SignUpPage() {
9+
const { isLoaded, signUp } = useSignUp();
10+
11+
if (!isLoaded) {
12+
// Handle loading state
13+
return null;
14+
}
15+
16+
return <div>The current sign-up attempt status is {signUp?.status}.</div>;
17+
}
18+
```
19+
20+
<!-- #endregion nextjs-01 -->

‎packages/react/src/hooks/useAuth.ts

+11
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,9 @@ import { createGetToken, createSignOut } from './utils';
2121
*
2222
* The following example demonstrates how to use the `useAuth()` hook to access the current auth state, like whether the user is signed in or not. It also includes a basic example for using the `getToken()` method to retrieve a session token for fetching data from an external resource.
2323
*
24+
* <Tabs items='React,Next.js'>
25+
* <Tab>
26+
*
2427
* ```tsx {{ filename: 'src/pages/ExternalDataPage.tsx' }}
2528
* import { useAuth } from '@clerk/clerk-react'
2629
*
@@ -58,6 +61,14 @@ import { createGetToken, createSignOut } from './utils';
5861
* )
5962
* }
6063
* ```
64+
*
65+
* </Tab>
66+
* <Tab>
67+
*
68+
* {@include ../../docs/use-auth.md#nextjs-01}
69+
*
70+
* </Tab>
71+
* </Tabs>
6172
*/
6273
export const useAuth = (initialAuthState: any = {}): UseAuthReturn => {
6374
useAssertWrappedByClerkProvider('useAuth');

‎packages/react/src/hooks/useSignIn.ts

+11
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ import { useAssertWrappedByClerkProvider } from './useAssertWrappedByClerkProvid
1313
*
1414
* The following example uses the `useSignIn()` hook to access the [`SignIn`](https://clerk.com/docs/references/javascript/sign-in/sign-in) object, which contains the current sign-in attempt status and methods to create a new sign-in attempt. The `isLoaded` property is used to handle the loading state.
1515
*
16+
* <Tabs items='React,Next.js'>
17+
* <Tab>
18+
*
1619
* ```tsx {{ filename: 'src/pages/SignInPage.tsx' }}
1720
* import { useSignIn } from '@clerk/clerk-react'
1821
*
@@ -28,6 +31,14 @@ import { useAssertWrappedByClerkProvider } from './useAssertWrappedByClerkProvid
2831
* }
2932
* ```
3033
*
34+
* </Tab>
35+
* <Tab>
36+
*
37+
* {@include ../../docs/use-sign-in.md#nextjs-01}
38+
*
39+
* </Tab>
40+
* </Tabs>
41+
*
3142
* @example
3243
* ### Create a custom sign-in flow with `useSignIn()`
3344
*

‎packages/react/src/hooks/useSignUp.ts

+11
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ import { useAssertWrappedByClerkProvider } from './useAssertWrappedByClerkProvid
1313
*
1414
* The following example uses the `useSignUp()` hook to access the [`SignUp`](https://clerk.com/docs/references/javascript/sign-up/sign-up) object, which contains the current sign-up attempt status and methods to create a new sign-up attempt. The `isLoaded` property is used to handle the loading state.
1515
*
16+
* <Tabs items='React,Next.js'>
17+
* <Tab>
18+
*
1619
* ```tsx {{ filename: 'src/pages/SignUpPage.tsx' }}
1720
* import { useSignUp } from '@clerk/clerk-react'
1821
*
@@ -28,6 +31,14 @@ import { useAssertWrappedByClerkProvider } from './useAssertWrappedByClerkProvid
2831
* }
2932
* ```
3033
*
34+
* </Tab>
35+
* <Tab>
36+
*
37+
* {@include ../../docs/use-sign-up.md#nextjs-01}
38+
*
39+
* </Tab>
40+
* </Tabs>
41+
*
3142
* @example
3243
* ### Create a custom sign-up flow with `useSignUp()`
3344
*

‎packages/shared/.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
11
/*/
22
!/src/
33
!/scripts/
4+
!/docs/

‎packages/shared/docs/use-clerk.md

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
<!-- #region nextjs-01 -->
2+
3+
```tsx {{ filename: 'app/page.tsx' }}
4+
'use client';
5+
6+
import { useClerk } from '@clerk/nextjs';
7+
8+
export default function HomePage() {
9+
const clerk = useClerk();
10+
11+
return <button onClick={() => clerk.openSignIn({})}>Sign in</button>;
12+
}
13+
```
14+
15+
<!-- #endregion nextjs-01 -->
+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
<!-- #region nextjs-01 -->
2+
3+
```tsx {{ filename: 'app/page.tsx' }}
4+
'use client';
5+
6+
import { useSessionList } from '@clerk/nextjs';
7+
8+
export default function HomePage() {
9+
const { isLoaded, sessions } = useSessionList();
10+
11+
if (!isLoaded) {
12+
// Handle loading state
13+
return null;
14+
}
15+
16+
return (
17+
<div>
18+
<p>Welcome back. You've been here {sessions.length} times before.</p>
19+
</div>
20+
);
21+
}
22+
```
23+
24+
<!-- #endregion nextjs-01 -->

‎packages/shared/docs/use-session.md

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
<!-- #region nextjs-01 -->
2+
3+
```tsx {{ filename: 'app/page.tsx' }}
4+
'use client';
5+
6+
import { useSession } from '@clerk/nextjs';
7+
8+
export default function HomePage() {
9+
const { isLoaded, session, isSignedIn } = useSession();
10+
11+
if (!isLoaded) {
12+
// Handle loading state
13+
return null;
14+
}
15+
if (!isSignedIn) {
16+
// Handle signed out state
17+
return null;
18+
}
19+
20+
return (
21+
<div>
22+
<p>This session has been active since {session.lastActiveAt.toLocaleString()}</p>
23+
</div>
24+
);
25+
}
26+
```
27+
28+
<!-- #endregion nextjs-01 -->

‎packages/shared/docs/use-user.md

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
<!-- #region nextjs-01 -->
2+
3+
```tsx {{ filename: 'app/page.tsx' }}
4+
'use client';
5+
6+
import { useUser } from '@clerk/nextjs';
7+
8+
export default function HomePage() {
9+
const { isLoaded, user } = useUser();
10+
11+
if (!isLoaded) {
12+
// Handle loading state
13+
return null;
14+
}
15+
16+
if (!user) return null;
17+
18+
const updateUser = async () => {
19+
await user.update({
20+
firstName: 'John',
21+
lastName: 'Doe',
22+
});
23+
};
24+
25+
return (
26+
<>
27+
<button onClick={updateUser}>Update your name</button>
28+
<p>user.firstName: {user?.firstName}</p>
29+
<p>user.lastName: {user?.lastName}</p>
30+
</>
31+
);
32+
}
33+
```
34+
35+
<!-- #endregion nextjs-01 -->
36+
37+
<!-- #region nextjs-02 -->
38+
39+
```tsx {{ filename: 'app/page.tsx' }}
40+
'use client';
41+
42+
import { useUser } from '@clerk/nextjs';
43+
44+
export default function HomePage() {
45+
const { isLoaded, user } = useUser();
46+
47+
if (!isLoaded) {
48+
// Handle loading state
49+
return null;
50+
}
51+
52+
if (!user) return null;
53+
54+
const updateUser = async () => {
55+
// Update data via an API endpoint
56+
const updateMetadata = await fetch('/api/updateMetadata');
57+
58+
// Check if the update was successful
59+
if (updateMetadata.message !== 'success') {
60+
throw new Error('Error updating');
61+
}
62+
63+
// If the update was successful, reload the user data
64+
await user.reload();
65+
};
66+
67+
return (
68+
<>
69+
<button onClick={updateUser}>Update your metadata</button>
70+
<p>user role: {user?.publicMetadata.role}</p>
71+
</>
72+
);
73+
}
74+
```
75+
76+
<!-- #endregion nextjs-02 -->

‎packages/shared/src/react/hooks/useClerk.ts

+11
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,9 @@ import { useAssertWrappedByClerkProvider, useClerkInstanceContext } from '../con
1414
*
1515
* The following example uses the `useClerk()` hook to access the `clerk` object. The `clerk` object is used to call the [`openSignIn()`](https://clerk.com/docs/references/javascript/clerk#sign-in) method to open the sign-in modal.
1616
*
17+
* <Tabs items='React,Next.js'>
18+
* <Tab>
19+
*
1720
* ```tsx {{ filename: 'src/Home.tsx' }}
1821
* import { useClerk } from '@clerk/clerk-react'
1922
*
@@ -23,6 +26,14 @@ import { useAssertWrappedByClerkProvider, useClerkInstanceContext } from '../con
2326
* return <button onClick={() => clerk.openSignIn({})}>Sign in</button>
2427
* }
2528
* ```
29+
*
30+
* </Tab>
31+
* <Tab>
32+
*
33+
* {@include ../../../docs/use-clerk.md#nextjs-01}
34+
*
35+
* </Tab>
36+
* </Tabs>
2637
*/
2738
export const useClerk = (): LoadedClerk => {
2839
useAssertWrappedByClerkProvider('useClerk');

‎packages/shared/src/react/hooks/useOrganization.tsx

+15-10
Original file line numberDiff line numberDiff line change
@@ -22,19 +22,24 @@ import {
2222
import type { PaginatedHookConfig, PaginatedResources, PaginatedResourcesWithDefault } from '../types';
2323
import { usePagesOrInfinite, useWithSafeValues } from './usePagesOrInfinite';
2424

25-
type UseOrganizationParams = {
25+
/**
26+
* @interface
27+
*/
28+
export type UseOrganizationParams = {
2629
/**
2730
* If set to `true`, all default properties will be used.
2831
*
2932
* Otherwise, accepts an object with the following optional properties:
3033
*
3134
* - `enrollmentMode`: A string that filters the domains by the provided enrollment mode.
35+
* - Any of the properties described in [Shared properties](#shared-properties).
3236
*/
3337
domains?: true | PaginatedHookConfig<GetDomainsParams>;
3438
/**
3539
* If set to `true`, all default properties will be used. Otherwise, accepts an object with the following optional properties:
3640
*
3741
* - `status`: A string that filters the membership requests by the provided status.
42+
* - Any of the properties described in [Shared properties](#shared-properties).
3843
*/
3944
membershipRequests?: true | PaginatedHookConfig<GetMembershipRequestParams>;
4045
/**
@@ -44,6 +49,7 @@ type UseOrganizationParams = {
4449
*
4550
* - `role`: An array of [`OrganizationCustomRoleKey`](/docs/references/javascript/types/organization-custom-role-key).
4651
* - `query`: A string that filters the memberships by the provided string.
52+
* - Any of the properties described in [Shared properties](#shared-properties).
4753
*/
4854
memberships?: true | PaginatedHookConfig<GetMembersParams>;
4955
/**
@@ -52,14 +58,15 @@ type UseOrganizationParams = {
5258
* Otherwise, accepts an object with the following optional properties:
5359
*
5460
* - `status`: A string that filters the invitations by the provided status.
61+
* - Any of the properties described in [Shared properties](#shared-properties).
5562
*/
5663
invitations?: true | PaginatedHookConfig<GetInvitationsParams>;
5764
};
5865

5966
/**
60-
* @inline
67+
* @interface
6168
*/
62-
type UseOrganizationReturn<T extends UseOrganizationParams> =
69+
export type UseOrganizationReturn<T extends UseOrganizationParams> =
6370
| {
6471
/**
6572
* A boolean that indicates whether Clerk has completed initialization. Initially `false`, becomes `true` once Clerk loads.
@@ -121,8 +128,6 @@ type UseOrganizationReturn<T extends UseOrganizationParams> =
121128
> | null;
122129
};
123130

124-
type UseOrganization = <T extends UseOrganizationParams>(params?: T) => UseOrganizationReturn<T>;
125-
126131
const undefinedPaginatedResource = {
127132
data: undefined,
128133
count: undefined,
@@ -149,7 +154,7 @@ const undefinedPaginatedResource = {
149154
*
150155
* To keep network usage to a minimum, developers are required to opt-in by specifying which resource they need to fetch and paginate through. By default, the `memberships`, `invitations`, `membershipRequests`, and `domains` attributes are not populated. You must pass `true` or an object with the desired properties to fetch and paginate the data.
151156
*
152-
* ```jsx
157+
* ```tsx
153158
* // invitations.data will never be populated.
154159
* const { invitations } = useOrganization()
155160
*
@@ -179,7 +184,7 @@ const undefinedPaginatedResource = {
179184
*
180185
* The following example demonstrates how to use the `infinite` property to fetch and append new data to the existing list. The `memberships` attribute will be populated with the first page of the organization's memberships. When the "Load more" button is clicked, the `fetchNext` helper function will be called to append the next page of memberships to the list.
181186
*
182-
* ```jsx
187+
* ```tsx
183188
* import { useOrganization } from '@clerk/clerk-react'
184189
*
185190
* export default function MemberList() {
@@ -225,7 +230,7 @@ const undefinedPaginatedResource = {
225230
*
226231
* Notice the difference between this example's pagination and the infinite pagination example above.
227232
*
228-
* ```jsx
233+
* ```tsx
229234
* import { useOrganization } from '@clerk/clerk-react'
230235
*
231236
* export default function MemberList() {
@@ -264,7 +269,7 @@ const undefinedPaginatedResource = {
264269
* }
265270
* ```
266271
*/
267-
export const useOrganization: UseOrganization = params => {
272+
export function useOrganization<T extends UseOrganizationParams>(params?: T): UseOrganizationReturn<T> {
268273
const {
269274
domains: domainListParams,
270275
membershipRequests: membershipRequestsListParams,
@@ -462,4 +467,4 @@ export const useOrganization: UseOrganization = params => {
462467
memberships,
463468
invitations,
464469
};
465-
};
470+
}

‎packages/shared/src/react/hooks/useOrganizationList.tsx

+154-27
Original file line numberDiff line numberDiff line change
@@ -16,17 +16,34 @@ import { useAssertWrappedByClerkProvider, useClerkInstanceContext, useUserContex
1616
import type { PaginatedHookConfig, PaginatedResources, PaginatedResourcesWithDefault } from '../types';
1717
import { usePagesOrInfinite, useWithSafeValues } from './usePagesOrInfinite';
1818

19-
type UseOrganizationListParams = {
19+
/**
20+
* @interface
21+
*/
22+
export type UseOrganizationListParams = {
2023
/**
21-
* `true` or an object with any of the properties described in [Shared properties](https://clerk.com/docs/references/react/use-organization-list#shared-properties). If set to `true`, all default properties will be used.
24+
* If set to `true`, all default properties will be used.
25+
*
26+
* Otherwise, accepts an object with the following optional properties:
27+
*
28+
* - Any of the properties described in [Shared properties](#shared-properties).
2229
*/
2330
userMemberships?: true | PaginatedHookConfig<GetUserOrganizationMembershipParams>;
2431
/**
25-
* `true` or an object with [`status: OrganizationInvitationStatus`](https://clerk.com/docs/references/react/use-organization-list#organization-invitation-status) or any of the properties described in [Shared properties](https://clerk.com/docs/references/react/use-organization-list#shared-properties). If set to `true`, all default properties will be used.
32+
* If set to `true`, all default properties will be used.
33+
*
34+
* Otherwise, accepts an object with the following optional properties:
35+
*
36+
* - `status`: A string that filters the invitations by the provided status.
37+
* - Any of the properties described in [Shared properties](#shared-properties).
2638
*/
2739
userInvitations?: true | PaginatedHookConfig<GetUserOrganizationInvitationsParams>;
2840
/**
29-
* `true` or an object with [`status: OrganizationSuggestionsStatus | OrganizationSuggestionStatus[]`](https://clerk.com/docs/references/react/use-organization-list#organization-suggestion-status) or any of the properties described in [Shared properties](https://clerk.com/docs/references/react/use-organization-list#shared-properties). If set to `true`, all default properties will be used.
41+
* If set to `true`, all default properties will be used.
42+
*
43+
* Otherwise, accepts an object with the following optional properties:
44+
*
45+
* - `status`: A string that filters the suggestions by the provided status.
46+
* - Any of the properties described in [Shared properties](#shared-properties).
3047
*/
3148
userSuggestions?: true | PaginatedHookConfig<GetUserOrganizationSuggestionsParams>;
3249
};
@@ -49,9 +66,10 @@ const undefinedPaginatedResource = {
4966
setData: undefined,
5067
} as const;
5168

52-
type UseOrganizationList = <T extends UseOrganizationListParams>(
53-
params?: T,
54-
) =>
69+
/**
70+
* @interface
71+
*/
72+
export type UseOrganizationListReturn<T extends UseOrganizationListParams> =
5573
| {
5674
/**
5775
* A boolean that indicates whether Clerk has completed initialization. Initially `false`, becomes `true` once Clerk loads.
@@ -79,35 +97,17 @@ type UseOrganizationList = <T extends UseOrganizationListParams>(
7997
userSuggestions: PaginatedResourcesWithDefault<OrganizationSuggestionResource>;
8098
}
8199
| {
82-
/**
83-
* A boolean that indicates whether Clerk has completed initialization. Initially `false`, becomes `true` once Clerk loads.
84-
*/
85100
isLoaded: boolean;
86-
/**
87-
* A function that returns a `Promise` which resolves to the newly created `Organization`.
88-
*/
89101
createOrganization: (params: CreateOrganizationParams) => Promise<OrganizationResource>;
90-
/**
91-
* A function that sets the active session and/or organization.
92-
*/
93102
setActive: SetActive;
94-
/**
95-
* Returns `PaginatedResources` which includes a list of the user's organization memberships.
96-
*/
97103
userMemberships: PaginatedResources<
98104
OrganizationMembershipResource,
99105
T['userMemberships'] extends { infinite: true } ? true : false
100106
>;
101-
/**
102-
* Returns `PaginatedResources` which includes a list of the user's organization invitations.
103-
*/
104107
userInvitations: PaginatedResources<
105108
UserOrganizationInvitationResource,
106109
T['userInvitations'] extends { infinite: true } ? true : false
107110
>;
108-
/**
109-
* Returns `PaginatedResources` which includes a list of suggestions for organizations that the user can join.
110-
*/
111111
userSuggestions: PaginatedResources<
112112
OrganizationSuggestionResource,
113113
T['userSuggestions'] extends { infinite: true } ? true : false
@@ -116,8 +116,135 @@ type UseOrganizationList = <T extends UseOrganizationListParams>(
116116

117117
/**
118118
* The `useOrganizationList()` hook provides access to the current user's organization memberships, invitations, and suggestions. It also includes methods for creating new organizations and managing the active organization.
119+
*
120+
* @example
121+
* ### Expanding and paginating attributes
122+
*
123+
* To keep network usage to a minimum, developers are required to opt-in by specifying which resource they need to fetch and paginate through. So by default, the `userMemberships`, `userInvitations`, and `userSuggestions` attributes are not populated. You must pass true or an object with the desired properties to fetch and paginate the data.
124+
*
125+
* ```tsx
126+
* // userMemberships.data will never be populated
127+
* const { userMemberships } = useOrganizationList()
128+
*
129+
* // Use default values to fetch userMemberships, such as initialPage = 1 and pageSize = 10
130+
* const { userMemberships } = useOrganizationList({
131+
* userMemberships: true,
132+
* })
133+
*
134+
* // Pass your own values to fetch userMemberships
135+
* const { userMemberships } = useOrganizationList({
136+
* userMemberships: {
137+
* pageSize: 20,
138+
* initialPage: 2, // skips the first page
139+
* },
140+
* })
141+
*
142+
* // Aggregate pages in order to render an infinite list
143+
* const { userMemberships } = useOrganizationList({
144+
* userMemberships: {
145+
* infinite: true,
146+
* },
147+
* })
148+
* ```
149+
*
150+
* @example
151+
* ### Infinite pagination
152+
*
153+
* The following example demonstrates how to use the `infinite` property to fetch and append new data to the existing list. The `userMemberships` attribute will be populated with the first page of the user's organization memberships. When the "Load more" button is clicked, the `fetchNext` helper function will be called to append the next page of memberships to the list.
154+
*
155+
* ```tsx {{ filename: 'src/components/JoinedOrganizations.tsx' }}
156+
* import { useOrganizationList } from '@clerk/clerk-react'
157+
* import React from 'react'
158+
*
159+
* const JoinedOrganizations = () => {
160+
* const { isLoaded, setActive, userMemberships } = useOrganizationList({
161+
* userMemberships: {
162+
* infinite: true,
163+
* },
164+
* })
165+
*
166+
* if (!isLoaded) {
167+
* return <>Loading</>
168+
* }
169+
*
170+
* return (
171+
* <>
172+
* <ul>
173+
* {userMemberships.data?.map((mem) => (
174+
* <li key={mem.id}>
175+
* <span>{mem.organization.name}</span>
176+
* <button onClick={() => setActive({ organization: mem.organization.id })}>Select</button>
177+
* </li>
178+
* ))}
179+
* </ul>
180+
*
181+
* <button disabled={!userMemberships.hasNextPage} onClick={() => userMemberships.fetchNext()}>
182+
* Load more
183+
* </button>
184+
* </>
185+
* )
186+
* }
187+
*
188+
* export default JoinedOrganizations
189+
* ```
190+
*
191+
* @example
192+
* ### Simple pagination
193+
*
194+
* The following example demonstrates how to use the `fetchPrevious` and `fetchNext` helper functions to paginate through the data. The `userInvitations` attribute will be populated with the first page of invitations. When the "Previous page" or "Next page" button is clicked, the `fetchPrevious` or `fetchNext` helper function will be called to fetch the previous or next page of invitations.
195+
*
196+
* Notice the difference between this example's pagination and the infinite pagination example above.
197+
*
198+
* ```tsx {{ filename: 'src/components/UserInvitationsTable.tsx' }}
199+
* import { useOrganizationList } from '@clerk/clerk-react'
200+
* import React from 'react'
201+
*
202+
* const UserInvitationsTable = () => {
203+
* const { isLoaded, userInvitations } = useOrganizationList({
204+
* userInvitations: {
205+
* infinite: true,
206+
* keepPreviousData: true,
207+
* },
208+
* })
209+
*
210+
* if (!isLoaded || userInvitations.isLoading) {
211+
* return <>Loading</>
212+
* }
213+
*
214+
* return (
215+
* <>
216+
* <table>
217+
* <thead>
218+
* <tr>
219+
* <th>Email</th>
220+
* <th>Org name</th>
221+
* </tr>
222+
* </thead>
223+
*
224+
* <tbody>
225+
* {userInvitations.data?.map((inv) => (
226+
* <tr key={inv.id}>
227+
* <th>{inv.emailAddress}</th>
228+
* <th>{inv.publicOrganizationData.name}</th>
229+
* </tr>
230+
* ))}
231+
* </tbody>
232+
* </table>
233+
*
234+
* <button disabled={!userInvitations.hasPreviousPage} onClick={userInvitations.fetchPrevious}>
235+
* Prev
236+
* </button>
237+
* <button disabled={!userInvitations.hasNextPage} onClick={userInvitations.fetchNext}>
238+
* Next
239+
* </button>
240+
* </>
241+
* )
242+
* }
243+
*
244+
* export default UserInvitationsTable
245+
* ```
119246
*/
120-
export const useOrganizationList: UseOrganizationList = params => {
247+
export function useOrganizationList<T extends UseOrganizationListParams>(params?: T): UseOrganizationListReturn<T> {
121248
const { userMemberships, userInvitations, userSuggestions } = params || {};
122249

123250
useAssertWrappedByClerkProvider('useOrganizationList');
@@ -253,4 +380,4 @@ export const useOrganizationList: UseOrganizationList = params => {
253380
userInvitations: invitations,
254381
userSuggestions: suggestions,
255382
};
256-
};
383+
}

‎packages/shared/src/react/hooks/useSession.ts

+11
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,9 @@ type UseSession = () => UseSessionReturn;
1212
*
1313
* The following example uses the `useSession()` hook to access the `Session` object, which has the `lastActiveAt` property. The `lastActiveAt` property is a `Date` object used to show the time the session was last active.
1414
*
15+
* <Tabs items='React,Next.js'>
16+
* <Tab>
17+
*
1518
* ```tsx {{ filename: 'src/Home.tsx' }}
1619
* import { useSession } from '@clerk/clerk-react'
1720
*
@@ -34,6 +37,14 @@ type UseSession = () => UseSessionReturn;
3437
* )
3538
* }
3639
* ```
40+
*
41+
* </Tab>
42+
* <Tab>
43+
*
44+
* {@include ../../../docs/use-session.md#nextjs-01}
45+
*
46+
* </Tab>
47+
* </Tabs>
3748
*/
3849
export const useSession: UseSession = () => {
3950
useAssertWrappedByClerkProvider('useSession');

‎packages/shared/src/react/hooks/useSessionList.ts

+11
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,9 @@ import { useAssertWrappedByClerkProvider, useClerkInstanceContext, useClientCont
1010
*
1111
* The following example uses `useSessionList()` to get a list of sessions that have been registered on the client device. The `sessions` property is used to show the number of times the user has visited the page.
1212
*
13+
* <Tabs items='React,Next.js'>
14+
* <Tab>
15+
*
1316
* ```tsx {{ filename: 'src/Home.tsx' }}
1417
* import { useSessionList } from '@clerk/clerk-react'
1518
*
@@ -28,6 +31,14 @@ import { useAssertWrappedByClerkProvider, useClerkInstanceContext, useClientCont
2831
* )
2932
* }
3033
* ```
34+
*
35+
* </Tab>
36+
* <Tab>
37+
*
38+
* {@include ../../../docs/use-session-list.md#nextjs-01}
39+
*
40+
* </Tab>
41+
* </Tabs>
3142
*/
3243
export const useSessionList = (): UseSessionListReturn => {
3344
useAssertWrappedByClerkProvider('useSessionList');

‎packages/shared/src/react/hooks/useUser.ts

+25
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,12 @@ import { useAssertWrappedByClerkProvider, useUserContext } from '../contexts';
3131
*
3232
* The following example uses the `useUser()` hook to access the [`User`](https://clerk.com/docs/references/javascript/user) object, which calls the [`update()`](https://clerk.com/docs/references/javascript/user#update) method to update the current user's information.
3333
*
34+
* <Tabs items='React,Next.js'>
35+
* <Tab>
36+
*
3437
* ```tsx {{ filename: 'src/Home.tsx' }}
38+
* import { useUser } from '@clerk/clerk-react'
39+
*
3540
* export default function Home() {
3641
* const { isLoaded, user } = useUser()
3742
*
@@ -58,13 +63,25 @@ import { useAssertWrappedByClerkProvider, useUserContext } from '../contexts';
5863
* )
5964
* }
6065
* ```
66+
* </Tab>
67+
* <Tab>
68+
*
69+
* {@include ../../../docs/use-user.md#nextjs-01}
70+
*
71+
* </Tab>
72+
* </Tabs>
6173
*
6274
* @example
6375
* ### Reload user data
6476
*
6577
* The following example uses the `useUser()` hook to access the [`User`](https://clerk.com/docs/references/javascript/user) object, which calls the [`reload()`](https://clerk.com/docs/references/javascript/user#reload) method to get the latest user's information.
6678
*
79+
* <Tabs items='React,Next.js'>
80+
* <Tab>
81+
*
6782
* ```tsx {{ filename: 'src/Home.tsx' }}
83+
* import { useUser } from '@clerk/clerk-react'
84+
*
6885
* export default function Home() {
6986
* const { isLoaded, user } = useUser()
7087
*
@@ -96,6 +113,14 @@ import { useAssertWrappedByClerkProvider, useUserContext } from '../contexts';
96113
* )
97114
* }
98115
* ```
116+
*
117+
* </Tab>
118+
* <Tab>
119+
*
120+
* {@include ../../../docs/use-user.md#nextjs-02}
121+
*
122+
* </Tab>
123+
* </Tabs>
99124
*/
100125
export function useUser(): UseUserReturn {
101126
useAssertWrappedByClerkProvider('useUser');

‎packages/shared/src/react/types.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,7 @@ export type PaginatedResourcesWithDefault<T> = {
8585
};
8686

8787
/**
88-
* @interface
88+
* @inline
8989
*/
9090
export type PaginatedHookConfig<T> = T & {
9191
/**

‎packages/shared/typedoc.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"$schema": "https://typedoc.org/schema.json",
3-
"entryPoints": ["./src/index.ts", "./src/react/index.ts", "./src/react/types.ts"],
3+
"entryPoints": ["./src/index.ts", "./src/react/types.ts", "./src/react/hooks/*.{ts,tsx}"],
44
"compilerOptions": {
55
"noImplicitAny": false
66
}

‎packages/types/src/clerk.ts

+6
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,9 @@ export type SignOutOptions = {
8181
redirectUrl?: string;
8282
};
8383

84+
/**
85+
* @inline
86+
*/
8487
export interface SignOut {
8588
(options?: SignOutOptions): Promise<void>;
8689

@@ -1529,6 +1532,9 @@ export type CreateBulkOrganizationInvitationParams = {
15291532
role: OrganizationCustomRoleKey;
15301533
};
15311534

1535+
/**
1536+
* @inline
1537+
*/
15321538
export interface CreateOrganizationParams {
15331539
name: string;
15341540
slug?: string;

‎packages/types/src/pagination.ts

+13-4
Original file line numberDiff line numberDiff line change
@@ -13,23 +13,32 @@ export type ClerkPaginationRequest<T = object> = {
1313
} & T;
1414

1515
/**
16-
* Pagination params in response
16+
* An interface that describes the response of a method that returns a paginated list of resources.
17+
*
18+
* > [!TIP]
19+
* > Clerk's SDKs always use `Promise<ClerkPaginatedResponse<T>>`. If the promise resolves, you will get back the properties. If the promise is rejected, you will receive a `ClerkAPIResponseError` or network error.
1720
*/
1821
export interface ClerkPaginatedResponse<T> {
22+
/**
23+
* An array that contains the fetched data.
24+
*/
1925
data: T[];
26+
/**
27+
* The total count of data that exist remotely.
28+
*/
2029
total_count: number;
2130
}
2231

2332
/**
24-
* Pagination params passed in FAPI client methods
33+
* @interface
2534
*/
2635
export type ClerkPaginationParams<T = object> = {
2736
/**
28-
* This is the starting point for your fetched results.
37+
* A number that specifies which page to fetch. For example, if `initialPage` is set to `10`, it will skip the first 9 pages and fetch the 10th page. Defaults to `1`.
2938
*/
3039
initialPage?: number;
3140
/**
32-
* Maximum number of items returned per request.
41+
* A number that specifies the maximum number of results to return per page. Defaults to `10`.
3342
*/
3443
pageSize?: number;
3544
} & T;

‎typedoc.config.mjs

+12
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ const typedocPluginMarkdownOptions = {
3333
fileExtension: '.mdx',
3434
excludeScopesInPaths: true,
3535
expandObjects: true,
36+
formatWithPrettier: true,
3637
};
3738

3839
/** @type {Partial<import("typedoc-plugin-replace-text").Config>} */
@@ -55,6 +56,17 @@ const typedocPluginReplaceTextOptions = {
5556
pattern: /```empty```/,
5657
replace: '',
5758
},
59+
{
60+
/**
61+
* In order to not render `<Tabs>` in the inline IntelliSense, the `items` prop was adjusted from `items={['item', 'item2']}` to `items='item,item2'`. It needs to be converted back so that clerk.com can render it properly.
62+
*/
63+
pattern: /Tabs items='([^']+)'/,
64+
replace: (_, match) =>
65+
`Tabs items={[${match
66+
.split(',')
67+
.map(item => `'${item.trim()}'`)
68+
.join(', ')}]}`,
69+
},
5870
],
5971
},
6072
};

0 commit comments

Comments
 (0)
Please sign in to comment.