diff --git a/docs-site b/docs-site index 5499986934..25153f44c8 160000 --- a/docs-site +++ b/docs-site @@ -1 +1 @@ -Subproject commit 54999869347cd3de075ff6f9a350c1c5b3bdc9ef +Subproject commit 25153f44c8771b3ed33cbcff5bd240a4c58692ef diff --git a/docs/docs.yml b/docs/docs.yml index af5d552e2c..d13cd96478 100644 --- a/docs/docs.yml +++ b/docs/docs.yml @@ -1083,6 +1083,8 @@ navigation: contents: - section: Functions contents: + - page: addPasskey + path: wallets/pages/reference/alchemy/wagmi-core/functions/addPasskey.mdx - page: createConfig path: wallets/pages/reference/alchemy/wagmi-core/functions/createConfig.mdx - page: getAuthClient @@ -1097,6 +1099,8 @@ navigation: path: wallets/pages/reference/alchemy/wagmi-core/functions/listAuthMethods.mdx - page: loginWithOauth path: wallets/pages/reference/alchemy/wagmi-core/functions/loginWithOauth.mdx + - page: loginWithPasskey + path: wallets/pages/reference/alchemy/wagmi-core/functions/loginWithPasskey.mdx - page: prepareCalls path: wallets/pages/reference/alchemy/wagmi-core/functions/prepareCalls.mdx - page: prepareSwap diff --git a/docs/pages/reference/alchemy/react/components/AlchemyUiProvider.mdx b/docs/pages/reference/alchemy/react/components/AlchemyUiProvider.mdx new file mode 100644 index 0000000000..b3e00c286d --- /dev/null +++ b/docs/pages/reference/alchemy/react/components/AlchemyUiProvider.mdx @@ -0,0 +1,72 @@ +--- +# This file is autogenerated +title: AlchemyUiProvider +description: Overview of the AlchemyUiProvider method +slug: wallets/reference/alchemy/react/components/AlchemyUiProvider +--- + +Provider for for Alchemy UI components. `AuthCard` and `AuthModal` can only be rendered within this provider. + +## Import + +```ts +import { AlchemyUiProvider } from "@alchemy/react"; +``` + +## Usage + +```tsx +import { AlchemyUiProvider, createConfig } from "@account-kit/react"; +import { sepolia } from "@account-kit/infra"; +import { QueryClient } from "@tanstack/react-query"; +import { wagmiConfig } from "./wagmiConfig"; + +const ui: AlchemyAccountsUIConfig = { + illustrationStyle: "linear", + auth: { + sections: [ + [{ type: "email" }], + [{ type: "social", authProviderId: "google", mode: "popup" }], + ], + }, +}; + +const queryClient = new QueryClient(); + +function App({ children }: React.PropsWithChildren) { + return ( + + + {children} + + + ); +} +``` + +## Parameters + +### props + +`React.PropsWithChildren` +alchemy accounts provider props + +### props.queryClient + +`QueryClient` +the react-query query client to use + +### props.ui + +`AlchemyAccountsUIConfig` +ui configuration to use + +### props.children + +`React.ReactNode | undefined` +react components that should have this accounts context + +## Returns + +`React.JSX.Element` +The element to wrap your application in for Alchemy UI context. diff --git a/docs/pages/reference/alchemy/react/components/AuthCard.mdx b/docs/pages/reference/alchemy/react/components/AuthCard.mdx new file mode 100644 index 0000000000..bc81259959 --- /dev/null +++ b/docs/pages/reference/alchemy/react/components/AuthCard.mdx @@ -0,0 +1,41 @@ +--- +# This file is autogenerated +title: AuthCard +description: Overview of the AuthCard method +slug: wallets/reference/alchemy/react/components/AuthCard +--- + +React component containing an Auth view with configured auth methods +and options based on the config passed to the AlchemyAccountProvider + +## Import + +```ts +import { AuthCard } from "@alchemy/react"; +``` + +## Usage + +```tsx +import { AuthCard, useAlchemyAccountContext } from "@account-kit/react"; + +function ComponentWithAuthCard() { + // assumes you've passed in a UI config to the Account Provider + // you can also directly set the properties on the AuthCard component + const { uiConfig } = useAlchemyAccountContext(); + + return ; +} +``` + +## Parameters + +### props + +`AuthCardProps` +Card Props + +## Returns + +`JSX.Element` +a react component containing the AuthCard diff --git a/docs/pages/reference/alchemy/react/components/AuthModal.mdx b/docs/pages/reference/alchemy/react/components/AuthModal.mdx new file mode 100644 index 0000000000..af12f50ddc --- /dev/null +++ b/docs/pages/reference/alchemy/react/components/AuthModal.mdx @@ -0,0 +1,19 @@ +--- +# This file is autogenerated +title: AuthModal +description: Overview of the AuthModal method +slug: wallets/reference/alchemy/react/components/AuthModal +--- + +Renders the Auth Modal component. Must be rendered within an `AlchemyUiProvider`. To customize this modal, use the `ui` prop of the `AlchemyUiProvider`. + +## Import + +```ts +import { AuthModal } from "@alchemy/react"; +``` + +## Returns + +`React.JSX.Element` +The rendered Auth Modal component. diff --git a/docs/pages/reference/alchemy/react/hooks/useAddPasskey.mdx b/docs/pages/reference/alchemy/react/hooks/useAddPasskey.mdx new file mode 100644 index 0000000000..dd1b02d3f7 --- /dev/null +++ b/docs/pages/reference/alchemy/react/hooks/useAddPasskey.mdx @@ -0,0 +1,39 @@ +--- +# This file is autogenerated +title: useAddPasskey +description: Overview of the useAddPasskey method +slug: wallets/reference/alchemy/react/hooks/useAddPasskey +--- + +React hook for adding a passkey to an already authenticated account. + +This hook uses the `addPasskey` mutation to add a passkey to the authenticated account. + +## Import + +```ts +import { useAddPasskey } from "@alchemy/react"; +``` + +## Usage + +```tsx twoslash +import { useAddPasskey } from "@alchemy/react"; + +function AddPasskeyForm() { + const { addPasskey, isPending } = useAddPasskey(); +} +``` + +## Parameters + +### parameters + +`UseAddPasskeyParameters` + +- Configuration options for the hook + +## Returns + +`UseAddPasskeyReturnType` +TanStack Query mutation object diff --git a/docs/pages/reference/alchemy/react/hooks/useAuthModal.mdx b/docs/pages/reference/alchemy/react/hooks/useAuthModal.mdx new file mode 100644 index 0000000000..b9188e4f66 --- /dev/null +++ b/docs/pages/reference/alchemy/react/hooks/useAuthModal.mdx @@ -0,0 +1,37 @@ +--- +# This file is autogenerated +title: useAuthModal +description: Overview of the useAuthModal method +slug: wallets/reference/alchemy/react/hooks/useAuthModal +--- + +A [hook](https://github.com/alchemyplatform/aa-sdk/blob/main/account-kit/react/src/hooks/useAuthModal.ts) that returns the open and close functions for the Auth Modal if uiConfig +is enabled on the Account Provider + +## Import + +```ts +import { useAuthModal } from "@alchemy/react"; +``` + +## Usage + +```tsx twoslash +import React from "react"; +import { useAuthModal } from "@account-kit/react"; + +const ComponentWithAuthModal = () => { + const { openAuthModal } = useAuthModal(); + + return ( +
+ +
+ ); +}; +``` + +## Returns + +`UseAuthModalResult` +an object containing methods for opening or closing the auth modal. [ref](https://github.com/alchemyplatform/aa-sdk/blob/main/account-kit/react/src/hooks/useAuthModal.ts#L4) diff --git a/docs/pages/reference/alchemy/react/hooks/useLoginWithPasskey.mdx b/docs/pages/reference/alchemy/react/hooks/useLoginWithPasskey.mdx new file mode 100644 index 0000000000..1424922953 --- /dev/null +++ b/docs/pages/reference/alchemy/react/hooks/useLoginWithPasskey.mdx @@ -0,0 +1,81 @@ +--- +# This file is autogenerated +title: useLoginWithPasskey +description: Overview of the useLoginWithPasskey method +slug: wallets/reference/alchemy/react/hooks/useLoginWithPasskey +--- + +React hook for Passkey authentication - initiates authentication flow with the specified options. + +This hook wraps the `loginWithPasskey` action with React Query mutation functionality, +providing loading states, error handling, and mutation management for the OAuth authentication flow. + +## Import + +```ts +import { useLoginWithPasskey } from "@alchemy/react"; +``` + +## Usage + +```tsx twoslash +import { useLoginWithPasskey } from "@alchemy/react"; + +function LoginForm() { + const { loginWithPasskey, isPending, error } = useLoginWithPasskey(); + + const handleGoogleLogin = () => { + loginWithPasskey({ + provider: "google", + mode: "popup", // or 'redirect' + }); + }; + + return ( + + ); +} +``` + +## Parameters + +### parameters + +`UseLoginWithPasskeyParameters` + +- Configuration options for the hook + +### parameters.config + +`Config` + +- Optional wagmi config override + +### parameters.mutation + +`UseMutationParameters` + +- Optional React Query mutation configuration + +## Returns + +`UseLoginWithPasskeyReturnType` +TanStack Query mutation object with the following properties: + +- `loginWithPasskey`: `(variables: LoginWithPasskeyParameters, options?) => void` - Mutation function to initiate OAuth login +- `loginWithPasskeyAsync`: `(variables: LoginWithPasskeyParameters, options?) => Promise` - Async mutation function that returns a promise +- `data`: `LoginWithPasskeyReturnType | undefined` - The last successfully resolved data (void) +- `error`: `Error | null` - The error object for the mutation, if an error was encountered +- `isError`: `boolean` - True if the mutation is in an error state +- `isIdle`: `boolean` - True if the mutation is in its initial idle state +- `isPending`: `boolean` - True if the mutation is currently executing +- `isSuccess`: `boolean` - True if the last mutation attempt was successful +- `failureCount`: `number` - The failure count for the mutation +- `failureReason`: `Error | null` - The failure reason for the mutation retry +- `isPaused`: `boolean` - True if the mutation has been paused +- `reset`: `() => void` - Function to reset the mutation to its initial state +- `status`: `'idle' | 'pending' | 'error' | 'success'` - Current status of the mutation +- `submittedAt`: `number` - Timestamp for when the mutation was submitted +- `variables`: `LoginWithPasskeyParameters | undefined` - The variables object passed to the mutation diff --git a/docs/pages/reference/alchemy/wagmi-core/functions/addPasskey.mdx b/docs/pages/reference/alchemy/wagmi-core/functions/addPasskey.mdx new file mode 100644 index 0000000000..bead8b8a73 --- /dev/null +++ b/docs/pages/reference/alchemy/wagmi-core/functions/addPasskey.mdx @@ -0,0 +1,33 @@ +--- +# This file is autogenerated +title: addPasskey +description: Overview of the addPasskey method +slug: wallets/reference/alchemy/wagmi-core/functions/addPasskey +--- + +Adds a passkey to the authenticated user's account. + +## Import + +```ts +import { addPasskey } from "@alchemy/wagmi-core"; +``` + +## Parameters + +### config + +`Config` + +- The shared Wagmi/Alchemy config + +### parameters + +`AddPasskeyParameters` + +- The parameters for the passkey creation + +## Returns + +`Promise` +Promise that resolves when the passkey is added diff --git a/docs/pages/reference/alchemy/wagmi-core/functions/loginWithPasskey.mdx b/docs/pages/reference/alchemy/wagmi-core/functions/loginWithPasskey.mdx new file mode 100644 index 0000000000..17e9c5afe2 --- /dev/null +++ b/docs/pages/reference/alchemy/wagmi-core/functions/loginWithPasskey.mdx @@ -0,0 +1,33 @@ +--- +# This file is autogenerated +title: loginWithPasskey +description: Overview of the loginWithPasskey method +slug: wallets/reference/alchemy/wagmi-core/functions/loginWithPasskey +--- + +Initiates Passkey authentication flow with the specified parameters. + +## Import + +```ts +import { loginWithPasskey } from "@alchemy/wagmi-core"; +``` + +## Parameters + +### config + +`Config` + +- The shared Wagmi/Alchemy config + +### parameters + +`LoginWithPasskeyParameters` + +- Passkey authentication parameters + +## Returns + +`Promise` +Promise that resolves when authentication completes and connection is established diff --git a/examples/react-v5-example/src/app/globals.css b/examples/react-v5-example/src/app/globals.css index a2dc41ecee..5720672942 100644 --- a/examples/react-v5-example/src/app/globals.css +++ b/examples/react-v5-example/src/app/globals.css @@ -1,4 +1,5 @@ @import "tailwindcss"; +@config '../../tailwind.config.ts'; :root { --background: #ffffff; diff --git a/examples/react-v5-example/src/app/page.tsx b/examples/react-v5-example/src/app/page.tsx index af38951147..861cc469f8 100644 --- a/examples/react-v5-example/src/app/page.tsx +++ b/examples/react-v5-example/src/app/page.tsx @@ -33,6 +33,8 @@ import { useUpdateEmail, useUpdatePhoneNumber, useAuthMethods, + AuthModal, + useAuthModal, } from "@alchemy/react"; import { zeroAddress, Address } from "viem"; import { useState } from "react"; @@ -91,10 +93,18 @@ export default function Home() { } const AuthenticationDemo = () => { + const { openAuthModal } = useAuthModal(); const [authMode, setAuthMode] = useState<"email" | "sms" | "oauth">("email"); return (
+ +
+ +
+
+ ); +}; diff --git a/packages/react/src/components/auth/card/auth-card.tsx b/packages/react/src/components/auth/card/auth-card.tsx new file mode 100644 index 0000000000..e17074ed70 --- /dev/null +++ b/packages/react/src/components/auth/card/auth-card.tsx @@ -0,0 +1,174 @@ +"use client"; + +import { + useCallback, + useEffect, + useMemo, + useRef, + type PropsWithChildren, +} from "react"; +import { useAuthModal } from "../../../hooks/useAuthModal.js"; +import { useElementHeight } from "../../../hooks/useElementHeight.js"; +import { Navigation } from "../../navigation.js"; +import { useAuthContext } from "../context.js"; +import { Footer } from "../sections/Footer.js"; +import { Step } from "./steps.js"; +import { useAccount, useDisconnect } from "wagmi"; +import { useAuthConfig } from "../../../hooks/useAuthConfig.js"; +export type AuthCardProps = { + className?: string; +}; + +/** + * React component containing an Auth view with configured auth methods + * and options based on the config passed to the AlchemyAccountProvider + * + * @example + * ```tsx + * import { AuthCard, useAlchemyAccountContext } from "@account-kit/react"; + * + * function ComponentWithAuthCard() { + * // assumes you've passed in a UI config to the Account Provider + * // you can also directly set the properties on the AuthCard component + * const { uiConfig } = useAlchemyAccountContext(); + * + * return ( + * + * ); + * } + * ``` + * + * @param {AuthCardProps} props Card Props + * @returns {JSX.Element} a react component containing the AuthCard + */ +export const AuthCard = (props: AuthCardProps) => { + return ; +}; + +// this isn't used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export const AuthCardContent = ({ + className, + showClose = false, +}: { + className?: string; + showClose?: boolean; +}) => { + const { closeAuthModal } = useAuthModal(); + const { isConnected } = useAccount(); + const { authStep, setAuthStep } = useAuthContext(); + const { disconnect } = useDisconnect(); + + const didGoBack = useRef(false); + + const { onAuthSuccess, addPasskeyOnSignup } = useAuthConfig(); + + const canGoBack = useMemo(() => { + return [ + "email_verify", + "otp_verify", + "passkey_verify", + "passkey_create", + "pick_eoa", + "wallet_connect", + "eoa_connect", + "oauth_completing", + ].includes(authStep.type); + }, [authStep]); + + const onBack = useCallback(() => { + switch (authStep.type) { + case "email_verify": + case "otp_verify": + case "passkey_verify": + case "passkey_create": + case "oauth_completing": + disconnect(); // Terminate any inflight authentication + didGoBack.current = true; + setAuthStep({ type: "initial" }); + break; + case "wallet_connect": + case "eoa_connect": + setAuthStep({ type: "pick_eoa" }); + break; + case "pick_eoa": + setAuthStep({ type: "initial" }); + break; + default: + console.warn("Unhandled back action for auth step", authStep); + } + }, [authStep, setAuthStep, disconnect]); + + const onClose = useCallback(() => { + if (!isConnected) { + // Terminate any inflight authentication + disconnect(); + } + + if (authStep.type === "passkey_create") { + setAuthStep({ type: "complete" }); + } else { + setAuthStep({ type: "initial" }); + } + closeAuthModal(); + }, [isConnected, authStep.type, closeAuthModal, disconnect, setAuthStep]); + + useEffect(() => { + if (authStep.type === "complete") { + didGoBack.current = false; + closeAuthModal(); + onAuthSuccess?.(); + } else if (authStep.type !== "initial") { + didGoBack.current = false; + } + }, [ + authStep, + setAuthStep, + onAuthSuccess, + closeAuthModal, + addPasskeyOnSignup, + isConnected, + ]); + + return ( +
+ {/* Wrapper container that sizes its height dynamically */} + + {(canGoBack || showClose) && ( + + )} +
+ +
+
+ +
+ ); +}; + +const DynamicHeight = ({ children }: PropsWithChildren) => { + const contentRef = useRef(null); + const { height } = useElementHeight(contentRef); + + return ( +
+
+ {children} +
+
+ ); +}; diff --git a/packages/react/src/components/auth/card/content.tsx b/packages/react/src/components/auth/card/content.tsx new file mode 100644 index 0000000000..46c890f952 --- /dev/null +++ b/packages/react/src/components/auth/card/content.tsx @@ -0,0 +1,50 @@ +import type { ReactNode } from "react"; + +interface CardContentProps { + header: ReactNode | string; + icon?: ReactNode; + description: ReactNode | string; + error?: Error | string; + className?: string; + secondaryButton?: { + title: string; + onClick: () => void; + }; +} + +// eslint-disable-next-line jsdoc/require-jsdoc +export const CardContent = ({ + header, + icon, + description, + secondaryButton, + className, +}: CardContentProps) => { + return ( + <> +
+ {icon && ( +
+ {icon} +
+ )} + {header} + {typeof description === "string" ? ( +

+ {description} +

+ ) : ( + description + )} +
+ {secondaryButton && ( + + )} + + ); +}; diff --git a/packages/react/src/components/auth/card/error/connection-error.tsx b/packages/react/src/components/auth/card/error/connection-error.tsx new file mode 100644 index 0000000000..bbd86f7fbe --- /dev/null +++ b/packages/react/src/components/auth/card/error/connection-error.tsx @@ -0,0 +1,68 @@ +"use client"; +import { useEffect } from "react"; +import { ls } from "../../../strings.js"; +import { Button } from "../../../button.js"; +import { useDisconnect } from "wagmi"; + +type ConnectionErrorProps = { + headerText: string; + bodyText: string; + tryAgainCTA?: string; + icon: React.ReactNode; + handleTryAgain?: () => void; + handleUseAnotherMethod?: () => void; + shouldDisconnect?: boolean; + handleSkip?: () => void; +}; + +export const ConnectionError = ({ + headerText, + bodyText, + tryAgainCTA, + icon, + handleTryAgain, + handleUseAnotherMethod, + shouldDisconnect = true, + handleSkip, +}: ConnectionErrorProps) => { + const { disconnect } = useDisconnect(); + useEffect(() => { + // Terminate any inflight authentication on Error... + if (shouldDisconnect) { + disconnect(); + } + }, [disconnect, shouldDisconnect]); + + return ( +
+
+
+ {icon} +
+
+

{headerText}

+

{bodyText}

+ + {handleUseAnotherMethod && ( + + )} + {handleSkip && ( + + )} +
+ ); +}; diff --git a/packages/react/src/components/auth/card/error/general-error.tsx b/packages/react/src/components/auth/card/error/general-error.tsx new file mode 100644 index 0000000000..261b00b1d0 --- /dev/null +++ b/packages/react/src/components/auth/card/error/general-error.tsx @@ -0,0 +1,16 @@ +import { Warning } from "../../../../icons/warning.js"; +import { ls } from "../../../strings.js"; + +export const GeneralError = () => { + return ( +
+ +
+

+ {ls.error.general.title} +

+

{ls.error.general.body}

+
+
+ ); +}; diff --git a/packages/react/src/components/auth/card/footer/email-not-reveived.tsx b/packages/react/src/components/auth/card/footer/email-not-reveived.tsx new file mode 100644 index 0000000000..ce32e9963b --- /dev/null +++ b/packages/react/src/components/auth/card/footer/email-not-reveived.tsx @@ -0,0 +1,70 @@ +import { useEffect, useMemo, useState } from "react"; +import { ls } from "../../../strings.js"; +import { + AuthStepStatus, + useAuthContext, + type AuthStep, +} from "../../context.js"; +import { Button } from "../../../button.js"; +import { useSendEmailOtp } from "../../../../hooks/useSendEmailOtp.js"; + +type EmailNotReceivedDisclaimerProps = { + authStep: Extract; +}; +export const EmailNotReceivedDisclaimer = ({ + authStep, +}: EmailNotReceivedDisclaimerProps) => { + const { setAuthStep } = useAuthContext(); + const [emailResent, setEmailResent] = useState(false); + const { sendEmailOtp } = useSendEmailOtp({ + mutation: { + onSuccess: () => { + setAuthStep({ type: "complete" }); + }, + }, + }); + + const isOTPVerifying = useMemo(() => { + return ( + authStep.type === "otp_verify" && + (authStep.status === AuthStepStatus.verifying || + authStep.status === AuthStepStatus.success) + ); + }, [authStep]); + + useEffect(() => { + if (emailResent) { + // set the text back to "Resend" after 2 seconds + setTimeout(() => { + setEmailResent(false); + }, 2000); + } + }, [emailResent]); + + return ( +
+ + {ls.loadingEmail.emailNotReceived} + + +
+ ); +}; diff --git a/packages/react/src/components/auth/card/footer/help-text.tsx b/packages/react/src/components/auth/card/footer/help-text.tsx new file mode 100644 index 0000000000..f008717b6f --- /dev/null +++ b/packages/react/src/components/auth/card/footer/help-text.tsx @@ -0,0 +1,18 @@ +import { useUiConfig } from "../../../../hooks/useUiConfig.js"; +import { ls } from "../../../strings.js"; + +export const HelpText = () => { + const supportUrl = useUiConfig(({ supportUrl }) => supportUrl); + + if (!supportUrl) return null; + return ( +
+ + {ls.loadingPasskey.supportText} + + + {ls.loadingPasskey.supportLink} + +
+ ); +}; diff --git a/packages/react/src/components/auth/card/footer/oauth-contact-support.tsx b/packages/react/src/components/auth/card/footer/oauth-contact-support.tsx new file mode 100644 index 0000000000..ef4e25212e --- /dev/null +++ b/packages/react/src/components/auth/card/footer/oauth-contact-support.tsx @@ -0,0 +1,23 @@ +import { useUiConfig } from "../../../../hooks/useUiConfig.js"; +import { ls } from "../../../strings.js"; + +export const OAuthContactSupport = () => { + const { supportUrl } = useUiConfig(({ supportUrl }) => ({ supportUrl })); + return ( + supportUrl && ( +
+ + {ls.oauthContactSupport.title} + + + {ls.oauthContactSupport.body} + +
+ ) + ); +}; diff --git a/packages/react/src/components/auth/card/footer/protected-by.tsx b/packages/react/src/components/auth/card/footer/protected-by.tsx new file mode 100644 index 0000000000..7ada68319c --- /dev/null +++ b/packages/react/src/components/auth/card/footer/protected-by.tsx @@ -0,0 +1,16 @@ +import { AlchemyLogo } from "../../../../icons/alchemy.js"; +import { ls } from "../../../strings.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const ProtectedBy = () => ( + // eslint-disable-next-line react/jsx-no-target-blank + + {ls.protectedBy.title} + + +); diff --git a/packages/react/src/components/auth/card/footer/registration-disclaimer.tsx b/packages/react/src/components/auth/card/footer/registration-disclaimer.tsx new file mode 100644 index 0000000000..66858c9374 --- /dev/null +++ b/packages/react/src/components/auth/card/footer/registration-disclaimer.tsx @@ -0,0 +1,15 @@ +import { ls } from "../../../strings.js"; + +export const RegistrationDisclaimer = () => ( +
+ {ls.login.tosPrefix} + + {ls.login.tosLink} + +
+); diff --git a/packages/react/src/components/auth/card/loading/email.tsx b/packages/react/src/components/auth/card/loading/email.tsx new file mode 100644 index 0000000000..2da32e7931 --- /dev/null +++ b/packages/react/src/components/auth/card/loading/email.tsx @@ -0,0 +1,61 @@ +import { useEffect, useState } from "react"; +import { EmailIllustration } from "../../../../icons/illustrations/email.js"; +import { Spinner } from "../../../../icons/spinner.js"; +import { ls } from "../../../strings.js"; +import { useAuthContext } from "../../context.js"; +import { useAccount } from "wagmi"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const LoadingEmail = () => { + const { authStep } = useAuthContext("email_verify"); + // yup, re-sent and resent. I'm not fixing it + const [emailResent, setEmailResent] = useState(false); + + useEffect(() => { + if (emailResent) { + // set the text back to "Resend" after 2 seconds + setTimeout(() => { + setEmailResent(false); + }, 2000); + } + }, [emailResent]); + + return ( +
+
+ +
+ +

{ls.loadingEmail.title}

+

+ {ls.loadingEmail.verificationSent} +
+ {authStep.email} +

+
+ ); +}; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const CompletingEmailAuth = () => { + const { isConnected } = useAccount(); + const { setAuthStep } = useAuthContext("email_completing"); + + useEffect(() => { + if (isConnected) { + setAuthStep({ type: "complete" }); + } + }, [isConnected, setAuthStep]); + + return ( +
+
+ +
+ +

+ {ls.completingEmail.body} +

+
+ ); +}; diff --git a/packages/react/src/components/auth/card/loading/oauth.tsx b/packages/react/src/components/auth/card/loading/oauth.tsx new file mode 100644 index 0000000000..925a1ad057 --- /dev/null +++ b/packages/react/src/components/auth/card/loading/oauth.tsx @@ -0,0 +1,62 @@ +import { OauthCancelledError } from "@account-kit/signer"; +import { useEffect } from "react"; +import { + ContinueWithOAuth, + OAuthConnectionFailed, +} from "../../../../icons/oauth.js"; +import { useAuthContext } from "../../context.js"; +import { useOAuthVerify } from "../../hooks/useOAuthVerify.js"; +import { ConnectionError } from "../error/connection-error.js"; +import { ls } from "../../../strings.js"; +import { getSocialProviderDisplayName } from "../../types.js"; +import { useAccount } from "wagmi"; + +export const CompletingOAuth = () => { + const { isConnected } = useAccount(); + const { setAuthStep, authStep } = useAuthContext("oauth_completing"); + const { authenticate } = useOAuthVerify({ config: authStep.config }); + const oauthWasCancelled = authStep.error instanceof OauthCancelledError; + + useEffect(() => { + if (isConnected) { + setAuthStep({ type: "complete" }); + } else if (oauthWasCancelled) { + setAuthStep({ type: "initial" }); + } + }, [isConnected, oauthWasCancelled, setAuthStep]); + + if (authStep.error && !oauthWasCancelled) { + return ( + setAuthStep({ type: "initial" })} + icon={ + + } + /> + ); + } + + const providerDisplayName = getSocialProviderDisplayName(authStep.config); + return ( +
+
+ +
+ +

{`Continue with ${providerDisplayName}`}

+

+ {`Follow the steps in the pop up window to sign in with ${providerDisplayName}`} +

+
+ ); +}; diff --git a/packages/react/src/components/auth/card/loading/otp.tsx b/packages/react/src/components/auth/card/loading/otp.tsx new file mode 100644 index 0000000000..86b51fdbb4 --- /dev/null +++ b/packages/react/src/components/auth/card/loading/otp.tsx @@ -0,0 +1,111 @@ +import { useState } from "react"; +import { EmailIllustration } from "../../../../icons/illustrations/email.js"; +import { ls } from "../../../strings.js"; +import { + OTPInput, + type OTPCodeType, + initialOTPValue, + isOTPCodeType, +} from "../../../otp-input/otp-input.js"; +import { Spinner } from "../../../../icons/spinner.js"; +import { AuthStepStatus, useAuthContext } from "../../context.js"; +import { useAccount } from "wagmi"; +import { useSubmitOtpCode } from "../../../../hooks/useSubmitOtpCode.js"; + +const AUTH_DELAY = 1000; + +export const LoadingOtp = () => { + const { isConnected } = useAccount(); + const { setAuthStep, authStep } = useAuthContext("otp_verify"); + const [otpCode, setOtpCode] = useState(initialOTPValue); + const [errorText, setErrorText] = useState(authStep.error?.message || ""); + const [titleText, setTitleText] = useState(ls.loadingOtp.title); + + const resetOTP = (errorText = "") => { + setOtpCode(initialOTPValue); + setErrorText(errorText); + setTitleText(ls.loadingOtp.title); + }; + + const { submitOtpCode } = useSubmitOtpCode({ + mutation: { + onError: (error: any) => { + console.error(error); + + setAuthStep({ ...authStep, error, status: AuthStepStatus.base }); + resetOTP(getUserErrorMessage(error)); + }, + onSuccess: () => { + if (isConnected) { + setAuthStep({ ...authStep, status: AuthStepStatus.success }); + setTitleText(ls.loadingOtp.verified); + + // Wait 3 seconds before completing the auth step + setTimeout(() => { + setAuthStep({ type: "complete" }); + }, AUTH_DELAY); + } + }, + }, + }); + + const setValue = async (otpCode: OTPCodeType) => { + setOtpCode(otpCode); + if (isOTPCodeType(otpCode)) { + const otp = otpCode.join(""); + + setAuthStep({ ...authStep, status: AuthStepStatus.verifying }); + setTitleText(ls.loadingOtp.verifying); + + submitOtpCode({ otpCode: otp }); + } + }; + + return ( +
+
+ + +
+

+ {titleText} +

+

+ {ls.loadingOtp.body} +

+

+ {authStep.email} +

+ +
+ ); +}; + +function getUserErrorMessage(error: Error | undefined): string { + if (!error) { + return ""; + } + // Errors from Alchemy have a JSON message. + try { + const message = JSON.parse(error.message).error; + if (message === "invalid OTP code") { + return ls.error.otp.invalid; + } + return message; + } catch (e) { + // Ignore + } + return error.message; +} diff --git a/packages/react/src/components/auth/card/loading/passkey.tsx b/packages/react/src/components/auth/card/loading/passkey.tsx new file mode 100644 index 0000000000..2d128ef2df --- /dev/null +++ b/packages/react/src/components/auth/card/loading/passkey.tsx @@ -0,0 +1,54 @@ +import { ConnectionFailed } from "../../../../icons/connectionFailed.js"; +import { LoadingPasskey } from "../../../../icons/passkey.js"; +import { ls } from "../../../strings.js"; +import { useAuthContext } from "../../context.js"; +import { usePasskeyVerify } from "../../hooks/usePasskeyVerify.js"; +import { ConnectionError } from "../error/connection-error.js"; + +export const LoadingPasskeyAuth = () => { + const { setAuthStep, authStep } = useAuthContext("passkey_verify"); + const { loginWithPasskey } = usePasskeyVerify(); + + if (authStep.error) { + return ( + } + handleTryAgain={loginWithPasskey} + handleUseAnotherMethod={() => setAuthStep({ type: "initial" })} + /> + ); + } + + return ( +
+
+ +
+ +

{ls.loadingPasskey.title}

+

+ {ls.loadingPasskey.body} +

+ +
+ {/* Hidden until we can read in support URLs from the config */} + {/*
+

+ {ls.loadingPasskey.supportText} +

+ +
*/} +
+ +
+ ); +}; diff --git a/packages/react/src/components/auth/card/main.tsx b/packages/react/src/components/auth/card/main.tsx new file mode 100644 index 0000000000..fa19bddbaf --- /dev/null +++ b/packages/react/src/components/auth/card/main.tsx @@ -0,0 +1,30 @@ +import { Fragment } from "react"; +import { Divider } from "../../divider.js"; +import { useAuthContext } from "../context.js"; +import { AuthSection } from "../sections/AuthSection.js"; +import { GeneralError } from "./error/general-error.js"; +import { useAuthConfig } from "../../../hooks/useAuthConfig.js"; + +export const MainAuthContent = () => { + const { authStep } = useAuthContext(); + const isError = authStep.type === "initial" && authStep.error; + const { header, sections, hideSignInText } = useAuthConfig(); + + return ( + <> + {header} + {!hideSignInText &&

Sign in

} + {isError && } + {sections?.map((section, idx) => { + return ( + + + {idx !== sections.length - 1 ? ( + + ) : null} + + ); + })} + + ); +}; diff --git a/packages/react/src/components/auth/card/passkey-added.tsx b/packages/react/src/components/auth/card/passkey-added.tsx new file mode 100644 index 0000000000..58207b8523 --- /dev/null +++ b/packages/react/src/components/auth/card/passkey-added.tsx @@ -0,0 +1,23 @@ +import { AddedPasskeyIllustration } from "../../../icons/illustrations/added-passkey.js"; +import { useAuthContext } from "../context.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export function PasskeyAdded() { + const { setAuthStep } = useAuthContext(); + + setTimeout(() => { + setAuthStep({ type: "complete" }); + }, 5000); + + return ( +
+
+ +
+

Passkey added!

+

+ You can use this passkey to sign in next time. +

+
+ ); +} diff --git a/packages/react/src/components/auth/card/steps.tsx b/packages/react/src/components/auth/card/steps.tsx new file mode 100644 index 0000000000..bec526e855 --- /dev/null +++ b/packages/react/src/components/auth/card/steps.tsx @@ -0,0 +1,32 @@ +import { useAuthContext } from "../context.js"; +import { AddPasskey } from "./add-passkey.js"; +import { CompletingEmailAuth, LoadingEmail } from "./loading/email.js"; +import { CompletingOAuth } from "./loading/oauth.js"; +import { LoadingPasskeyAuth } from "./loading/passkey.js"; +import { MainAuthContent } from "./main.js"; +import { PasskeyAdded } from "./passkey-added.js"; +import { LoadingOtp } from "./loading/otp.js"; + +export const Step = () => { + const { authStep } = useAuthContext(); + switch (authStep.type) { + case "email_verify": + return ; + case "otp_verify": + return ; + case "passkey_verify": + return ; + case "email_completing": + return ; + case "oauth_completing": + return ; + case "passkey_create": + return ; + case "passkey_create_success": + return ; + case "complete": + case "initial": + default: + return ; + } +}; diff --git a/packages/react/src/components/auth/context.ts b/packages/react/src/components/auth/context.ts new file mode 100644 index 0000000000..05ee8d700b --- /dev/null +++ b/packages/react/src/components/auth/context.ts @@ -0,0 +1,103 @@ +"use client"; + +import type { Connector } from "@wagmi/core"; +import { createContext, useContext } from "react"; +import type { AuthType } from "./types"; + +export enum AuthStepStatus { + base = "base", + success = "success", + error = "error", + verifying = "verifying", +} + +export type AuthStep = + | { type: "email_verify"; email: string } + | { + type: "otp_verify"; + email: string; + error?: Error; + status?: AuthStepStatus; + } + | { + type: "totp_verify"; + previousStep: "magicLink"; + factorId: string; + email: string; + error?: Error; + } + | { + type: "totp_verify"; + previousStep: "otp"; + error?: Error; + } + | { type: "passkey_verify"; error?: Error } + | { type: "passkey_create"; error?: Error } + | { type: "passkey_create_success" } + | { type: "email_completing" } + | { + type: "oauth_completing"; + config: Extract; + error?: Error; + } + | { type: "initial"; error?: Error } + | { type: "complete" } + | { type: "eoa_connect"; connector: Connector; error?: Error } + | { type: "wallet_connect"; error?: Error } + | { type: "pick_eoa" }; + +type AuthContextType< + TType extends AuthStep["type"] | undefined = AuthStep["type"] | undefined, +> = TType extends undefined + ? { + authStep: AuthStep; + setAuthStep: (step: AuthStep) => void; + resetAuthStep: () => void; + } + : { + authStep: Extract }>; + setAuthStep: (step: AuthStep) => void; + resetAuthStep: () => void; + }; + +export const AuthModalContext = createContext( + undefined, +); + +export function useAuthContext< + TType extends AuthStep["type"] | undefined = AuthStep["type"] | undefined, +>(type?: TType): AuthContextType; + +/** + * A custom hook that provides the authentication context based on the specified authentication step type. It ensures that the hook is used within an `AuthModalProvider` and throws an error if the context is not available or if the current auth step type does not match the expected type. + * + * @example + * ```tsx twoslash + * import { useAuthContext } from "@account-kit/react"; + * + * const { authStep } = useAuthContext(); + * ``` + * + * @param {AuthStep["type"]} [type] Optional type of authentication step to validate against the current context + * @returns {AuthContextType} The authentication context for the current component + * @throws Will throw an error if the hook is not used within an `AuthModalProvider` or if the current auth step type does not match the expected type + */ export function useAuthContext( + type?: AuthStep["type"] | undefined, +): AuthContextType { + const context = useOptionalAuthContext(); + + if (!context) { + throw new Error( + "useAuthModalContext must be used within a AuthModalProvider", + ); + } + + if (type && context.authStep.type !== type) { + throw new Error(`expected authstep to be ${type}`); + } + + return context; +} + +export const useOptionalAuthContext = (): AuthContextType | undefined => + useContext(AuthModalContext); diff --git a/packages/react/src/components/auth/hooks/useOAuthVerify.ts b/packages/react/src/components/auth/hooks/useOAuthVerify.ts new file mode 100644 index 0000000000..1b40de86b3 --- /dev/null +++ b/packages/react/src/components/auth/hooks/useOAuthVerify.ts @@ -0,0 +1,51 @@ +"use client"; +import { useCallback } from "react"; +import { useAuthContext } from "../context.js"; +import type { AuthType } from "../types.js"; +import { useLoginWithOauth } from "../../../hooks/useLoginWithOauth.js"; + +export type UseOAuthVerifyReturnType = { + authenticate: () => void; + isPending: boolean; +}; + +export const useOAuthVerify = ({ + config, +}: { + config: Extract; +}): UseOAuthVerifyReturnType => { + const { setAuthStep } = useAuthContext(); + const { loginWithOauth, isPending } = useLoginWithOauth({ + mutation: { + onMutate: () => { + setAuthStep({ + config, + type: "oauth_completing", + }); + }, + onError: (err) => { + console.error(err); + setAuthStep({ + type: "oauth_completing", + config, + error: err, + }); + }, + onSuccess: () => { + setAuthStep({ type: "complete" }); + }, + }, + }); + + const authenticate = useCallback(() => { + loginWithOauth({ + ...config, + provider: config.authProviderId, + }); + }, [config, loginWithOauth]); + + return { + isPending, + authenticate, + }; +}; diff --git a/packages/react/src/components/auth/hooks/usePasskeyVerify.ts b/packages/react/src/components/auth/hooks/usePasskeyVerify.ts new file mode 100644 index 0000000000..030c9da13b --- /dev/null +++ b/packages/react/src/components/auth/hooks/usePasskeyVerify.ts @@ -0,0 +1,44 @@ +import { useCallback } from "react"; +import { useAuthContext } from "../context.js"; +import { useLoginWithPasskey } from "../../../hooks/useLoginWithPasskey.js"; + +export type UsePasskeyVerifyReturnType = { + loginWithPasskey: () => void; + isPending: boolean; +}; + +/** + * Internal UI component hook used to complete passkey verification. + * + * This is used to log in with a passkey, not create a new passkey + * + * @returns {UsePasskeyVerifyReturnType} an authenticate function to do passkey verification and a boolean indicating if the operation is pending + */ +export const usePasskeyVerify = (): UsePasskeyVerifyReturnType => { + const { setAuthStep } = useAuthContext(); + + const { loginWithPasskey: loginWithPasskey_, isPending } = + useLoginWithPasskey({ + mutation: { + onMutate: () => { + setAuthStep({ type: "passkey_verify" }); + }, + onError: (err) => { + setAuthStep({ type: "passkey_verify", error: err }); + }, + onSuccess: () => { + setAuthStep({ type: "complete" }); + }, + }, + }); + + const loginWithPasskey = useCallback( + () => loginWithPasskey_({}), + [loginWithPasskey_], + ); + + return { + isPending, + loginWithPasskey, + }; +}; diff --git a/packages/react/src/components/auth/modal.tsx b/packages/react/src/components/auth/modal.tsx new file mode 100644 index 0000000000..0a08d09665 --- /dev/null +++ b/packages/react/src/components/auth/modal.tsx @@ -0,0 +1,48 @@ +import { useEffect, useRef } from "react"; +import { useAuthModal } from "../../hooks/useAuthModal.js"; +import { useUiConfig } from "../../hooks/useUiConfig.js"; +import { Dialog } from "../dialog/dialog.js"; +import { AuthCardContent } from "./card/auth-card.js"; +import { useAuthContext } from "./context.js"; + +/** + * Renders the Auth Modal component. Must be rendered within an `AlchemyUiProvider`. To customize this modal, use the `ui` prop of the `AlchemyUiProvider`. + * + * @returns {React.JSX.Element} The rendered Auth Modal component. + */ +export const AuthModal = () => { + const { modalBaseClassName } = useUiConfig( + ({ modalBaseClassName, auth, uiMode = "modal" }) => ({ + modalBaseClassName, + addPasskeyOnSignup: auth?.addPasskeyOnSignup, + uiMode, + }), + ); + + const { isOpen, closeAuthModal } = useAuthModal(); + const { resetAuthStep } = useAuthContext(); + + // Reset the auth step to the initial state when the modal is closed. Aside + // from generally being better UX, this prevents the modal from getting stuck + // in the "complete" state after successfully logging in and then + // disconnecting. + const previousIsOpen = useRef(isOpen); + useEffect(() => { + if (previousIsOpen.current && !isOpen) { + resetAuthStep(); + } + previousIsOpen.current = isOpen; + }, [isOpen, resetAuthStep]); + + return ( + +
+ +
+
+ ); +}; diff --git a/packages/react/src/components/auth/sections/AuthSection.tsx b/packages/react/src/components/auth/sections/AuthSection.tsx new file mode 100644 index 0000000000..44bcd912e3 --- /dev/null +++ b/packages/react/src/components/auth/sections/AuthSection.tsx @@ -0,0 +1,32 @@ +import type { AuthType } from "../types.js"; +import { EmailAuth } from "./EmailAuth.js"; +import { ExternalWalletsAuth } from "./InjectedProvidersAuth.js"; +import { OAuth } from "./OAuth.js"; +import { PasskeyAuth } from "./PasskeyAuth.js"; + +type AuthSectionProps = { + authTypes: AuthType[]; +}; + +// This is not used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export const AuthSection = ({ authTypes, ...props }: AuthSectionProps) => { + return ( +
+ {authTypes.map((authType, index) => { + switch (authType.type) { + case "email": + return ; + case "passkey": + return ; + case "social": + return ; + case "external_wallets": + return ; + default: + throw new Error("Not implemented"); + } + })} +
+ ); +}; diff --git a/packages/react/src/components/auth/sections/EmailAuth.tsx b/packages/react/src/components/auth/sections/EmailAuth.tsx new file mode 100644 index 0000000000..f5d40ee3f8 --- /dev/null +++ b/packages/react/src/components/auth/sections/EmailAuth.tsx @@ -0,0 +1,119 @@ +import { useForm } from "@tanstack/react-form"; +import { zodValidator } from "@tanstack/zod-form-adapter"; +import { memo } from "react"; +import { z } from "zod"; +import { ChevronRight } from "../../../icons/chevron.js"; +import { MailIcon } from "../../../icons/mail.js"; +import { ls } from "../../strings.js"; +import { Button } from "../../button.js"; +import { Input } from "../../input.js"; +import { useAuthContext } from "../context.js"; +import type { AuthType } from "../types.js"; +import { useSendEmailOtp } from "../../../hooks/useSendEmailOtp.js"; + +type EmailAuthProps = Extract; + +// this is not used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export const EmailAuth = memo( + ({ + hideButton = false, + buttonLabel = ls.login.email.button, + placeholder = ls.login.email.placeholder, + }: EmailAuthProps) => { + const { setAuthStep } = useAuthContext(); + const { sendEmailOtp, isPending: isSendingEmailOtpPending } = + useSendEmailOtp({ + mutation: { + onMutate: (params) => { + setAuthStep({ type: "otp_verify", email: params.email }); + }, + onError: (e) => { + console.error(e); + const error = + e instanceof Error ? e : new Error("An Unknown error"); + setAuthStep({ type: "initial", error }); + }, + }, + }); + + // TODO: support magic link email + + const form = useForm({ + defaultValues: { + email: "", + }, + onSubmit: async ({ value: { email } }) => { + sendEmailOtp({ email }); + }, + validatorAdapter: zodValidator(), + }); + + return ( +
{ + e.stopPropagation(); + e.preventDefault(); + form.handleSubmit(); + }} + > +
+ + {(field) => ( + field.handleChange(e.target.value)} + onBlur={field.handleBlur} + placeholder={placeholder} + type="email" + iconLeft={} + iconRight={ + hideButton ? ( + + ) : undefined + } + disabled={isSendingEmailOtpPending} + /> + )} + + [ + state.canSubmit, + state.isSubmitting, + state.values.email, + ]} + > + {([canSubmit, isSubmitting, email]) => + !hideButton ? ( + + ) : null + } + +
+
+ ); + }, +); diff --git a/packages/react/src/components/auth/sections/Footer.tsx b/packages/react/src/components/auth/sections/Footer.tsx new file mode 100644 index 0000000000..82bf09be3f --- /dev/null +++ b/packages/react/src/components/auth/sections/Footer.tsx @@ -0,0 +1,43 @@ +import { EmailNotReceivedDisclaimer } from "../card/footer/email-not-reveived.js"; +import { HelpText } from "../card/footer/help-text.js"; +import { OAuthContactSupport } from "../card/footer/oauth-contact-support.js"; +import { ProtectedBy } from "../card/footer/protected-by.js"; +import { RegistrationDisclaimer } from "../card/footer/registration-disclaimer.js"; +import type { AuthStep } from "../context.js"; + +type FooterProps = { + authStep: AuthStep; +}; + +const RenderFooterText = ({ authStep }: FooterProps) => { + switch (authStep.type) { + case "initial": + return ; + case "email_verify": + case "otp_verify": + return ; + case "passkey_create": + case "wallet_connect": + case "passkey_verify": + return ; + case "oauth_completing": + return ; + case "email_completing": + case "totp_verify": + case "passkey_create_success": + case "eoa_connect": + case "pick_eoa": + case "complete": + return null; + } +}; +export const Footer = ({ authStep }: FooterProps) => { + return ( +
+ +
+ +
+
+ ); +}; diff --git a/packages/react/src/components/auth/sections/InjectedProvidersAuth.tsx b/packages/react/src/components/auth/sections/InjectedProvidersAuth.tsx new file mode 100644 index 0000000000..f5ae8870d2 --- /dev/null +++ b/packages/react/src/components/auth/sections/InjectedProvidersAuth.tsx @@ -0,0 +1,17 @@ +import { WalletIcon } from "../../../icons/wallet.js"; +import { Button } from "../../button.js"; +import { useAuthContext } from "../context.js"; + +export const ExternalWalletsAuth = () => { + const { setAuthStep } = useAuthContext(); + + return ( + + ); +}; diff --git a/packages/react/src/components/auth/sections/OAuth.tsx b/packages/react/src/components/auth/sections/OAuth.tsx new file mode 100644 index 0000000000..b754282c65 --- /dev/null +++ b/packages/react/src/components/auth/sections/OAuth.tsx @@ -0,0 +1,98 @@ +import { memo } from "react"; +import { + AppleIcon, + FacebookIcon, + GoogleIcon, + TwitchIcon, +} from "../../../icons/auth-icons/index.js"; +import { assertNever } from "../../../utils.js"; +import { Button } from "../../button.js"; +import { useOAuthVerify } from "../hooks/useOAuthVerify.js"; +import type { AuthType } from "../types.js"; + +type Props = Extract; + +// Not used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export const OAuth = memo(({ ...config }: Props) => { + const { authenticate } = useOAuthVerify({ config }); + + switch (config.authProviderId) { + case "google": + return ( + + ); + case "facebook": + return ( + + ); + case "apple": + return ( + + ); + case "twitch": + return ( + + ); + case "auth0": + return ( + + ); + default: + assertNever( + config, + `unhandled authProviderId ${ + (config as any).authProviderId + } passed into auth sections`, + ); + } +}); diff --git a/packages/react/src/components/auth/sections/PasskeyAuth.tsx b/packages/react/src/components/auth/sections/PasskeyAuth.tsx new file mode 100644 index 0000000000..cf1432c436 --- /dev/null +++ b/packages/react/src/components/auth/sections/PasskeyAuth.tsx @@ -0,0 +1,27 @@ +import { memo } from "react"; +import { PasskeyIcon } from "../../../icons/passkey.js"; +import { ls } from "../../strings.js"; +import { Button } from "../../button.js"; +import { usePasskeyVerify } from "../hooks/usePasskeyVerify.js"; + +type Props = { + label?: string; +}; + +// Not used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export const PasskeyAuth = memo( + ({ label = ls.login.passkey.button }: Props) => { + const { loginWithPasskey } = usePasskeyVerify(); + + return ( + + ); + }, +); diff --git a/packages/react/src/components/auth/types.ts b/packages/react/src/components/auth/types.ts new file mode 100644 index 0000000000..c670696b5d --- /dev/null +++ b/packages/react/src/components/auth/types.ts @@ -0,0 +1,44 @@ +import type { + KnownAuthProvider, + OauthRedirectConfig, +} from "@account-kit/signer"; +import type { WalletConnectParameters } from "wagmi/connectors"; +import { capitalize } from "../../utils.js"; + +export type AuthType = + | { + // TODO: this should support setting redirectParams which will be added to the email redirect + type: "email"; + /** @deprecated This option will be overriden by dashboard settings. Please use the dashboard settings instead. This option will be removed in a future release. */ + emailMode?: "magicLink" | "otp"; + hideButton?: boolean; + buttonLabel?: string; + placeholder?: string; + } + | { type: "passkey" } + | { type: "external_wallets"; walletConnect?: WalletConnectParameters } + | ({ type: "social"; scope?: string; claims?: string } & ( + | { + authProviderId: "auth0"; + isCustomProvider?: false; + auth0Connection?: string; + displayName: string; + logoUrl: string; + logoUrlDark?: string; + } + | { + authProviderId: KnownAuthProvider; + isCustomProvider?: false; + auth0Connection?: never; + displayName?: never; + logoUrl?: never; + logoUrlDark?: never; + } + ) & + OauthRedirectConfig); + +export function getSocialProviderDisplayName( + authType: Extract, +): string { + return authType.displayName ?? capitalize(authType.authProviderId); +} diff --git a/packages/react/src/components/button.tsx b/packages/react/src/components/button.tsx new file mode 100644 index 0000000000..2f407995b2 --- /dev/null +++ b/packages/react/src/components/button.tsx @@ -0,0 +1,46 @@ +import { + type ButtonHTMLAttributes, + type DetailedHTMLProps, + type ReactNode, +} from "react"; + +type ButtonProps = ( + | { variant?: "primary" | "secondary" | "link"; icon?: never } + | { variant: "social"; icon?: string | ReactNode } +) & + Omit< + DetailedHTMLProps< + ButtonHTMLAttributes, + HTMLButtonElement + >, + "variant" | "ref" + >; + +export const Button = ({ + variant, + children, + icon, + className, + ...props +}: ButtonProps) => { + const btnClass = (() => { + switch (variant) { + case "secondary": + return "akui-btn-secondary"; + case "social": + return "akui-btn-auth"; + case "link": + return "akui-btn-link"; + case "primary": + default: + return "akui-btn-primary"; + } + })(); + + return ( + + ); +}; diff --git a/packages/react/src/components/constants.ts b/packages/react/src/components/constants.ts new file mode 100644 index 0000000000..e668c4eef8 --- /dev/null +++ b/packages/react/src/components/constants.ts @@ -0,0 +1 @@ +export const IS_SIGNUP_QP = "aa-is-signup"; diff --git a/packages/react/src/components/dialog/dialog.tsx b/packages/react/src/components/dialog/dialog.tsx new file mode 100644 index 0000000000..57a2c40caa --- /dev/null +++ b/packages/react/src/components/dialog/dialog.tsx @@ -0,0 +1,107 @@ +"use client"; + +import { + useCallback, + useEffect, + useLayoutEffect, + useRef, + useState, + type ReactNode, +} from "react"; +import { createPortal } from "react-dom"; +import { RemoveScroll } from "react-remove-scroll"; +import { FocusTrap } from "./focustrap.js"; + +type DialogProps = { + isOpen: boolean; + onClose: () => void; + children: ReactNode; + fullWidth?: boolean; +}; + +/** + * Dialog component that renders a modal dialog. + * + * @param {DialogProps} props - The props for the Dialog component. + * @returns {JSX.Element | null} The rendered Dialog component. + */ +export const Dialog = ({ + isOpen, + onClose, + children, + fullWidth = false, +}: DialogProps) => { + const [isScrollLocked, setScrollLocked] = useState(false); + + const [renderPortal, setRenderPortal] = useState(false); + + const dialogCardRef = useRef(null); + + const handleBackgroundClick = useCallback(() => { + onClose(); + }, [onClose]); + + useLayoutEffect(() => { + const dialogCard = dialogCardRef.current; + + if (isOpen) { + setRenderPortal(true); + return; + } + + const renderPortalHandler = () => { + setRenderPortal(false); + }; + + dialogCard?.addEventListener("animationend", renderPortalHandler); + + return () => { + dialogCard?.removeEventListener("animationend", renderPortalHandler); + }; + }, [isOpen]); + + useEffect(() => { + const handleEscape = (event: KeyboardEvent) => { + if (event.key === "Escape" && isOpen) { + onClose(); + } + }; + + document.addEventListener("keydown", handleEscape); + return () => document.removeEventListener("keydown", handleEscape); + }, [isOpen, onClose]); + + useEffect(() => { + // Has to run in the browser + setScrollLocked(getComputedStyle(document.body).overflow !== "hidden"); + }, []); + + return renderPortal + ? createPortal( + + {/* Overlay */} +
+ +
event.stopPropagation()} + > + {children} +
+
+
+
, + document.body, + ) + : null; +}; diff --git a/packages/react/src/components/dialog/focustrap.tsx b/packages/react/src/components/dialog/focustrap.tsx new file mode 100644 index 0000000000..dcb2cb2130 --- /dev/null +++ b/packages/react/src/components/dialog/focustrap.tsx @@ -0,0 +1,80 @@ +// Adapted from: https://hiddedevries.nl/en/blog/2017-01-29-using-javascript-to-trap-focus-in-an-element + +import { useRef, useEffect, type PropsWithChildren } from "react"; + +function useTrapFocus() { + const ref = useRef(null); + + const handleKeyDown = (event: KeyboardEvent) => { + if (!ref.current) { + return; + } + + // Elements that can receive focus + const focusableElements = ref.current.querySelectorAll(` + a[href]:not([disabled]), + button:not([disabled]), + textarea:not([disabled]), + input[type="text"]:not([disabled]), + input[type="radio"]:not([disabled]), + input[type="checkbox"]:not([disabled]), + select:not([disabled]) + `); + + const firstFocusableElement = focusableElements[0]; + const lastFocusableElement = + focusableElements[focusableElements.length - 1]; + + const isPressingTab = event.key === "Tab"; + + if (!isPressingTab) { + return; + } + + if (event.shiftKey) { + // Shift + tab + if (document.activeElement === firstFocusableElement) { + lastFocusableElement.focus(); + event.preventDefault(); + } + } else { + // Just tab + if (document.activeElement === lastFocusableElement) { + firstFocusableElement.focus(); + event.preventDefault(); + } + } + }; + + useEffect(() => { + const el = ref.current; + if (!el) { + return; + } + + el.addEventListener("keydown", handleKeyDown); + el.focus({ preventScroll: true }); + + return () => { + el.removeEventListener("keydown", handleKeyDown); + }; + }, []); + + return { ref }; +} + +export const FocusTrap = ({ children }: PropsWithChildren<{}>) => { + const { ref } = useTrapFocus(); + + useEffect(() => { + if (ref.current) { + ref.current.focus({ preventScroll: true }); + } + }, [ref]); + + return ( +
+ {children} +
+ ); +}; diff --git a/packages/react/src/components/divider.tsx b/packages/react/src/components/divider.tsx new file mode 100644 index 0000000000..12217b021d --- /dev/null +++ b/packages/react/src/components/divider.tsx @@ -0,0 +1,13 @@ +import { ls } from "./strings.js"; + +// this isn't used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export const Divider = () => { + return ( +
+
+

{ls.login.or}

+
+
+ ); +}; diff --git a/packages/react/src/components/error.tsx b/packages/react/src/components/error.tsx new file mode 100644 index 0000000000..05307a63ab --- /dev/null +++ b/packages/react/src/components/error.tsx @@ -0,0 +1,17 @@ +interface ErrorContainerProps { + error?: Error | string; +} + +// eslint-disable-next-line jsdoc/require-jsdoc +export const ErrorContainer = ({ error }: ErrorContainerProps) => { + return ( +
+
+ Error +
+
+ {error != null ? error.toString() : "Oops! Something went wrong"} +
+
+ ); +}; diff --git a/packages/react/src/components/input.tsx b/packages/react/src/components/input.tsx new file mode 100644 index 0000000000..f8a61b0228 --- /dev/null +++ b/packages/react/src/components/input.tsx @@ -0,0 +1,48 @@ +type BaseProps = { + iconLeft?: React.ReactNode; + iconRight?: React.ReactNode; + error?: string; +} & Omit, "ref">; + +type Props = { + label?: string; + hint?: string; +} & BaseProps; + +const BaseInput = ({ + iconLeft, + iconRight, + error, + className, + ...props +}: BaseProps) => { + return ( + + ); +}; + +// this isn't used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export const Input = ({ label, hint, ...props }: Props) => { + if (label || hint || props.error) { + return ( + + ); + } + + return ; +}; diff --git a/packages/react/src/components/navigation.tsx b/packages/react/src/components/navigation.tsx new file mode 100644 index 0000000000..16f7b6d9fe --- /dev/null +++ b/packages/react/src/components/navigation.tsx @@ -0,0 +1,46 @@ +import { BackArrow, X } from "../icons/nav.js"; +import { Button } from "./button.js"; + +interface NavigationProps { + onBack?: () => void; + onClose: () => void; + showBack?: boolean; + showClose: boolean; +} + +// eslint-disable-next-line jsdoc/require-jsdoc +export const Navigation = ({ + showBack = false, + showClose, + onBack, + onClose, +}: NavigationProps) => { + return ( +
+ + + +
+ ); +}; diff --git a/packages/react/src/components/notification.tsx b/packages/react/src/components/notification.tsx new file mode 100644 index 0000000000..208ae50e56 --- /dev/null +++ b/packages/react/src/components/notification.tsx @@ -0,0 +1,29 @@ +type NotificationProps = { + type: "success" | "warning" | "error"; + message: string; + className?: string; +}; + +// this isn't used externally +// eslint-disable-next-line jsdoc/require-jsdoc +export function Notification({ className, type, message }: NotificationProps) { + const bgColor = (() => { + switch (type) { + case "success": + return "bg-bg-surface-success"; + case "warning": + return "bg-bg-surface-warning"; + case "error": + return "bg-bg-surface-error"; + } + })(); + return ( +
+ {message} +
+ ); +} diff --git a/packages/react/src/components/otp-input/otp-input.tsx b/packages/react/src/components/otp-input/otp-input.tsx new file mode 100644 index 0000000000..b4eaab5b09 --- /dev/null +++ b/packages/react/src/components/otp-input/otp-input.tsx @@ -0,0 +1,183 @@ +import React, { useEffect, useRef, useState } from "react"; +import { ls } from "../strings.js"; + +export type OTPCodeType = [string, string, string, string, string, string]; +export const initialOTPValue: OTPCodeType = ["", "", "", "", "", ""]; +const OTP_LENGTH = 6; +type OTPInputProps = { + errorText?: string; + value: OTPCodeType; + setValue: (otpCode: OTPCodeType) => void; + setErrorText: React.Dispatch>; + disabled?: boolean; + handleReset: () => void; + className?: string; + isVerified?: boolean; +}; + +export const isOTPCodeType = (arg: string[]): arg is OTPCodeType => { + return ( + Array.isArray(arg) && + arg.every((item: string) => typeof item === "string" && item !== "") && + arg.length === OTP_LENGTH + ); +}; + +export const OTPInput: React.FC = ({ + value, + setValue, + errorText, + disabled, + setErrorText, + handleReset, + className, + isVerified, +}) => { + const [autoComplete, setAutoComplete] = useState(""); + const [activeElement, setActiveElement] = useState(0); + + const refs = useRef>([]); + // Initialize refs + useEffect(() => { + refs.current = refs.current.slice(0, OTP_LENGTH); + refs.current[0]?.focus(); + }, []); + // Select active element when active element value changes + useEffect(() => { + if (activeElement !== null && refs.current[activeElement]) { + refs.current[activeElement]?.select(); + refs.current[activeElement]?.focus(); + } + }, [activeElement]); + + useEffect(() => { + const newValue = autoComplete.split(""); + if (isOTPCodeType(newValue)) { + setValue(newValue); + } + }, [autoComplete, setValue]); + + const handleChange = (e: React.ChangeEvent, i: number) => { + //Fix for ios chrome autocomplete + if (e.target.value.length === OTP_LENGTH) { + const chromeIOSAutocomplete = e.target.value.split(""); + if (isOTPCodeType(chromeIOSAutocomplete)) { + setValue(chromeIOSAutocomplete); + setActiveElement(null); + return; + } + } + const newValue = [...value] as OTPCodeType; + newValue.splice(i, 1, e.target.value); + setErrorText(""); + setValue(newValue); + focusNextElement(); + }; + + const handleClick = (i: number) => { + refs.current[i]?.select(); + setActiveElement(i); + setErrorText(""); + }; + + const focusNextElement = () => { + const nextElement = activeElement ? activeElement + 1 : 1; + setActiveElement(nextElement); + }; + + const focusPreviousElement = () => { + const previousElement = activeElement ? activeElement - 1 : 0; + setActiveElement(previousElement); + }; + + const handlePaste = (e: React.ClipboardEvent) => { + e.preventDefault(); + const pasteData = e.clipboardData + .getData("text/plain") + .split("") + .slice(0, OTP_LENGTH); + if (isOTPCodeType(pasteData)) { + setValue(pasteData); + } else { + setErrorText(ls.error.otp.invalid); + } + }; + + const handleKeydown = (e: React.KeyboardEvent) => { + if (activeElement === null) return; + switch (e.key) { + case "Backspace": + e.preventDefault(); + const newValue = [...value] as OTPCodeType; + newValue.splice(activeElement, 1, ""); + setValue(newValue); + focusPreviousElement(); + break; + case "ArrowLeft": + e.preventDefault(); + focusPreviousElement(); + break; + case "ArrowRight": + e.preventDefault(); + if (activeElement < OTP_LENGTH - 1) focusNextElement(); + break; + case "Spacebar": { + break; + } + } + }; + + return ( +
+ {/* Input for autocomplete, visibility hidden */} + setAutoComplete(e.target.value)} + onClick={handleReset} + /> +
+ {initialOTPValue.map((_, i) => ( + (refs.current[i] = el)} + tabIndex={i + 1} + type="text" + aria-label={`One time password input for the ${i + 1} digit`} + inputMode="numeric" + pattern="[0-9]*" + //Fix for ios chrome autocomplete + maxLength={i === 0 ? OTP_LENGTH : 1} + onFocus={() => setActiveElement(i)} + onPaste={handlePaste} + onChange={(e) => handleChange(e, i)} + onClick={() => handleClick(i)} + onInput={focusNextElement} + onKeyDown={handleKeydown} + key={i} + disabled={disabled || isVerified} + value={value[i]} + aria-invalid={!!errorText} + /> + ))} +
+ {errorText && ( +

{errorText}

+ )} +
+ ); +}; diff --git a/packages/react/src/components/strings.ts b/packages/react/src/components/strings.ts new file mode 100644 index 0000000000..b37469d59d --- /dev/null +++ b/packages/react/src/components/strings.ts @@ -0,0 +1,112 @@ +const STRINGS = { + "en-US": { + login: { + tosPrefix: "By signing in, you agree to the", + tosLink: "Terms of Service", + email: { + placeholder: "Email", + button: "Continue", + }, + passkey: { + button: "I have a passkey", + }, + or: "or", + }, + addPasskey: { + title: "Add a passkey", + continue: "Continue", + skip: "Skip", + simplerLoginTitle: "Simpler login", + simplerLoginDescription: + "Create a passkey to enable quick and easy login with Face ID or Touch ID.", + enhancedSecurityTitle: "Enhanced security", + enhancedSecurityDescription: + "Prevent phishing and theft by registering a passkey with your device.", + }, + loadingEmail: { + title: "Check your email", + verificationSent: "We sent a verification link to", + emailNotReceived: "Didn't receive the email?", + resend: "Resend", + resent: "Done!", + }, + loadingOtp: { + title: "Enter verification code", + body: "We sent a verification code to", + notReceived: "Didn't receive code?", + resend: "Resend", + verifying: "Verifying...", + verified: "Verified!", + }, + completingEmail: { + body: "Completing login. Please wait a few seconds for this to screen to update.", + }, + completingOtp: { + title: "Verifying...", + body: "It may take a moment to complete authentication.", + }, + loadingPasskey: { + title: "Continue with passkey", + body: "Follow the prompt to verify your passkey.", + supportText: "Having trouble?", + supportLink: "Contact support", + }, + protectedBy: { + title: "protected by", + }, + error: { + general: { + title: "Permission denied", + body: "The request is currently not allowed by the agent or the platform. Try again later.", + }, + connection: { + passkeyTitle: "Connection failed", + passkeyBody: + "Passkey request timed out or canceled by the agent. You may have to use another method to register a passkey for your account.", + oauthTitle: "Couldn't connect to ", + oauthBody: "The connection failed or canceled", + otpTitle: "Connection failed", + otpBody: "The code could not be verified", + walletTitle: "Couldn't connect to ", + walletBody: "The wallet’s connection failed or canceled", + timedOutTitle: "Connection timed out", + timedOutBody: "It looks like you need more time.", + }, + cta: { + tryAgain: "Try again", + useAnotherMethod: "Use another method", + skip: "Skip", + }, + customErrorMessages: { + eoa: { + walletConnect: { + chainIdNotFound: { + heading: "The connected wallet does not support this network", + body: "The wallet connection failed.", + tryAgainCTA: "Try again with a different wallet", + }, + walletConnectParamsNotFound: { + heading: "Couldn't connect to WalletConnect", + body: "The WalletConnect configuration is missing or incorrect.", + tryAgainCTA: "Try again", + }, + }, + default: { + heading: "Couldn't connect to ", + body: "The wallet's connection failed or was canceled.", + tryAgainCTA: "Try again", + }, + }, + }, + otp: { + invalid: "The code you entered is incorrect", + }, + }, + oauthContactSupport: { + title: "Need help?", + body: "Contact support", + }, + }, +}; + +export const ls = STRINGS["en-US"]; diff --git a/packages/react/src/hooks/internal/useIllustrationStyle.ts b/packages/react/src/hooks/internal/useIllustrationStyle.ts new file mode 100644 index 0000000000..49a6abaa6e --- /dev/null +++ b/packages/react/src/hooks/internal/useIllustrationStyle.ts @@ -0,0 +1,7 @@ +import { useUiConfig } from "../useUiConfig.js"; + +export function useIllustrationStyle() { + return useUiConfig(({ illustrationStyle }) => ({ + illustrationStyle: illustrationStyle ?? "outline", + })); +} diff --git a/packages/react/src/hooks/useAddPasskey.ts b/packages/react/src/hooks/useAddPasskey.ts new file mode 100644 index 0000000000..925a85a442 --- /dev/null +++ b/packages/react/src/hooks/useAddPasskey.ts @@ -0,0 +1,66 @@ +"use client"; + +import { useMutation } from "@tanstack/react-query"; +import type { UseMutationParameters, UseMutationReturnType } from "wagmi/query"; +import { useConfig } from "wagmi"; +import { + type AddPasskeyParameters, + type AddPasskeyReturnType, +} from "@alchemy/wagmi-core"; +import type { ConfigParameter } from "../types"; +import { + addPasskeyMutationOptions, + type AddPasskeyMutate, + type AddPasskeyMutateAsync, +} from "../query/addPasskey.js"; + +export type UseAddPasskeyParameters = ConfigParameter & { + mutation?: + | UseMutationParameters + | undefined; +}; + +export type UseAddPasskeyReturnType = UseMutationReturnType< + AddPasskeyReturnType, + Error, + AddPasskeyParameters +> & { + addPasskey: AddPasskeyMutate; + addPasskeyAsync: AddPasskeyMutateAsync; +}; + +/** + * React hook for adding a passkey to an already authenticated account. + * + * This hook uses the `addPasskey` mutation to add a passkey to the authenticated account. + * + * @param {UseAddPasskeyParameters} parameters - Configuration options for the hook + * @returns {UseAddPasskeyReturnType} TanStack Query mutation object + * + * @example + * ```tsx twoslash + * import { useAddPasskey } from "@alchemy/react"; + * + * function AddPasskeyForm() { + * const { addPasskey, isPending } = useAddPasskey(); + * } + * ``` + */ +export function useAddPasskey( + parameters: UseAddPasskeyParameters = {}, +): UseAddPasskeyReturnType { + const { mutation } = parameters; + const config = useConfig(parameters); + const mutationOptions = addPasskeyMutationOptions(config); + + const { mutate, mutateAsync, ...result } = useMutation({ + ...mutation, + ...mutationOptions, + }); + + return { + ...result, + addPasskey: mutate, + addPasskeyAsync: mutateAsync, + }; +} diff --git a/packages/react/src/hooks/useAuthConfig.ts b/packages/react/src/hooks/useAuthConfig.ts new file mode 100644 index 0000000000..33420a00b8 --- /dev/null +++ b/packages/react/src/hooks/useAuthConfig.ts @@ -0,0 +1,10 @@ +import { useUiConfig, type UiConfigStore } from "./useUiConfig.js"; + +// TODO: give this a selector param to prevent unnecessary re-renders +export function useAuthConfig(): NonNullable { + const { auth } = useUiConfig(); + if (!auth) { + throw new Error("Auth config should be present in UiConfig"); + } + return auth; +} diff --git a/packages/react/src/hooks/useAuthModal.ts b/packages/react/src/hooks/useAuthModal.ts new file mode 100644 index 0000000000..3e0a8c72c5 --- /dev/null +++ b/packages/react/src/hooks/useAuthModal.ts @@ -0,0 +1,42 @@ +import { useCallback } from "react"; +import { useUiConfig } from "./useUiConfig.js"; + +export type UseAuthModalResult = { + openAuthModal: () => void; + closeAuthModal: () => void; +}; + +/** + * A [hook](https://github.com/alchemyplatform/aa-sdk/blob/main/account-kit/react/src/hooks/useAuthModal.ts) that returns the open and close functions for the Auth Modal if uiConfig + * is enabled on the Account Provider + * + * @returns {UseAuthModalResult} an object containing methods for opening or closing the auth modal. [ref](https://github.com/alchemyplatform/aa-sdk/blob/main/account-kit/react/src/hooks/useAuthModal.ts#L4) + * + * @example + * ```tsx twoslash + * import React from 'react'; + * import { useAuthModal } from "@account-kit/react"; + * + * const ComponentWithAuthModal = () => { + * const { openAuthModal } = useAuthModal(); + * + * return ( + *
+ * + *
+ * ); + * }; + * ``` + */ +export const useAuthModal = () => { + const { isModalOpen, setModalOpen } = useUiConfig(); + + const openAuthModal = useCallback(() => setModalOpen(true), [setModalOpen]); + const closeAuthModal = useCallback(() => setModalOpen(false), [setModalOpen]); + + return { + isOpen: isModalOpen, + openAuthModal, + closeAuthModal, + }; +}; diff --git a/packages/react/src/hooks/useElementHeight.ts b/packages/react/src/hooks/useElementHeight.ts new file mode 100644 index 0000000000..b794d1a856 --- /dev/null +++ b/packages/react/src/hooks/useElementHeight.ts @@ -0,0 +1,27 @@ +"use client"; + +import { type MutableRefObject, useLayoutEffect, useState } from "react"; +import { useResizeObserver } from "./useResizeObserver.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export function useElementHeight( + target: MutableRefObject, +) { + const [height, setHeight] = useState(); + + useLayoutEffect(() => { + if (target.current) { + setHeight(target.current.getBoundingClientRect().height); + } + }, [target]); + + useResizeObserver({ + ref: target, + box: "border-box", + onResize: (size) => { + setHeight(size.height); + }, + }); + + return { height }; +} diff --git a/packages/react/src/hooks/useLoginWithPasskey.ts b/packages/react/src/hooks/useLoginWithPasskey.ts new file mode 100644 index 0000000000..562502ae50 --- /dev/null +++ b/packages/react/src/hooks/useLoginWithPasskey.ts @@ -0,0 +1,103 @@ +"use client"; + +import { useMutation } from "@tanstack/react-query"; +import type { UseMutationParameters, UseMutationReturnType } from "wagmi/query"; +import { useConfig } from "wagmi"; +import { + type LoginWithPasskeyParameters, + type LoginWithPasskeyReturnType, +} from "@alchemy/wagmi-core"; +import type { ConfigParameter } from "../types"; +import { + loginWithPasskeyMutationOptions, + type LoginWithPasskeyMutate, + type LoginWithPasskeyMutateAsync, +} from "../query/loginWithPasskey.js"; + +export type UseLoginWithPasskeyParameters = ConfigParameter & { + mutation?: + | UseMutationParameters< + LoginWithPasskeyReturnType, + Error, + LoginWithPasskeyParameters + > + | undefined; +}; + +export type UseLoginWithPasskeyReturnType = UseMutationReturnType< + LoginWithPasskeyReturnType, + Error, + LoginWithPasskeyParameters +> & { + loginWithPasskey: LoginWithPasskeyMutate; + loginWithPasskeyAsync: LoginWithPasskeyMutateAsync; +}; + +/** + * React hook for Passkey authentication - initiates authentication flow with the specified options. + * + * This hook wraps the `loginWithPasskey` action with React Query mutation functionality, + * providing loading states, error handling, and mutation management for the OAuth authentication flow. + * + * @param {UseLoginWithPasskeyParameters} parameters - Configuration options for the hook + * @param {Config} [parameters.config] - Optional wagmi config override + * @param {UseMutationParameters} [parameters.mutation] - Optional React Query mutation configuration + * @returns {UseLoginWithPasskeyReturnType} TanStack Query mutation object with the following properties: + * - `loginWithPasskey`: `(variables: LoginWithPasskeyParameters, options?) => void` - Mutation function to initiate OAuth login + * - `loginWithPasskeyAsync`: `(variables: LoginWithPasskeyParameters, options?) => Promise` - Async mutation function that returns a promise + * - `data`: `LoginWithPasskeyReturnType | undefined` - The last successfully resolved data (void) + * - `error`: `Error | null` - The error object for the mutation, if an error was encountered + * - `isError`: `boolean` - True if the mutation is in an error state + * - `isIdle`: `boolean` - True if the mutation is in its initial idle state + * - `isPending`: `boolean` - True if the mutation is currently executing + * - `isSuccess`: `boolean` - True if the last mutation attempt was successful + * - `failureCount`: `number` - The failure count for the mutation + * - `failureReason`: `Error | null` - The failure reason for the mutation retry + * - `isPaused`: `boolean` - True if the mutation has been paused + * - `reset`: `() => void` - Function to reset the mutation to its initial state + * - `status`: `'idle' | 'pending' | 'error' | 'success'` - Current status of the mutation + * - `submittedAt`: `number` - Timestamp for when the mutation was submitted + * - `variables`: `LoginWithPasskeyParameters | undefined` - The variables object passed to the mutation + * + * @example + * ```tsx twoslash + * import { useLoginWithPasskey } from '@alchemy/react'; + * + * function LoginForm() { + * const { loginWithPasskey, isPending, error } = useLoginWithPasskey(); + * + * const handleGoogleLogin = () => { + * loginWithPasskey({ + * provider: 'google', + * mode: 'popup' // or 'redirect' + * }); + * }; + * + * return ( + * + * ); + * } + * ``` + */ +export function useLoginWithPasskey( + parameters: UseLoginWithPasskeyParameters = {}, +): UseLoginWithPasskeyReturnType { + const { mutation } = parameters; + + const config = useConfig(parameters); + + const mutationOptions = loginWithPasskeyMutationOptions(config); + + const { mutate, mutateAsync, ...result } = useMutation({ + ...mutation, + ...mutationOptions, + }); + + return { + ...result, + loginWithPasskey: mutate, + loginWithPasskeyAsync: mutateAsync, + }; +} diff --git a/packages/react/src/hooks/useResizeObserver.ts b/packages/react/src/hooks/useResizeObserver.ts new file mode 100644 index 0000000000..b25ab2ed55 --- /dev/null +++ b/packages/react/src/hooks/useResizeObserver.ts @@ -0,0 +1,106 @@ +"use client"; + +import type { RefObject } from "react"; +import { useEffect, useRef, useState } from "react"; + +type Size = { + width: number | undefined; + height: number | undefined; +}; + +type UseResizeObserverOptions = { + ref: RefObject; + onResize?: (size: Size) => void; + box?: "border-box" | "content-box" | "device-pixel-content-box"; +}; + +const initialSize: Size = { + width: undefined, + height: undefined, +}; + +// See: https://usehooks-ts.com/react-hook/use-resize-observer +// eslint-disable-next-line jsdoc/require-jsdoc +export function useResizeObserver( + options: UseResizeObserverOptions, +): Size { + const { ref, box = "content-box" } = options; + const [{ width, height }, setSize] = useState(initialSize); + const previousSize = useRef({ ...initialSize }); + const onResize = useRef<((size: Size) => void) | undefined>(undefined); + onResize.current = options.onResize; + + const [isMounted, setIsMounted] = useState(false); + + useEffect(() => { + setIsMounted(true); + return () => setIsMounted(false); + }, []); + + useEffect(() => { + if (!ref.current) return; + + if (typeof window === "undefined" || !("ResizeObserver" in window)) return; + + const observer = new ResizeObserver(([entry]) => { + const boxProp = + box === "border-box" + ? "borderBoxSize" + : box === "device-pixel-content-box" + ? "devicePixelContentBoxSize" + : "contentBoxSize"; + + const newWidth = extractSize(entry, boxProp, "inlineSize"); + const newHeight = extractSize(entry, boxProp, "blockSize"); + + const hasChanged = + previousSize.current.width !== newWidth || + previousSize.current.height !== newHeight; + + if (hasChanged) { + const newSize: Size = { width: newWidth, height: newHeight }; + previousSize.current.width = newWidth; + previousSize.current.height = newHeight; + + if (onResize.current) { + onResize.current(newSize); + } else { + if (isMounted) { + setSize(newSize); + } + } + } + }); + + observer.observe(ref.current, { box }); + + return () => { + observer.disconnect(); + }; + }, [box, ref, isMounted]); + + return { width, height }; +} + +type BoxSizesKey = keyof Pick< + ResizeObserverEntry, + "borderBoxSize" | "contentBoxSize" | "devicePixelContentBoxSize" +>; + +function extractSize( + entry: ResizeObserverEntry, + box: BoxSizesKey, + sizeType: keyof ResizeObserverSize, +): number | undefined { + if (!entry[box]) { + if (box === "contentBoxSize") { + return entry.contentRect[sizeType === "inlineSize" ? "width" : "height"]; + } + return undefined; + } + + return Array.isArray(entry[box]) + ? entry[box][0][sizeType] + : // @ts-ignore Support Firefox's non-standard behavior + (entry[box][sizeType] as number); +} diff --git a/packages/react/src/hooks/useUiConfig.tsx b/packages/react/src/hooks/useUiConfig.tsx new file mode 100644 index 0000000000..036a9c6e2b --- /dev/null +++ b/packages/react/src/hooks/useUiConfig.tsx @@ -0,0 +1,79 @@ +"use client"; + +import { + createContext, + useContext, + useMemo, + useState, + type PropsWithChildren, +} from "react"; +import type { + AlchemyAccountsUIConfig, + AuthIllustrationStyle, +} from "../types.js"; + +type AlchemyAccountsUIConfigWithDefaults = Omit< + Required, + "auth" +> & { + auth: NonNullable>; +}; + +export type UiConfigStore = AlchemyAccountsUIConfig & { + isModalOpen: boolean; + setModalOpen: (isOpen: boolean) => void; +}; + +export const DEFAULT_UI_CONFIG: AlchemyAccountsUIConfigWithDefaults = { + illustrationStyle: "flat" as AuthIllustrationStyle, + auth: { + addPasskeyOnSignup: false, + header: null, + hideError: false, + sections: [[{ type: "email" }], [{ type: "passkey" }]], + onAuthSuccess: () => {}, + hideSignInText: false, + }, + modalBaseClassName: "", + supportUrl: "", + uiMode: "modal", +}; + +const UiConfigContext = createContext(undefined); + +// TODO: use the selector param to prevent unnecessary re-renders +export function useUiConfig( + selector?: (state: UiConfigStore) => T, +): T; + +export function useUiConfig(): UiConfigStore { + const store = useContext(UiConfigContext); + + if (!store) { + throw new Error("useUiConfig must be called within a UiConfigProvider"); + } + + return store; +} + +export function UiConfigProvider({ + children, + initialConfig, +}: PropsWithChildren<{ initialConfig?: AlchemyAccountsUIConfig }>) { + const [isModalOpen, setIsModalOpen] = useState(false); + + const value = useMemo( + () => ({ + ...initialConfig, + isModalOpen, + setModalOpen: setIsModalOpen, + }), + [initialConfig, isModalOpen], + ); + + return ( + + {children} + + ); +} diff --git a/packages/react/src/icons/EOAConnectionFailed.tsx b/packages/react/src/icons/EOAConnectionFailed.tsx new file mode 100644 index 0000000000..ace4625631 --- /dev/null +++ b/packages/react/src/icons/EOAConnectionFailed.tsx @@ -0,0 +1,60 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../hooks/internal/useIllustrationStyle.js"; + +const Ring = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + const isRingGrey = + illustrationStyle === "filled" || illustrationStyle === "flat"; + + return ( + + + + ); +}; + +const Cross = () => ( + + + + +); + +export const EOAConnectionFailed = { + Ring, + Cross, +}; diff --git a/packages/react/src/icons/alchemy.tsx b/packages/react/src/icons/alchemy.tsx new file mode 100644 index 0000000000..541d08d671 --- /dev/null +++ b/packages/react/src/icons/alchemy.tsx @@ -0,0 +1,57 @@ +import type { SVGProps } from "react"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const AlchemyLogo = ({ + fill = "currentColor", + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + + + + + +); diff --git a/packages/react/src/icons/auth-icons/apple.tsx b/packages/react/src/icons/auth-icons/apple.tsx new file mode 100644 index 0000000000..bb2e4e13c7 --- /dev/null +++ b/packages/react/src/icons/auth-icons/apple.tsx @@ -0,0 +1,21 @@ +import type { SVGProps } from "react"; + +export const AppleIcon = ( + props: JSX.IntrinsicAttributes & SVGProps, +) => { + return ( + + + + ); +}; diff --git a/packages/react/src/icons/auth-icons/discord.tsx b/packages/react/src/icons/auth-icons/discord.tsx new file mode 100644 index 0000000000..31482bbda9 --- /dev/null +++ b/packages/react/src/icons/auth-icons/discord.tsx @@ -0,0 +1,13 @@ +import type { SVGProps } from "react"; + +export const DiscordIcon = ({ + fill, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); diff --git a/packages/react/src/icons/auth-icons/facebook.tsx b/packages/react/src/icons/auth-icons/facebook.tsx new file mode 100644 index 0000000000..cb63a2a9df --- /dev/null +++ b/packages/react/src/icons/auth-icons/facebook.tsx @@ -0,0 +1,27 @@ +import type { SVGProps } from "react"; + +export const FacebookIcon = ( + props: JSX.IntrinsicAttributes & SVGProps, +) => { + return ( + + + + + + + ); +}; diff --git a/packages/react/src/icons/auth-icons/google.tsx b/packages/react/src/icons/auth-icons/google.tsx new file mode 100644 index 0000000000..259a38c7c9 --- /dev/null +++ b/packages/react/src/icons/auth-icons/google.tsx @@ -0,0 +1,39 @@ +import type { SVGProps } from "react"; + +export const GoogleIcon = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + return ( + + + + + + + ); +}; diff --git a/packages/react/src/icons/auth-icons/index.ts b/packages/react/src/icons/auth-icons/index.ts new file mode 100644 index 0000000000..5790138152 --- /dev/null +++ b/packages/react/src/icons/auth-icons/index.ts @@ -0,0 +1,7 @@ +import { DiscordIcon } from "./discord.js"; +import { FacebookIcon } from "./facebook.js"; +import { GoogleIcon } from "./google.js"; +import { AppleIcon } from "./apple.js"; +import { TwitchIcon } from "./twitch.js"; + +export { DiscordIcon, GoogleIcon, FacebookIcon, AppleIcon, TwitchIcon }; diff --git a/packages/react/src/icons/auth-icons/twitch.tsx b/packages/react/src/icons/auth-icons/twitch.tsx new file mode 100644 index 0000000000..cc1ed916a5 --- /dev/null +++ b/packages/react/src/icons/auth-icons/twitch.tsx @@ -0,0 +1,37 @@ +import type { SVGProps } from "react"; + +export const TwitchIcon = ( + props: JSX.IntrinsicAttributes & SVGProps, +) => { + return ( + + + + + + ); +}; diff --git a/packages/react/src/icons/chevron.tsx b/packages/react/src/icons/chevron.tsx new file mode 100644 index 0000000000..68a2dda07a --- /dev/null +++ b/packages/react/src/icons/chevron.tsx @@ -0,0 +1,23 @@ +import type { SVGProps } from "react"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const ChevronRight = ({ + fill = "currentColor", + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); diff --git a/packages/react/src/icons/coinbaseWallet.tsx b/packages/react/src/icons/coinbaseWallet.tsx new file mode 100644 index 0000000000..e896dd282c --- /dev/null +++ b/packages/react/src/icons/coinbaseWallet.tsx @@ -0,0 +1,38 @@ +export const CoinbaseWallet = ({ + className, + ...props +}: React.JSX.IntrinsicAttributes & React.SVGProps) => ( +
+ + + + + + + + + + + +
+); diff --git a/packages/react/src/icons/connectionFailed.tsx b/packages/react/src/icons/connectionFailed.tsx new file mode 100644 index 0000000000..d345a549b4 --- /dev/null +++ b/packages/react/src/icons/connectionFailed.tsx @@ -0,0 +1,144 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../hooks/internal/useIllustrationStyle.js"; + +export const ConnectionFailed = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + + )} + {illustrationStyle === "filled" && ( + + )} + {illustrationStyle === "linear" && ( + + )} + {illustrationStyle === "flat" && ( + + )} + + ); +}; + +const ConnectionFailedOutline = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); + +const ConnectionFailedFilled = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); +const ConnectionFailedLinear = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const ConnectionFailedFlat = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); diff --git a/packages/react/src/icons/illustrations/add-passkey.tsx b/packages/react/src/icons/illustrations/add-passkey.tsx new file mode 100644 index 0000000000..bbc635944a --- /dev/null +++ b/packages/react/src/icons/illustrations/add-passkey.tsx @@ -0,0 +1,319 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../../hooks/internal/useIllustrationStyle.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const AddPasskeyIllustration = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + <> + + + + )} + {illustrationStyle === "linear" && ( + <> + + + + )} + {illustrationStyle === "filled" && ( + <> + + + + )} + {illustrationStyle === "flat" && ( + <> + + + + )} + + ); +}; + +const AddPasskeyOutlineLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); + +const AddPasskeyLinearLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + +); + +const AddPasskeyFilledLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + +); + +const AddPasskeyFlatLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + +); + +const AddPasskeyOutlineDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); + +const AddPasskeyLinearDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + +); + +const AddPasskeyFilledDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + +); + +const AddPasskeyFlatDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + +); diff --git a/packages/react/src/icons/illustrations/added-passkey.tsx b/packages/react/src/icons/illustrations/added-passkey.tsx new file mode 100644 index 0000000000..3f8e027b7c --- /dev/null +++ b/packages/react/src/icons/illustrations/added-passkey.tsx @@ -0,0 +1,377 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../../hooks/internal/useIllustrationStyle.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const AddedPasskeyIllustration = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + <> + + + + )} + {illustrationStyle === "linear" && ( + <> + + + + )} + {illustrationStyle === "filled" && ( + <> + + + + )} + {illustrationStyle === "flat" && ( + <> + + + + )} + + ); +}; + +const AddedPasskeyOutlineLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const AddedPasskeyLinearLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + + +); + +const AddedPasskeyFilledLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const AddedPasskeyFlatLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + + +); + +const AddedPasskeyOutlineDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const AddedPasskeyLinearDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + + +); + +const AddedPasskeyFilledDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const AddedPasskeyFlatDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + + + +); diff --git a/packages/react/src/icons/illustrations/email.tsx b/packages/react/src/icons/illustrations/email.tsx new file mode 100644 index 0000000000..9ac1b09f3f --- /dev/null +++ b/packages/react/src/icons/illustrations/email.tsx @@ -0,0 +1,371 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../../hooks/internal/useIllustrationStyle.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const EmailIllustration = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + <> + + + + )} + {illustrationStyle === "linear" && ( + <> + + + + )} + {illustrationStyle === "filled" && ( + <> + + + + )} + {illustrationStyle === "flat" && ( + <> + + + + )} + + ); +}; + +const EmailOutlineLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + +); + +const EmailLinearLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const EmailFilledLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const EmailFlatLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const EmailOutlineDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + +); + +const EmailLinearDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const EmailFilledDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const EmailFlatDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); diff --git a/packages/react/src/icons/illustrations/passkey.tsx b/packages/react/src/icons/illustrations/passkey.tsx new file mode 100644 index 0000000000..afb8ffa001 --- /dev/null +++ b/packages/react/src/icons/illustrations/passkey.tsx @@ -0,0 +1,271 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../../hooks/internal/useIllustrationStyle.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const PasskeyIllustration = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + <> + + + + )} + {illustrationStyle === "linear" && ( + <> + + + + )} + {illustrationStyle === "filled" && ( + <> + + + + )} + {illustrationStyle === "flat" && ( + <> + + + + )} + + ); +}; + +const PasskeyOutlineLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyLinearLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const PasskeyFilledLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyFlatLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const PasskeyOutlineDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyLinearDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); + +const PasskeyFilledDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyFlatDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + + +); diff --git a/packages/react/src/icons/illustrations/passkeys.tsx b/packages/react/src/icons/illustrations/passkeys.tsx new file mode 100644 index 0000000000..b011623061 --- /dev/null +++ b/packages/react/src/icons/illustrations/passkeys.tsx @@ -0,0 +1,496 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../../hooks/internal/useIllustrationStyle.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const PasskeySmileyIllustration = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + <> + + + + )} + {illustrationStyle === "linear" && ( + <> + + + + )} + {illustrationStyle === "filled" && ( + <> + + + + )} + {illustrationStyle === "flat" && ( + <> + + + + )} + + ); +}; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const PasskeyShieldIllustration = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + <> + + + + )} + {illustrationStyle === "linear" && ( + <> + + + + )} + {illustrationStyle === "filled" && ( + <> + + + + )} + {illustrationStyle === "flat" && ( + <> + + + + )} + + ); +}; + +const PasskeySmileyOutlineLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeySmileyLinearLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeySmileyFilledLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeySmileyFlatLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const PasskeySmileyOutlineDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeySmileyLinearDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeySmileyFilledDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeySmileyFlatDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const PasskeyShieldOutlineLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyShieldLinearLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyShieldFilledLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyShieldFlatLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const PasskeyShieldOutlineDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyShieldLinearDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyShieldFilledDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const PasskeyShieldFlatDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); diff --git a/packages/react/src/icons/illustrations/success.tsx b/packages/react/src/icons/illustrations/success.tsx new file mode 100644 index 0000000000..938ad8daa8 --- /dev/null +++ b/packages/react/src/icons/illustrations/success.tsx @@ -0,0 +1,265 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../../hooks/internal/useIllustrationStyle.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const SuccessIllustration = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + <> + + + + )} + {illustrationStyle === "linear" && ( + <> + + + + )} + {illustrationStyle === "filled" && ( + <> + + + + )} + {illustrationStyle === "flat" && ( + <> + + + + )} + + ); +}; + +const SuccessOutlineLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); + +const SuccessLinearLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const SuccessFilledLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const SuccessFlatLight = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); + +const SuccessOutlineDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); + +const SuccessLinearDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const SuccessFilledDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +const SuccessFlatDark = ({ + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + +); diff --git a/packages/react/src/icons/mail.tsx b/packages/react/src/icons/mail.tsx new file mode 100644 index 0000000000..610f835b72 --- /dev/null +++ b/packages/react/src/icons/mail.tsx @@ -0,0 +1,23 @@ +import type { SVGProps } from "react"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const MailIcon = ({ + fill = "currentColor", + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); diff --git a/packages/react/src/icons/metamask.tsx b/packages/react/src/icons/metamask.tsx new file mode 100644 index 0000000000..6b0c8839c5 --- /dev/null +++ b/packages/react/src/icons/metamask.tsx @@ -0,0 +1,138 @@ +export const MetaMask = ({ + className, + ...props +}: React.JSX.IntrinsicAttributes & React.SVGProps) => { + return ( +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ ); +}; diff --git a/packages/react/src/icons/nav.tsx b/packages/react/src/icons/nav.tsx new file mode 100644 index 0000000000..d3257e3f77 --- /dev/null +++ b/packages/react/src/icons/nav.tsx @@ -0,0 +1,40 @@ +import type { SVGProps } from "react"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export const BackArrow = ({ + fill = "currentColor", + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); + +// eslint-disable-next-line jsdoc/require-jsdoc +export const X = ({ + fill = "currentColor", + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + +); diff --git a/packages/react/src/icons/oauth.tsx b/packages/react/src/icons/oauth.tsx new file mode 100644 index 0000000000..cc22c2d693 --- /dev/null +++ b/packages/react/src/icons/oauth.tsx @@ -0,0 +1,69 @@ +import type { KnownAuthProvider } from "@account-kit/signer"; +import { Spinner } from "./spinner.js"; +import { GoogleIcon, FacebookIcon, TwitchIcon } from "./auth-icons/index.js"; + +interface ContinueWithOAuthProps { + provider: KnownAuthProvider; +} + +interface OAuthConnectionFailedWithProps { + provider: KnownAuthProvider; + logoUrl?: string; + logoUrlDark?: string; + auth0Connection?: string; +} + +// TO DO: extend for BYO auth provider +export function ContinueWithOAuth({ provider }: ContinueWithOAuthProps) { + return ( +
+ + {(provider === "google" && ) || + (provider === "facebook" && ) || + (provider === "twitch" && )} +
+ ); +} + +// TO DO: extend for BYO auth provider +export function OAuthConnectionFailed({ + provider, + logoUrl, + logoUrlDark, + auth0Connection, +}: OAuthConnectionFailedWithProps) { + return ( +
+
+ + + + + +
+ {(provider === "google" && ) || + (provider === "facebook" && ) || + (provider === "auth0" && logoUrl && ( + <> + {auth0Connection} + {auth0Connection} + + ))} +
+ ); +} diff --git a/packages/react/src/icons/passkey.tsx b/packages/react/src/icons/passkey.tsx new file mode 100644 index 0000000000..d072813835 --- /dev/null +++ b/packages/react/src/icons/passkey.tsx @@ -0,0 +1,31 @@ +import type { SVGProps } from "react"; +import { Spinner } from "./spinner.js"; +import { PasskeyIllustration } from "./illustrations/passkey.js"; + +// eslint-disable-next-line jsdoc/require-jsdoc +export function LoadingPasskey() { + return ( +
+ + +
+ ); +} + +// eslint-disable-next-line jsdoc/require-jsdoc +export const PasskeyIcon = ({ + fill = "currentColor", + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); diff --git a/packages/react/src/icons/spinner.tsx b/packages/react/src/icons/spinner.tsx new file mode 100644 index 0000000000..abe51c2f8d --- /dev/null +++ b/packages/react/src/icons/spinner.tsx @@ -0,0 +1,24 @@ +// This spinner icon has to be in png format because svg does not support angular gradients +const SpinnerLightModeBase64 = + ""; + +const SpinnerDarkModeBase64 = + ""; + +// eslint-disable-next-line jsdoc/require-jsdoc +export function Spinner({ className }: { className?: string }) { + return ( + <> + Loading + Loading + + ); +} diff --git a/packages/react/src/icons/threeStars.tsx b/packages/react/src/icons/threeStars.tsx new file mode 100644 index 0000000000..7d855ea0b6 --- /dev/null +++ b/packages/react/src/icons/threeStars.tsx @@ -0,0 +1,30 @@ +export function ThreeStarsIcon({ className }: { className?: string }) { + return ( + + + + + + + + + + ); +} diff --git a/packages/react/src/icons/timeout.tsx b/packages/react/src/icons/timeout.tsx new file mode 100644 index 0000000000..33463cc728 --- /dev/null +++ b/packages/react/src/icons/timeout.tsx @@ -0,0 +1,181 @@ +import { type SVGProps } from "react"; +import { useIllustrationStyle } from "../hooks/internal/useIllustrationStyle.js"; + +export const Timeout = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => { + const { illustrationStyle } = useIllustrationStyle(); + + return ( + <> + {illustrationStyle === "outline" && ( + + )} + {illustrationStyle === "filled" && ( + + )} + {illustrationStyle === "linear" && ( + + )} + {illustrationStyle === "flat" && ( + + )} + + ); +}; +const TimedOutIconOutline = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); + +const TimedOutIconFilled = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); +const TimedOutIconLinear = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + + +); + +const TimedOutIconFlat = ({ + className, + ...props +}: JSX.IntrinsicAttributes & SVGProps) => ( + + + + + +); diff --git a/packages/react/src/icons/wallet.tsx b/packages/react/src/icons/wallet.tsx new file mode 100644 index 0000000000..fbf1c84c80 --- /dev/null +++ b/packages/react/src/icons/wallet.tsx @@ -0,0 +1,20 @@ +const SvgComponent = ({ + fill = "currentColor", + ...props +}: React.JSX.IntrinsicAttributes & React.SVGProps) => ( + + + +); +export { SvgComponent as WalletIcon }; diff --git a/packages/react/src/icons/walletConnect.tsx b/packages/react/src/icons/walletConnect.tsx new file mode 100644 index 0000000000..9e155e67a5 --- /dev/null +++ b/packages/react/src/icons/walletConnect.tsx @@ -0,0 +1,29 @@ +import * as React from "react"; + +const SvgComponent = ({ + className, + ...props +}: React.JSX.IntrinsicAttributes & React.SVGProps) => ( +
+ + + + +
+); +export { SvgComponent as WalletConnectIcon }; diff --git a/packages/react/src/icons/walletConnectIcon.tsx b/packages/react/src/icons/walletConnectIcon.tsx new file mode 100644 index 0000000000..cb0a134bb9 --- /dev/null +++ b/packages/react/src/icons/walletConnectIcon.tsx @@ -0,0 +1,22 @@ +export const WalletConnectIcon = ({ + className, + ...props +}: React.JSX.IntrinsicAttributes & React.SVGProps) => { + return ( +
+ + + +
+ ); +}; diff --git a/packages/react/src/icons/warning.tsx b/packages/react/src/icons/warning.tsx new file mode 100644 index 0000000000..65c5f8e2a4 --- /dev/null +++ b/packages/react/src/icons/warning.tsx @@ -0,0 +1,18 @@ +export const Warning = () => { + return ( + + + + ); +}; diff --git a/packages/react/src/index.ts b/packages/react/src/index.ts index 4fbd954f9b..6b424cb83a 100644 --- a/packages/react/src/index.ts +++ b/packages/react/src/index.ts @@ -10,12 +10,18 @@ export type * from "./hooks/useSubmitOtpCode.js"; export { useLoginWithOauth } from "./hooks/useLoginWithOauth.js"; export type * from "./hooks/useLoginWithOauth.js"; +export { useLoginWithPasskey } from "./hooks/useLoginWithPasskey.js"; +export type * from "./hooks/useLoginWithPasskey.js"; + export { useHandleOauthRedirect } from "./hooks/useHandleOauthRedirect.js"; export type * from "./hooks/useHandleOauthRedirect.js"; export { useSendVerificationCode } from "./hooks/useSendVerificationCode.js"; export type * from "./hooks/useSendVerificationCode.js"; +export { useAddPasskey } from "./hooks/useAddPasskey.js"; +export type * from "./hooks/useAddPasskey.js"; + export { useUpdateEmail } from "./hooks/useUpdateEmail.js"; export type * from "./hooks/useUpdateEmail.js"; @@ -42,3 +48,14 @@ export type * from "./hooks/useSendPreparedCalls.js"; export { useAuthMethods } from "./hooks/useAuthMethods.js"; export type * from "./hooks/useAuthMethods.js"; + +export { useAuthModal } from "./hooks/useAuthModal.js"; +export type * from "./hooks/useAuthModal.js"; + +export { AlchemyUiProvider } from "./components/AlchemyUiProvider.js"; +export type * from "./components/AlchemyUiProvider.js"; + +export { AuthModal } from "./components/auth/modal.js"; +export { AuthCard } from "./components/auth/card/auth-card.js"; + +export { type AlchemyAccountsUIConfig } from "./types.js"; diff --git a/packages/react/src/query/addPasskey.ts b/packages/react/src/query/addPasskey.ts new file mode 100644 index 0000000000..c5ae564576 --- /dev/null +++ b/packages/react/src/query/addPasskey.ts @@ -0,0 +1,34 @@ +import { + addPasskey, + type AddPasskeyParameters, + type AddPasskeyReturnType, +} from "@alchemy/wagmi-core"; +import type { MutateOptions, MutationOptions } from "@tanstack/react-query"; +import type { Config } from "wagmi"; + +export type AddPasskeyMutate = ( + variables: AddPasskeyParameters, + options?: + | MutateOptions + | undefined, +) => void; + +export type AddPasskeyMutateAsync = ( + variables: AddPasskeyParameters, + options?: + | MutateOptions + | undefined, +) => Promise; + +export function addPasskeyMutationOptions(config: Config) { + return { + mutationKey: ["addPasskey"], + mutationFn: (variables: AddPasskeyParameters) => { + return addPasskey(config, variables); + }, + } as const satisfies MutationOptions< + AddPasskeyReturnType, + Error, + AddPasskeyParameters + >; +} diff --git a/packages/react/src/query/loginWithPasskey.ts b/packages/react/src/query/loginWithPasskey.ts new file mode 100644 index 0000000000..877e244a87 --- /dev/null +++ b/packages/react/src/query/loginWithPasskey.ts @@ -0,0 +1,42 @@ +import { + loginWithPasskey, + type LoginWithPasskeyParameters, + type LoginWithPasskeyReturnType, +} from "@alchemy/wagmi-core"; +import type { MutateOptions, MutationOptions } from "@tanstack/react-query"; +import type { Config } from "wagmi"; + +export type LoginWithPasskeyMutate = ( + variables: LoginWithPasskeyParameters, + options?: + | MutateOptions< + LoginWithPasskeyReturnType, + Error, + LoginWithPasskeyParameters + > + | undefined, +) => void; + +export type LoginWithPasskeyMutateAsync = ( + variables: LoginWithPasskeyParameters, + options?: + | MutateOptions< + LoginWithPasskeyReturnType, + Error, + LoginWithPasskeyParameters + > + | undefined, +) => Promise; + +export function loginWithPasskeyMutationOptions(config: Config) { + return { + mutationKey: ["loginWithPasskey"], + mutationFn: (variables: LoginWithPasskeyParameters) => { + return loginWithPasskey(config, variables); + }, + } as const satisfies MutationOptions< + LoginWithPasskeyReturnType, + Error, + LoginWithPasskeyParameters + >; +} diff --git a/packages/react/src/types.ts b/packages/react/src/types.ts index 1ca07078af..7f69781d77 100644 --- a/packages/react/src/types.ts +++ b/packages/react/src/types.ts @@ -5,11 +5,57 @@ import type { UseQueryOptions, } from "@tanstack/react-query"; import type { Compute, ExactPartial, Omit } from "@wagmi/core/internal"; +import type { ReactNode } from "react"; +import type { AuthType } from "./components/auth/types"; export type ConfigParameter = { config?: AlchemyConfig | undefined; }; +export type AlchemyAccountsUIConfig = { + auth?: { + /** + * If this is true, then auth components will prompt users to add + * a passkey after signing in for the first time + */ + addPasskeyOnSignup?: boolean; + header?: ReactNode; + /** + * If hideError is true, then the auth component will not + * render the global error component + */ + hideError?: boolean; + onAuthSuccess?: () => void; + /** + * Each section can contain multiple auth types which will be grouped together + * and separated by an OR divider + */ + sections: AuthType[][]; + /** + * Whether to show the "Sign in" header text in the first auth step + */ + hideSignInText?: boolean; + }; + illustrationStyle?: "outline" | "linear" | "filled" | "flat" | undefined; + /** + * This class name will be applied to any modals that are rendered + */ + modalBaseClassName?: string; + /** + * This is the URL that will be used to link to the support page + */ + supportUrl?: string | undefined; + /** + * Set to "embedded" if the auth component will be rendered within a parent + * component in your UI. The default "modal" should be used if the auth component will be rendered in a modal overlay. + */ + uiMode?: "modal" | "embedded"; +}; + +export type AuthIllustrationStyle = NonNullable< + AlchemyAccountsUIConfig["illustrationStyle"] +>; + // From wagmi. export type QueryParameter< queryFnData = unknown, diff --git a/packages/react/src/utils.ts b/packages/react/src/utils.ts new file mode 100644 index 0000000000..2fa4a999a3 --- /dev/null +++ b/packages/react/src/utils.ts @@ -0,0 +1,10 @@ +export function capitalize(str: string) { + return str + .split(" ") + .map((word) => word.charAt(0).toUpperCase() + word.slice(1)) + .join(" "); +} + +export function assertNever(_: never, message: string): never { + throw new Error(message); +} diff --git a/packages/wagmi-core/src/actions/addPasskey.ts b/packages/wagmi-core/src/actions/addPasskey.ts new file mode 100644 index 0000000000..bc9dbb7ab4 --- /dev/null +++ b/packages/wagmi-core/src/actions/addPasskey.ts @@ -0,0 +1,28 @@ +import { type Config } from "@wagmi/core"; +import { resolveAlchemyAuthConnector } from "../utils/resolveAuthConnector.js"; +import { assertAuthSession } from "../utils/assertAuthSession.js"; +import type { CredentialCreationOptionOverrides } from "@alchemy/auth"; + +export type AddPasskeyParameters = + | CredentialCreationOptionOverrides + | undefined; + +export type AddPasskeyReturnType = void; + +/** + * Adds a passkey to the authenticated user's account. + * + * @param {Config} config - The shared Wagmi/Alchemy config + * @param {AddPasskeyParameters} parameters - The parameters for the passkey creation + * @returns {Promise} Promise that resolves when the passkey is added + */ +export async function addPasskey( + config: Config, + parameters: AddPasskeyParameters, +): Promise { + const { connector } = resolveAlchemyAuthConnector(config); + + const authSession = connector.getAuthSession(); + assertAuthSession(authSession, "addPasskey"); + await authSession.addPasskey(parameters); +} diff --git a/packages/wagmi-core/src/actions/loginWithPasskey.ts b/packages/wagmi-core/src/actions/loginWithPasskey.ts new file mode 100644 index 0000000000..73aea69fdc --- /dev/null +++ b/packages/wagmi-core/src/actions/loginWithPasskey.ts @@ -0,0 +1,26 @@ +import { type Config } from "@wagmi/core"; +import { resolveAlchemyAuthConnector } from "../utils/resolveAuthConnector.js"; +import type { LoginWithPasskeyParams } from "@alchemy/auth"; + +export type LoginWithPasskeyParameters = LoginWithPasskeyParams; +export type LoginWithPasskeyReturnType = void; + +/** + * Initiates Passkey authentication flow with the specified parameters. + * + * @param {Config} config - The shared Wagmi/Alchemy config + * @param {LoginWithPasskeyParameters} parameters - Passkey authentication parameters + * @returns {Promise} Promise that resolves when authentication completes and connection is established + */ +export async function loginWithPasskey( + config: Config, + parameters: LoginWithPasskeyParameters, +): Promise { + const { connector, connectAlchemyAuth } = resolveAlchemyAuthConnector(config); + const authClient = connector.getAuthClient(); + + const authSession = await authClient.loginWithPasskey(parameters); + + connector.setAuthSession(authSession); + await connectAlchemyAuth(); +} diff --git a/packages/wagmi-core/src/index.ts b/packages/wagmi-core/src/index.ts index c03774839e..c47cfe1500 100644 --- a/packages/wagmi-core/src/index.ts +++ b/packages/wagmi-core/src/index.ts @@ -22,6 +22,12 @@ export { type LoginWithOauthReturnType, } from "./actions/loginWithOauth.js"; +export { + loginWithPasskey, + type LoginWithPasskeyParameters, + type LoginWithPasskeyReturnType, +} from "./actions/loginWithPasskey.js"; + export { handleOauthRedirect, type HandleOauthRedirectParameters, @@ -34,6 +40,12 @@ export { type SendVerificationCodeReturnType, } from "./actions/sendVerificationCode.js"; +export { + addPasskey, + type AddPasskeyParameters, + type AddPasskeyReturnType, +} from "./actions/addPasskey.js"; + export { updateEmail, type UpdateEmailParameters, diff --git a/yarn.lock b/yarn.lock index 112d3dd96a..791fb86adf 100644 --- a/yarn.lock +++ b/yarn.lock @@ -9135,11 +9135,6 @@ resolved "https://registry.yarnpkg.com/@types/react-dom/-/react-dom-19.1.7.tgz#2863f2aa89e023592b981204ef92c5221b286410" integrity sha512-i5ZzwYpqjmrKenzkoLM2Ibzt6mAsM7pxB6BCIouEVVmgiqaMj1TjaK7hnA36hbW5aZv20kx7Lw6hWzPWg0Rurw== -"@types/react-dom@^19.1.9": - version "19.1.9" - resolved "https://registry.yarnpkg.com/@types/react-dom/-/react-dom-19.1.9.tgz#5ab695fce1e804184767932365ae6569c11b4b4b" - integrity sha512-qXRuZaOsAdXKFyOhRBg6Lqqc0yay13vN7KrIg4L7N4aaHN68ma9OK3NE1BoDFgFOTfM7zg+3/8+2n8rLUH3OKQ== - "@types/react-syntax-highlighter@^15.5.13": version "15.5.13" resolved "https://registry.yarnpkg.com/@types/react-syntax-highlighter/-/react-syntax-highlighter-15.5.13.tgz#c5baf62a3219b3bf28d39cfea55d0a49a263d1f2" @@ -9169,13 +9164,6 @@ dependencies: csstype "^3.0.2" -"@types/react@^19.1.13": - version "19.1.13" - resolved "https://registry.yarnpkg.com/@types/react/-/react-19.1.13.tgz#fc650ffa680d739a25a530f5d7ebe00cdd771883" - integrity sha512-hHkbU/eoO3EG5/MZkuFSKmYqPbSVk5byPFa3e7y/8TybHiLMACgI8seVYlicwk7H5K/rI2px9xrQp/C+AUDTiQ== - dependencies: - csstype "^3.0.2" - "@types/semver@^7.3.12": version "7.7.0" resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.7.0.tgz#64c441bdae033b378b6eef7d0c3d77c329b9378e"