diff --git a/.changeset/dirty-panthers-build.md b/.changeset/dirty-panthers-build.md
new file mode 100644
index 00000000000..4cdea456237
--- /dev/null
+++ b/.changeset/dirty-panthers-build.md
@@ -0,0 +1,13 @@
+---
+'@clerk/chrome-extension': minor
+'@clerk/localizations': minor
+'@clerk/clerk-js': minor
+'@clerk/nextjs': minor
+'@clerk/clerk-react': minor
+'@clerk/types': minor
+---
+
+Introduce the new brand new component
+
+- Lists all the memberships, invitations or suggestions an active user may have
+- Powered by our `useOrganizationList` react hook
diff --git a/packages/chrome-extension/src/__snapshots__/exports.test.ts.snap b/packages/chrome-extension/src/__snapshots__/exports.test.ts.snap
index 8f82a739b75..448480e58fd 100644
--- a/packages/chrome-extension/src/__snapshots__/exports.test.ts.snap
+++ b/packages/chrome-extension/src/__snapshots__/exports.test.ts.snap
@@ -9,6 +9,7 @@ exports[`public exports should not include a breaking change 1`] = `
"CreateOrganization",
"MagicLinkErrorCode",
"MultisessionAppSupport",
+ "OrganizationList",
"OrganizationProfile",
"OrganizationSwitcher",
"RedirectToCreateOrganization",
diff --git a/packages/clerk-js/bundlewatch.config.json b/packages/clerk-js/bundlewatch.config.json
index 36d5708d06a..4703390f128 100644
--- a/packages/clerk-js/bundlewatch.config.json
+++ b/packages/clerk-js/bundlewatch.config.json
@@ -8,6 +8,7 @@
{ "path": "./dist/impersonationfab*.js", "maxSize": "5KB" },
{ "path": "./dist/organizationprofile*.js", "maxSize": "10KB" },
{ "path": "./dist/organizationswitcher*.js", "maxSize": "5KB" },
+ { "path": "./dist/organizationlist*.js", "maxSize": "5.5KB" },
{ "path": "./dist/signin*.js", "maxSize": "10KB" },
{ "path": "./dist/signup*.js", "maxSize": "10KB" },
{ "path": "./dist/userbutton*.js", "maxSize": "5KB" },
diff --git a/packages/clerk-js/src/core/clerk.ts b/packages/clerk-js/src/core/clerk.ts
index b1462c15ad7..36a2a5cc3dc 100644
--- a/packages/clerk-js/src/core/clerk.ts
+++ b/packages/clerk-js/src/core/clerk.ts
@@ -32,6 +32,7 @@ import type {
InstanceType,
ListenerCallback,
OrganizationInvitationResource,
+ OrganizationListProps,
OrganizationMembershipResource,
OrganizationProfileProps,
OrganizationResource,
@@ -506,6 +507,23 @@ export default class Clerk implements ClerkInterface {
void this.#componentControls?.ensureMounted().then(controls => controls.unmountComponent({ node }));
};
+ public mountOrganizationList = (node: HTMLDivElement, props?: OrganizationListProps) => {
+ this.assertComponentsReady(this.#componentControls);
+ void this.#componentControls?.ensureMounted({ preloadHint: 'OrganizationList' }).then(controls =>
+ controls.mountComponent({
+ name: 'OrganizationList',
+ appearanceKey: 'organizationList',
+ node,
+ props,
+ }),
+ );
+ };
+
+ public unmountOrganizationList = (node: HTMLDivElement): void => {
+ this.assertComponentsReady(this.#componentControls);
+ void this.#componentControls?.ensureMounted().then(controls => controls.unmountComponent({ node }));
+ };
+
public mountUserButton = (node: HTMLDivElement, props?: UserButtonProps) => {
this.assertComponentsReady(this.#componentControls);
void this.#componentControls?.ensureMounted({ preloadHint: 'UserButton' }).then(controls =>
diff --git a/packages/clerk-js/src/ui/lazyModules/components.ts b/packages/clerk-js/src/ui/lazyModules/components.ts
index 3f1d60a9ff6..f4fe93f9a47 100644
--- a/packages/clerk-js/src/ui/lazyModules/components.ts
+++ b/packages/clerk-js/src/ui/lazyModules/components.ts
@@ -10,6 +10,7 @@ const componentImportPaths = {
import(/* webpackChunkName: "organizationprofile" */ './../components/OrganizationProfile'),
OrganizationSwitcher: () =>
import(/* webpackChunkName: "organizationswitcher" */ './../components/OrganizationSwitcher'),
+ OrganizationList: () => import(/* webpackChunkName: "organizationlist" */ './../components/OrganizationList'),
ImpersonationFab: () => import(/* webpackChunkName: "impersonationfab" */ './../components/ImpersonationFab'),
} as const;
@@ -50,6 +51,10 @@ export const OrganizationSwitcher = lazy(() =>
componentImportPaths.OrganizationSwitcher().then(module => ({ default: module.OrganizationSwitcher })),
);
+export const OrganizationList = lazy(() =>
+ componentImportPaths.OrganizationList().then(module => ({ default: module.OrganizationList })),
+);
+
export const ImpersonationFab = lazy(() =>
componentImportPaths.ImpersonationFab().then(module => ({ default: module.ImpersonationFab })),
);
@@ -64,6 +69,7 @@ export const ClerkComponents = {
UserButton,
UserProfile,
OrganizationSwitcher,
+ OrganizationList,
OrganizationProfile,
CreateOrganization,
SignInModal,
diff --git a/packages/nextjs/src/client-boundary/uiComponents.tsx b/packages/nextjs/src/client-boundary/uiComponents.tsx
index 52b84390d52..ef57bd5d6de 100644
--- a/packages/nextjs/src/client-boundary/uiComponents.tsx
+++ b/packages/nextjs/src/client-boundary/uiComponents.tsx
@@ -15,6 +15,7 @@ export {
SignInButton,
SignUpButton,
SignOutButton,
+ OrganizationList,
} from '@clerk/clerk-react';
export const SignIn = (props: SignInProps) => {
diff --git a/packages/nextjs/src/index.ts b/packages/nextjs/src/index.ts
index 55c663215ad..eb4c122a96c 100644
--- a/packages/nextjs/src/index.ts
+++ b/packages/nextjs/src/index.ts
@@ -29,6 +29,7 @@ export {
CreateOrganization,
SignInButton,
SignOutButton,
+ OrganizationList,
} from './client-boundary/uiComponents';
/**
diff --git a/packages/react/src/components/index.ts b/packages/react/src/components/index.ts
index 55050610cb4..1d00b724a32 100644
--- a/packages/react/src/components/index.ts
+++ b/packages/react/src/components/index.ts
@@ -6,6 +6,7 @@ export {
OrganizationSwitcher,
OrganizationProfile,
CreateOrganization,
+ OrganizationList,
} from './uiComponents';
export {
diff --git a/packages/react/src/components/uiComponents.tsx b/packages/react/src/components/uiComponents.tsx
index f5621dff59a..63bd8fe3982 100644
--- a/packages/react/src/components/uiComponents.tsx
+++ b/packages/react/src/components/uiComponents.tsx
@@ -1,5 +1,6 @@
import type {
CreateOrganizationProps,
+ OrganizationListProps,
OrganizationProfileProps,
OrganizationSwitcherProps,
SignInProps,
@@ -142,3 +143,14 @@ export const OrganizationSwitcher = withClerk(({ clerk, ...props }: WithClerkPro
/>
);
}, 'OrganizationSwitcher');
+
+export const OrganizationList = withClerk(({ clerk, ...props }: WithClerkProp) => {
+ return (
+
+ );
+}, 'OrganizationList');
diff --git a/packages/react/src/isomorphicClerk.ts b/packages/react/src/isomorphicClerk.ts
index 8927f0d8d61..2a6b3a930dc 100644
--- a/packages/react/src/isomorphicClerk.ts
+++ b/packages/react/src/isomorphicClerk.ts
@@ -11,6 +11,7 @@ import type {
HandleMagicLinkVerificationParams,
HandleOAuthCallbackParams,
ListenerCallback,
+ OrganizationListProps,
OrganizationMembershipResource,
OrganizationResource,
RedirectOptions,
@@ -71,6 +72,7 @@ export default class IsomorphicClerk {
private premountOrganizationProfileNodes = new Map();
private premountCreateOrganizationNodes = new Map();
private premountOrganizationSwitcherNodes = new Map();
+ private premountOrganizationListNodes = new Map();
private premountMethodCalls = new Map, MethodCallback>();
private loadedListeners: Array<() => void> = [];
@@ -266,6 +268,10 @@ export default class IsomorphicClerk {
clerkjs.mountUserButton(node, props);
});
+ this.premountOrganizationListNodes.forEach((props: OrganizationListProps, node: HTMLDivElement) => {
+ clerkjs.mountOrganizationList(node, props);
+ });
+
this.#loaded = true;
this.emitLoaded();
return this.clerkjs;
@@ -528,6 +534,22 @@ export default class IsomorphicClerk {
}
};
+ mountOrganizationList = (node: HTMLDivElement, props: OrganizationListProps): void => {
+ if (this.clerkjs && this.#loaded) {
+ this.clerkjs.mountOrganizationList(node, props);
+ } else {
+ this.premountOrganizationListNodes.set(node, props);
+ }
+ };
+
+ unmountOrganizationList = (node: HTMLDivElement): void => {
+ if (this.clerkjs && this.#loaded) {
+ this.clerkjs.unmountOrganizationList(node);
+ } else {
+ this.premountOrganizationListNodes.delete(node);
+ }
+ };
+
mountUserButton = (node: HTMLDivElement, userButtonProps: UserButtonProps): void => {
if (this.clerkjs && this.#loaded) {
this.clerkjs.mountUserButton(node, userButtonProps);
diff --git a/packages/types/src/clerk.ts b/packages/types/src/clerk.ts
index bc334cffd59..90cbf6f8375 100644
--- a/packages/types/src/clerk.ts
+++ b/packages/types/src/clerk.ts
@@ -218,14 +218,14 @@ export interface Clerk {
/**
* Mount an organization profile component at the target element.
- * @param targetNode Target to mount the UserProfile component.
+ * @param targetNode Target to mount the OrganizationProfile component.
* @param props Configuration parameters.
*/
mountOrganizationProfile: (targetNode: HTMLDivElement, props?: OrganizationProfileProps) => void;
/**
* Unmount the organization profile component from the target node.
- * @param targetNode Target node to unmount the UserProfile component from.
+ * @param targetNode Target node to unmount the OrganizationProfile component from.
*/
unmountOrganizationProfile: (targetNode: HTMLDivElement) => void;
@@ -244,17 +244,30 @@ export interface Clerk {
/**
* Mount an organization switcher component at the target element.
- * @param targetNode Target to mount the UserProfile component.
+ * @param targetNode Target to mount the OrganizationSwitcher component.
* @param props Configuration parameters.
*/
mountOrganizationSwitcher: (targetNode: HTMLDivElement, props?: OrganizationSwitcherProps) => void;
/**
* Unmount the organization profile component from the target node.*
- * @param targetNode Target node to unmount the UserProfile component from.
+ * @param targetNode Target node to unmount the OrganizationSwitcher component from.
*/
unmountOrganizationSwitcher: (targetNode: HTMLDivElement) => void;
+ /**
+ * Mount an organization list component at the target element.
+ * @param targetNode Target to mount the OrganizationList component.
+ * @param props Configuration parameters.
+ */
+ mountOrganizationList: (targetNode: HTMLDivElement, props?: OrganizationListProps) => void;
+
+ /**
+ * Unmount the organization list component from the target node.*
+ * @param targetNode Target node to unmount the OrganizationList component from.
+ */
+ unmountOrganizationList: (targetNode: HTMLDivElement) => void;
+
/**
* Register a listener that triggers a callback each time important Clerk resources are changed.
* Allows to hook up at different steps in the sign up, sign in processes.