diff --git a/README.md b/README.md
index 1241ea0..01ab6af 100644
--- a/README.md
+++ b/README.md
@@ -11,6 +11,7 @@ The monorepo for Logto SDKs written in C#.
 - [src/Logto.AspNetCore.Authentication.Tests](./src/Logto.AspNetCore.Authentication.Tests): Tests for the ASP.NET Core authentication middleware.
 - [sample](./sample): Sample ASP.NET Core web application that shows how to use the ASP.NET Core authentication middleware.
 - [sample-mvc](./sample-mvc): Sample ASP.NET Core web MVC application that shows how to use the ASP.NET Core authentication middleware.
+- [sample-wasm](./sample-wasm): Sample Blazor WebAssembly application that shows how to use Blorc.OpenIdConnect to authenticate users with Logto.
 
 ## Resources
 
diff --git a/logto-csharp.sln b/logto-csharp.sln
index f959424..88b5a42 100644
--- a/logto-csharp.sln
+++ b/logto-csharp.sln
@@ -13,6 +13,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "logto-csharp-sample", "samp
 EndProject
 Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "sample-mvc", "sample-mvc\sample-mvc.csproj", "{2C4D9EC2-8697-4217-82E8-953835F990CA}"
 EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "sample-wasm", "sample-wasm\sample-wasm.csproj", "{A4D02F83-0AF6-4886-9DDC-D0E109780CD4}"
+EndProject
 Global
 	GlobalSection(SolutionConfigurationPlatforms) = preSolution
 		Debug|Any CPU = Debug|Any CPU
@@ -38,6 +40,10 @@ Global
 		{2C4D9EC2-8697-4217-82E8-953835F990CA}.Debug|Any CPU.Build.0 = Debug|Any CPU
 		{2C4D9EC2-8697-4217-82E8-953835F990CA}.Release|Any CPU.ActiveCfg = Release|Any CPU
 		{2C4D9EC2-8697-4217-82E8-953835F990CA}.Release|Any CPU.Build.0 = Release|Any CPU
+		{A4D02F83-0AF6-4886-9DDC-D0E109780CD4}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+		{A4D02F83-0AF6-4886-9DDC-D0E109780CD4}.Debug|Any CPU.Build.0 = Debug|Any CPU
+		{A4D02F83-0AF6-4886-9DDC-D0E109780CD4}.Release|Any CPU.ActiveCfg = Release|Any CPU
+		{A4D02F83-0AF6-4886-9DDC-D0E109780CD4}.Release|Any CPU.Build.0 = Release|Any CPU
 	EndGlobalSection
 	GlobalSection(NestedProjects) = preSolution
 		{D05B1B5F-D560-492A-A566-95888C8C5C43} = {D44D6C8F-1A29-4796-930F-B37ADB539EA3}
diff --git a/sample-wasm/App.razor b/sample-wasm/App.razor
new file mode 100644
index 0000000..e4461cf
--- /dev/null
+++ b/sample-wasm/App.razor
@@ -0,0 +1,14 @@
+@using sample_wasm.Pages
+
+<CascadingAuthenticationState>
+    <Router AppAssembly="@typeof(App).Assembly">
+        <Found Context="routeData">
+            <Home />
+            <FocusOnNavigate RouteData="@routeData" Selector="h1" />
+        </Found>
+        <NotFound>
+            <PageTitle>Not found</PageTitle>
+            <p role="alert">Sorry, there's nothing at this address.</p>
+        </NotFound>
+    </Router>
+</CascadingAuthenticationState>
diff --git a/sample-wasm/Pages/Home.razor b/sample-wasm/Pages/Home.razor
new file mode 100644
index 0000000..9b2b9a8
--- /dev/null
+++ b/sample-wasm/Pages/Home.razor
@@ -0,0 +1,34 @@
+@page "/"
+
+<section class="p-4 flex flex-col gap-2">
+    <h1 class="text-2xl font-bold">Logto Blazor WASM sample</h1>
+    <p>This is the sample application for Logto integration with Blazor WASM.</p>
+    <section class="space-y-2">
+        <AuthorizeView>
+            <Authorized>
+                <p class="text-emerald-700 text-s font-bold">
+                    You are signed in as @(@User?.Profile?.Name ?? "(unknown name)").
+                </p>
+                <h2 class="text-xl font-bold">Profile</h2>
+                <ul class="list-disc list-inside">
+                    <li>Email: @(@User?.Profile?.Email ?? "(null)")</li>
+                    <li>Email verified: @(@User?.Profile?.EmailVerified ?? false)</li>
+                </ul>
+                <p>Access token: @(@User?.AccessToken ?? "(null)")</p>
+                <button class="bg-violet-700 hover:bg-violet-800 text-white px-4 py-2 rounded text-sm"
+                    @onclick="OnLogoutButtonClickAsync">
+                    Sign out
+                </button>
+            </Authorized>
+            <NotAuthorized>
+                <p class="text-amber-600 text-s font-bold">
+                    You are not signed in.
+                </p>
+                <button class="bg-violet-700 hover:bg-violet-800 text-white px-4 py-2 rounded text-sm"
+                    @onclick="OnLoginButtonClickAsync">
+                    Sign in
+                </button>
+            </NotAuthorized>
+        </AuthorizeView>
+    </section>
+</section>
diff --git a/sample-wasm/Pages/Home.razor.cs b/sample-wasm/Pages/Home.razor.cs
new file mode 100644
index 0000000..dbd0638
--- /dev/null
+++ b/sample-wasm/Pages/Home.razor.cs
@@ -0,0 +1,56 @@
+namespace sample_wasm.Pages;
+using System.Threading.Tasks;
+using Microsoft.AspNetCore.Authorization;
+using Microsoft.AspNetCore.Components;
+using Microsoft.AspNetCore.Components.Web;
+using Blorc.OpenIdConnect;
+using Microsoft.AspNetCore.Components.Authorization;
+
+[Authorize]
+public partial class Home : ComponentBase
+{
+    [Inject]
+    public required IUserManager UserManager { get; set; }
+    public TimeSpan? SignOutTimeSpan { get; set; }
+
+    public User<Profile>? User { get; set; }
+
+    [CascadingParameter]
+    protected Task<AuthenticationState>? AuthenticationStateTask { get; set; }
+
+    protected override async Task OnInitializedAsync()
+    {
+        User = await UserManager.GetUserAsync<User<Profile>>(AuthenticationStateTask!);
+
+        UserManager.UserActivity += OnUserManagerUserActivity;
+        UserManager.UserInactivity += OnUserManagerUserInactivity;
+    }
+
+    private void OnUserManagerUserInactivity(object? sender, UserInactivityEventArgs args)
+    {
+        SignOutTimeSpan = args.SignOutTimeSpan;
+        StateHasChanged();
+    }
+
+    private void OnUserManagerUserActivity(object? sender, UserActivityEventArgs args)
+    {
+        SignOutTimeSpan = null;
+        StateHasChanged();
+    }
+
+    private async Task OnLoginButtonClickAsync(MouseEventArgs obj)
+    {
+        await UserManager.SignInRedirectAsync();
+    }
+
+    private async Task OnLogoutButtonClickAsync(MouseEventArgs obj)
+    {
+        await UserManager.SignOutRedirectAsync();
+    }
+
+    public void Dispose()
+    {
+        UserManager.UserActivity -= OnUserManagerUserActivity;
+        UserManager.UserInactivity -= OnUserManagerUserInactivity;
+    }
+}
diff --git a/sample-wasm/Program.cs b/sample-wasm/Program.cs
new file mode 100644
index 0000000..8b5a9f7
--- /dev/null
+++ b/sample-wasm/Program.cs
@@ -0,0 +1,30 @@
+using Microsoft.AspNetCore.Components.Web;
+using Microsoft.AspNetCore.Components.WebAssembly.Hosting;
+using Blorc.OpenIdConnect;
+using Blorc.Services;
+using sample_wasm;
+
+var builder = WebAssemblyHostBuilder.CreateDefault(args);
+builder.RootComponents.Add<App>("#app");
+builder.RootComponents.Add<HeadOutlet>("head::after");
+
+builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri(builder.HostEnvironment.BaseAddress) });
+
+builder.Services.AddBlorcCore();
+builder.Services.AddAuthorizationCore();
+builder.Services.AddBlorcOpenIdConnect(
+    options =>
+    {
+        builder.Configuration.Bind("IdentityServer", options);
+    });
+
+var webAssemblyHost = builder.Build();
+
+await webAssemblyHost
+    .ConfigureDocumentAsync(async documentService =>
+    {
+        await documentService.InjectBlorcCoreJsAsync();
+        await documentService.InjectOpenIdConnectAsync();
+    });
+
+await webAssemblyHost.RunAsync();
diff --git a/sample-wasm/Properties/launchSettings.json b/sample-wasm/Properties/launchSettings.json
new file mode 100644
index 0000000..522e588
--- /dev/null
+++ b/sample-wasm/Properties/launchSettings.json
@@ -0,0 +1,31 @@
+{
+  "$schema": "http://json.schemastore.org/launchsettings.json",
+  "iisSettings": {
+    "windowsAuthentication": false,
+    "anonymousAuthentication": true,
+    "iisExpress": {
+      "applicationUrl": "http://localhost:40255",
+      "sslPort": 44360
+    }
+  },
+  "profiles": {
+    "https": {
+      "commandName": "Project",
+      "dotnetRunMessages": true,
+      "launchBrowser": true,
+      "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}",
+      "applicationUrl": "https://localhost:7119;http://localhost:5025",
+      "environmentVariables": {
+        "ASPNETCORE_ENVIRONMENT": "Development"
+      }
+    },
+    "IIS Express": {
+      "commandName": "IISExpress",
+      "launchBrowser": true,
+      "inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}",
+      "environmentVariables": {
+        "ASPNETCORE_ENVIRONMENT": "Development"
+      }
+    }
+  }
+}
diff --git a/sample-wasm/README.md b/sample-wasm/README.md
new file mode 100644
index 0000000..5bb34ed
--- /dev/null
+++ b/sample-wasm/README.md
@@ -0,0 +1,75 @@
+# Logto ASP.NET Blazor WebAssembly sample project
+
+This sample project shows how to use the [Blorc.OpenIdConnect](https://github.com/WildGums/Blorc.OpenIdConnect) to authenticate users with Logto in a Blazor WebAssembly application.
+
+## Prerequisites
+
+- .NET 6.0 or higher
+- A [Logto Cloud](https://logto.io/) account or a self-hosted Logto
+- A Logto single-page application created
+
+### Optional
+
+- Set up an API resource in Logto
+
+If you don't have the Logto application created, please follow the [⚡ Get started](https://docs.logto.io/docs/tutorials/get-started/) guide to create one.
+
+## Configuration
+
+Create an `appsettings.Development.json` (or `appsettings.json`) with the following structure:
+
+```jsonc
+{
+  // ...
+  "IdentityServer": {
+    "Authority": "https://<your-logto-endpoint>/oidc",
+    "ClientId": "<your-logto-app-id>",
+    "PostLogoutRedirectUri": "<your-app-url>", // Remember to configure this in Logto
+    "RedirectUri": "<your-app-url>", // Remember to configure this in Logto
+    "ResponseType": "code",
+    "Scope": "openid profile" // Add more scopes if needed
+  }
+}
+```
+
+### Fetch user info
+
+For some special claims, such as `custom_data`, calling the `/userinfo` endpoint is required. To enable this feature, add the following configuration:
+
+```jsonc
+{
+  // ...
+  "IdentityServer": {
+    // ...
+    "LoadUserInfo": true
+  }
+}
+```
+
+> [!Caution]
+> Since WebAssembly is a client-side application, the token request will only be sent to the server-side once. Due to this nature, `LoadUserInfo` is conflict with fetching access token for API resources.
+
+### JWT access token
+
+If you need to fetch an access token in JWT format for an API resource, add the following configuration:
+
+```jsonc
+{
+  // ...
+  "IdentityServer": {
+    // ...
+    "Resource": "https://<your-api-resource-indicator>",
+    "ExtraTokenParams": {
+      "resource": "https://<your-api-resource-indicator>" // Ensure the key is lowercase
+    }
+  }
+}
+```
+
+The value of `Resource` and `ExtraTokenParams.resource` should be the same.
+
+## Run the sample
+
+```bash
+dotnet run # or `dotnet watch` to run in watch mode
+```
diff --git a/sample-wasm/_Imports.razor b/sample-wasm/_Imports.razor
new file mode 100644
index 0000000..ae4e992
--- /dev/null
+++ b/sample-wasm/_Imports.razor
@@ -0,0 +1,10 @@
+@using System.Net.Http
+@using System.Net.Http.Json
+@using Microsoft.AspNetCore.Components.Authorization
+@using Microsoft.AspNetCore.Components.Forms
+@using Microsoft.AspNetCore.Components.Routing
+@using Microsoft.AspNetCore.Components.Web
+@using Microsoft.AspNetCore.Components.Web.Virtualization
+@using Microsoft.AspNetCore.Components.WebAssembly.Http
+@using Microsoft.JSInterop
+@using sample_wasm
diff --git a/sample-wasm/sample-wasm.csproj b/sample-wasm/sample-wasm.csproj
new file mode 100644
index 0000000..b7544a1
--- /dev/null
+++ b/sample-wasm/sample-wasm.csproj
@@ -0,0 +1,15 @@
+<Project Sdk="Microsoft.NET.Sdk.BlazorWebAssembly">
+
+  <PropertyGroup>
+    <TargetFramework>net8.0</TargetFramework>
+    <Nullable>enable</Nullable>
+    <ImplicitUsings>enable</ImplicitUsings>
+  </PropertyGroup>
+
+  <ItemGroup>
+    <PackageReference Include="Blorc.OpenIdConnect" Version="1.9.0-beta0001" />
+    <PackageReference Include="Microsoft.AspNetCore.Components.WebAssembly" Version="8.0.1" />
+    <PackageReference Include="Microsoft.AspNetCore.Components.WebAssembly.DevServer" Version="8.0.1" PrivateAssets="all" />
+  </ItemGroup>
+
+</Project>
diff --git a/sample-wasm/wwwroot/css/app.css b/sample-wasm/wwwroot/css/app.css
new file mode 100644
index 0000000..5181190
--- /dev/null
+++ b/sample-wasm/wwwroot/css/app.css
@@ -0,0 +1,101 @@
+@import url('open-iconic/font/css/open-iconic-bootstrap.min.css');
+
+html, body {
+    font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif;
+}
+
+h1:focus {
+    outline: none;
+}
+
+a, .btn-link {
+    color: #0071c1;
+}
+
+.btn-primary {
+    color: #fff;
+    background-color: #1b6ec2;
+    border-color: #1861ac;
+}
+
+.btn:focus, .btn:active:focus, .btn-link.nav-link:focus, .form-control:focus, .form-check-input:focus {
+  box-shadow: 0 0 0 0.1rem white, 0 0 0 0.25rem #258cfb;
+}
+
+.content {
+    padding-top: 1.1rem;
+}
+
+.valid.modified:not([type=checkbox]) {
+    outline: 1px solid #26b050;
+}
+
+.invalid {
+    outline: 1px solid red;
+}
+
+.validation-message {
+    color: red;
+}
+
+#blazor-error-ui {
+    background: lightyellow;
+    bottom: 0;
+    box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2);
+    display: none;
+    left: 0;
+    padding: 0.6rem 1.25rem 0.7rem 1.25rem;
+    position: fixed;
+    width: 100%;
+    z-index: 1000;
+}
+
+    #blazor-error-ui .dismiss {
+        cursor: pointer;
+        position: absolute;
+        right: 0.75rem;
+        top: 0.5rem;
+    }
+
+.blazor-error-boundary {
+    background: url() no-repeat 1rem/1.8rem, #b32121;
+    padding: 1rem 1rem 1rem 3.7rem;
+    color: white;
+}
+
+    .blazor-error-boundary::after {
+        content: "An error has occurred."
+    }
+
+.loading-progress {
+    position: relative;
+    display: block;
+    width: 8rem;
+    height: 8rem;
+    margin: 20vh auto 1rem auto;
+}
+
+    .loading-progress circle {
+        fill: none;
+        stroke: #e0e0e0;
+        stroke-width: 0.6rem;
+        transform-origin: 50% 50%;
+        transform: rotate(-90deg);
+    }
+
+        .loading-progress circle:last-child {
+            stroke: #1b6ec2;
+            stroke-dasharray: calc(3.141 * var(--blazor-load-percentage, 0%) * 0.8), 500%;
+            transition: stroke-dasharray 0.05s ease-in-out;
+        }
+
+.loading-progress-text {
+    position: absolute;
+    text-align: center;
+    font-weight: bold;
+    inset: calc(20vh + 3.25rem) 0 auto 0.2rem;
+}
+
+    .loading-progress-text:after {
+        content: var(--blazor-load-percentage-text, "Loading");
+    }
diff --git a/sample-wasm/wwwroot/favicon.png b/sample-wasm/wwwroot/favicon.png
new file mode 100644
index 0000000..8422b59
Binary files /dev/null and b/sample-wasm/wwwroot/favicon.png differ
diff --git a/sample-wasm/wwwroot/icon-192.png b/sample-wasm/wwwroot/icon-192.png
new file mode 100644
index 0000000..166f56d
Binary files /dev/null and b/sample-wasm/wwwroot/icon-192.png differ
diff --git a/sample-wasm/wwwroot/index.html b/sample-wasm/wwwroot/index.html
new file mode 100644
index 0000000..35b2a17
--- /dev/null
+++ b/sample-wasm/wwwroot/index.html
@@ -0,0 +1,33 @@
+<!DOCTYPE html>
+<html lang="en">
+
+<head>
+    <meta charset="utf-8" />
+    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
+    <title>Logto Blazor WASM sample</title>
+    <base href="/" />
+    <link rel="stylesheet" href="css/app.css" />
+    <link rel="icon" type="image/png" href="favicon.png" />
+    <script src="https://cdn.tailwindcss.com"></script>
+</head>
+
+<body>
+    <div id="app">
+        <svg class="loading-progress">
+            <circle r="40%" cx="50%" cy="50%" />
+            <circle r="40%" cx="50%" cy="50%" />
+        </svg>
+        <div class="loading-progress-text"></div>
+    </div>
+
+    <div id="blazor-error-ui">
+        An unhandled error has occurred.
+        <a href="" class="reload">Reload</a>
+        <a class="dismiss">🗙</a>
+    </div>
+    <script src="_content/Microsoft.AspNetCore.Components.WebAssembly.Authentication/AuthenticationService.js"></script>
+    <script src="_framework/blazor.webassembly.js"></script>
+    <script src="_content/Blorc.Core/injector.js"></script>
+</body>
+
+</html>
diff --git a/src/Logto.AspNetCore.Authentication/Logto.AspNetCore.Authentication.csproj b/src/Logto.AspNetCore.Authentication/Logto.AspNetCore.Authentication.csproj
index 6ca40b7..cdf2455 100644
--- a/src/Logto.AspNetCore.Authentication/Logto.AspNetCore.Authentication.csproj
+++ b/src/Logto.AspNetCore.Authentication/Logto.AspNetCore.Authentication.csproj
@@ -1,46 +1,49 @@
-<Project Sdk="Microsoft.NET.Sdk">
-
-  <PropertyGroup>
-    <TargetFrameworks>net6.0;net7.0;net8.0</TargetFrameworks>
-    <Nullable>enable</Nullable>
-    <IsPackable>true</IsPackable>
-  </PropertyGroup>
-
-  <PropertyGroup>
-    <PackageId>Logto.AspNetCore.Authentication</PackageId>
-    <Version>0.1.1</Version>
-    <Description>Logto ASP.Net Core authentication SDK.</Description>
-    <Authors>Logto</Authors>
-    <Company>Silverhand, Inc.</Company>
-    <Copyright>Silverhand, Inc.</Copyright>
-    <PackageTags>logto;auth;authentication;identity;oauth2;openid-connect;oidc</PackageTags>
-    <PackageReadmeFile>README.md</PackageReadmeFile>
-    <PackageIcon>Logto.png</PackageIcon>
-    <PackageProjectUrl>https://logto.io/</PackageProjectUrl>
-    <PackageLicenseExpression>MIT</PackageLicenseExpression>
-    <RepositoryType>git</RepositoryType>
-    <RepositoryUrl>https://github.com/logto-io/csharp</RepositoryUrl>
-  </PropertyGroup>
-
-  <ItemGroup>
-    <FrameworkReference Include="Microsoft.AspNetCore.App" />
-  </ItemGroup>
-
-  <ItemGroup>
-    <PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect" Version="6.0.*"
-      Condition="'$(TargetFramework)' == 'net6.0'" />
-    <PackageReference Include="Microsoft.IdentityModel.Protocols.OpenIdConnect" Version="6.*"
-      Condition="'$(TargetFramework)' == 'net6.0'" />
-    <PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect" Version="7.0.*"
-      Condition="'$(TargetFramework)' == 'net7.0'" />
-    <PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect" Version="8.0.*"
-      Condition="'$(TargetFramework)' == 'net8.0'" />
-    <PackageReference Include="System.Text.Encodings.Web" Version="4.7.2" />
-  </ItemGroup>
-
-  <ItemGroup>
-    <None Include="README.md" Pack="true" PackagePath="" />
-    <None Include="Logto.png" Pack="true" PackagePath="" />
-  </ItemGroup>
-
-</Project>
+<Project Sdk="Microsoft.NET.Sdk">
+
+    <PropertyGroup>
+        <TargetFrameworks>net6.0;net7.0;net8.0</TargetFrameworks>
+        <Nullable>enable</Nullable>
+        <IsPackable>true</IsPackable>
+    </PropertyGroup>
+
+    <PropertyGroup>
+        <PackageId>Logto.AspNetCore.Authentication</PackageId>
+        <Version>0.1.1</Version>
+        <Description>Logto ASP.Net Core authentication SDK.</Description>
+        <Authors>Logto</Authors>
+        <Company>Silverhand, Inc.</Company>
+        <Copyright>Silverhand, Inc.</Copyright>
+        <PackageTags>logto;auth;authentication;identity;oauth2;openid-connect;oidc</PackageTags>
+        <PackageReadmeFile>README.md</PackageReadmeFile>
+        <PackageIcon>Logto.png</PackageIcon>
+        <PackageProjectUrl>https://logto.io/</PackageProjectUrl>
+        <PackageLicenseExpression>MIT</PackageLicenseExpression>
+        <RepositoryType>git</RepositoryType>
+        <RepositoryUrl>https://github.com/logto-io/csharp</RepositoryUrl>
+    </PropertyGroup>
+
+    <ItemGroup>
+        <FrameworkReference Include="Microsoft.AspNetCore.App" />
+    </ItemGroup>
+
+    <ItemGroup>
+        <PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect"
+            Version="6.0.*"
+            Condition="'$(TargetFramework)' == 'net6.0'" />
+        <PackageReference Include="Microsoft.IdentityModel.Protocols.OpenIdConnect" Version="6.*"
+            Condition="'$(TargetFramework)' == 'net6.0'" />
+        <PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect"
+            Version="7.0.*"
+            Condition="'$(TargetFramework)' == 'net7.0'" />
+        <PackageReference Include="Microsoft.AspNetCore.Authentication.OpenIdConnect"
+            Version="8.0.*"
+            Condition="'$(TargetFramework)' == 'net8.0'" />
+        <PackageReference Include="System.Text.Encodings.Web" Version="4.7.2" />
+    </ItemGroup>
+
+    <ItemGroup>
+        <None Include="README.md" Pack="true" PackagePath="" />
+        <None Include="Logto.png" Pack="true" PackagePath="" />
+    </ItemGroup>
+
+</Project>