From 4d73428e4d2123058d4ab113db6bce7dac8b0a9d Mon Sep 17 00:00:00 2001 From: Tom Milewski Date: Wed, 29 Nov 2023 01:20:39 -0800 Subject: [PATCH] fix(clerk-react): Fill out missing methods and properties from IsomorphicClerk (#2228) --- packages/react/src/isomorphicClerk.ts | 156 +++++++++++++++++++------- 1 file changed, 117 insertions(+), 39 deletions(-) diff --git a/packages/react/src/isomorphicClerk.ts b/packages/react/src/isomorphicClerk.ts index c969ee903f6..51a7884b385 100644 --- a/packages/react/src/isomorphicClerk.ts +++ b/packages/react/src/isomorphicClerk.ts @@ -64,6 +64,13 @@ type MethodCallback = () => Promise | unknown; type IsomorphicLoadedClerk = Omit< LoadedClerk, + | 'buildSignInUrl' + | 'buildSignUpUrl' + | 'buildUserProfileUrl' + | 'buildCreateOrganizationUrl' + | 'buildOrganizationProfileUrl' + | 'buildHomeUrl' + | 'buildUrlWithAuth' | 'redirectWithAuth' | 'redirectToSignIn' | 'redirectToSignUp' @@ -96,14 +103,29 @@ type IsomorphicLoadedClerk = Omit< // TODO: Align return type (maybe not possible or correct) getOrganization: (organizationId: string) => Promise; - mountUserButton: (targetNode: HTMLDivElement, userButtonProps: UserButtonProps) => void; - mountOrganizationList: (targetNode: HTMLDivElement, props: OrganizationListProps) => void; - mountOrganizationSwitcher: (targetNode: HTMLDivElement, props: OrganizationSwitcherProps) => void; - mountOrganizationProfile: (targetNode: HTMLDivElement, props: OrganizationProfileProps) => void; - mountCreateOrganization: (targetNode: HTMLDivElement, props: CreateOrganizationProps) => void; - mountSignUp: (targetNode: HTMLDivElement, signUpProps: SignUpProps) => void; - mountSignIn: (targetNode: HTMLDivElement, signInProps: SignInProps) => void; - mountUserProfile: (targetNode: HTMLDivElement, userProfileProps: UserProfileProps) => void; + // TODO: Align return type + buildSignInUrl: (opts?: RedirectOptions) => string | void; + // TODO: Align return type + buildSignUpUrl: (opts?: RedirectOptions) => string | void; + // TODO: Align return type + buildUserProfileUrl: () => string | void; + // TODO: Align return type + buildCreateOrganizationUrl: () => string | void; + // TODO: Align return type + buildOrganizationProfileUrl: () => string | void; + // TODO: Align return type + buildHomeUrl: () => string | void; + // TODO: Align return type + buildUrlWithAuth: (to: string, opts?: BuildUrlWithAuthParams | undefined) => string | void; + + mountUserButton: (node: HTMLDivElement, props: UserButtonProps) => void; + mountOrganizationList: (node: HTMLDivElement, props: OrganizationListProps) => void; + mountOrganizationSwitcher: (node: HTMLDivElement, props: OrganizationSwitcherProps) => void; + mountOrganizationProfile: (node: HTMLDivElement, props: OrganizationProfileProps) => void; + mountCreateOrganization: (node: HTMLDivElement, props: CreateOrganizationProps) => void; + mountSignUp: (node: HTMLDivElement, props: SignUpProps) => void; + mountSignIn: (node: HTMLDivElement, props: SignInProps) => void; + mountUserProfile: (node: HTMLDivElement, props: UserProfileProps) => void; client: ClientResource | undefined; }; @@ -194,43 +216,99 @@ export class IsomorphicClerk implements IsomorphicLoadedClerk { void this.loadClerkJS(); } - /** - * These are missing from IsomorphicClerk. - * These need to be aligned because `useClerk` exported by `@clerk/clerk-react` returns LoadedClerk - * Calling `const {buildUrlWithAuth} = useClerk()` will result in a runtime error as the function does not exist - */ - sdkMetadata?: SDKMetadata | undefined; - frontendApi: string; - isSatellite: boolean; - instanceType?: InstanceType | undefined; - isStandardBrowser?: boolean | undefined; - buildUrlWithAuth(to: string, opts?: BuildUrlWithAuthParams | undefined): string { - throw new Error('Method not implemented.'); - } - buildSignInUrl(opts?: RedirectOptions | undefined): string { - throw new Error('Method not implemented.'); + get sdkMetadata(): SDKMetadata | undefined { + return this.clerkjs?.sdkMetadata || this.options.sdkMetadata || undefined; } - buildSignUpUrl(opts?: RedirectOptions | undefined): string { - throw new Error('Method not implemented.'); - } - buildUserProfileUrl(): string { - throw new Error('Method not implemented.'); + + get instanceType(): InstanceType | undefined { + return this.clerkjs?.instanceType; } - buildCreateOrganizationUrl(): string { - throw new Error('Method not implemented.'); + + get frontendApi(): string { + return this.clerkjs?.frontendApi || ''; } - buildOrganizationProfileUrl(): string { - throw new Error('Method not implemented.'); + + get isStandardBrowser(): boolean { + return this.clerkjs?.isStandardBrowser || false; } - buildHomeUrl(): string { - throw new Error('Method not implemented.'); + + get isSatellite(): boolean { + return this.clerkjs?.isSatellite || false; } - handleUnauthenticated: () => Promise; - isReady: () => boolean; - /** - * END OF METHODS THAT NEED TO BE ALIGNED - */ + isReady = (): boolean => Boolean(this.clerkjs?.isReady()); + + buildSignInUrl = (opts?: RedirectOptions): string | void => { + const callback = () => this.clerkjs?.buildSignInUrl(opts) || ''; + if (this.clerkjs && this.#loaded) { + return callback(); + } else { + this.premountMethodCalls.set('buildSignInUrl', callback); + } + }; + + buildSignUpUrl = (opts?: RedirectOptions): string | void => { + const callback = () => this.clerkjs?.buildSignUpUrl(opts) || ''; + if (this.clerkjs && this.#loaded) { + return callback(); + } else { + this.premountMethodCalls.set('buildSignUpUrl', callback); + } + }; + + buildUserProfileUrl = (): string | void => { + const callback = () => this.clerkjs?.buildUserProfileUrl() || ''; + if (this.clerkjs && this.#loaded) { + return callback(); + } else { + this.premountMethodCalls.set('buildUserProfileUrl', callback); + } + }; + + buildCreateOrganizationUrl = (): string | void => { + const callback = () => this.clerkjs?.buildCreateOrganizationUrl() || ''; + if (this.clerkjs && this.#loaded) { + return callback(); + } else { + this.premountMethodCalls.set('buildCreateOrganizationUrl', callback); + } + }; + + buildOrganizationProfileUrl = (): string | void => { + const callback = () => this.clerkjs?.buildOrganizationProfileUrl() || ''; + if (this.clerkjs && this.#loaded) { + return callback(); + } else { + this.premountMethodCalls.set('buildOrganizationProfileUrl', callback); + } + }; + + buildHomeUrl = (): string | void => { + const callback = () => this.clerkjs?.buildHomeUrl() || ''; + if (this.clerkjs && this.#loaded) { + return callback(); + } else { + this.premountMethodCalls.set('buildHomeUrl', callback); + } + }; + + buildUrlWithAuth = (to: string, opts?: BuildUrlWithAuthParams | undefined): string | void => { + const callback = () => this.clerkjs?.buildUrlWithAuth(to, opts) || ''; + if (this.clerkjs && this.#loaded) { + return callback(); + } else { + this.premountMethodCalls.set('buildUrlWithAuth', callback); + } + }; + + handleUnauthenticated = async (): Promise => { + const callback = () => this.clerkjs?.handleUnauthenticated(); + if (this.clerkjs && this.#loaded) { + return callback() as Promise; + } else { + this.premountMethodCalls.set('handleUnauthenticated', callback); + } + }; async loadClerkJS(): Promise { if (this.mode !== 'browser' || this.#loaded) {