Passkey Login Authentication

If a user has added a passkey to their account, or they initially signed up with a passkey, you can easily authenticate them using that passkey. This provides a secure, passwordless authentication experience.

You can implement Passkey Login authentication in two ways:

Pre-built UI Components

Account Kit provides pre-built UI components that handle the entire Passkey Login authentication flow with minimal code.

Step 1: Add Authentication Components to Your Page

Before configuring your authentication, first add one of the pre-built components to your application:

Using Modal Authentication

To add authentication in a modal popup:

1import React from "react";
2import { useAuthModal } from "@account-kit/react";
3
4export default function MyPage() {
5 const { openAuthModal } = useAuthModal();
6
7 return <button onClick={openAuthModal}>Sign in</button>;
8}

For more details on modal configuration, see the Modal Authentication documentation.

Or:

Using Embedded Authentication

To embed authentication directly in your page:

1import React from "react";
2import { AuthCard } from "@account-kit/react";
3
4export default function MyLoginPage() {
5 return (
6 <div className="flex flex-row p-4 bg-white border border-gray-200 rounded-lg">
7 <AuthCard />
8 </div>
9 );
10}

For more details on embedded authentication, see the Embedded Authentication documentation.

Step 2: Configure Passkey Login in UI Components

After adding the components, configure the Passkey Login authentication in your application config:

To customize the Passkey Login authentication experience in your pre-built components, configure the UI as follows:

1import { AlchemyAccountsUIConfig, createConfig } from "@account-kit/react";
2import { sepolia, alchemy } from "@account-kit/infra";
3
4const uiConfig: AlchemyAccountsUIConfig = {
5 auth: {
6 sections: [
7 [
8 // Include passkey login in a section
9 { type: "passkey" },
10
11 // You can combine with other authentication methods
12 { type: "email" },
13 ],
14 ],
15 },
16};
17
18export const config = createConfig(
19 {
20 transport: alchemy({ apiKey: "your-api-key" }),
21 chain: sepolia,
22 },
23 uiConfig
24);

Passkey login configuration accepts the following options:

1type PasskeyAuthType = {
2 type: "passkey";
3};

You can find the full type definition in the Account Kit source code.

For more details on UI component customization, see the UI Components documentation.

Custom UI

If you need complete control over the user experience, you can implement your own custom UI for Passkey Login authentication using Account Kit hooks.

Option 1: Passkey Login with Email

If the user’s passkey is associated with an email, you can use the email to help identify the correct passkey:

1import { useAuthenticate } from "@account-kit/react";
2
3// Inside your component
4const { authenticate } = useAuthenticate();
5
6// When the user wants to log in with their passkey and email
7const handlePasskeyLogin = (email: string) => {
8 authenticate(
9 {
10 type: "passkey",
11 email,
12 },
13 {
14 onSuccess: () => {
15 // Success - user authenticated with passkey
16 },
17 onError: (error) => {
18 // Handle error
19 },
20 }
21 );
22};

Option 2: Passkey Login without Email

If you want to authenticate a user with just their passkey (without requiring an email), you can use this approach:

1import { useAuthenticate } from "@account-kit/react";
2
3// Inside your component
4const { authenticate } = useAuthenticate();
5
6// When the user wants to log in with just their passkey
7const handlePasskeyOnlyLogin = () => {
8 authenticate(
9 {
10 type: "passkey",
11 createNew: false, // Important: set to false to prevent creating a new passkey
12 },
13 {
14 onSuccess: () => {
15 // Success - user authenticated with passkey
16 },
17 onError: (error) => {
18 // Handle error
19 },
20 }
21 );
22};

Step 3: Track Authentication Status

Use the useSignerStatus hook to determine if the user is authenticated:

1import { useSignerStatus } from "@account-kit/react";
2
3// Inside your component
4const { isConnected } = useSignerStatus();
5
6// You can use isConnected to conditionally render UI