SDK
React SDK
@authon/react — Authon用のReactコンポーネントとフック。Create React App、Vite、React 18+プロジェクトで動作。
npm: @authon/reactReact 18+TypeScript
インストール
bash
npm install @authon/reactAuthonProvider
アプリのルートを AuthonProvider. でラップ。すべてのフックとコンポーネントはこのプロバイダー内でレンダリングされる必要があります。
src/main.tsx
import { StrictMode } from "react";
import { createRoot } from "react-dom/client";
import { AuthonProvider } from "@authon/react";
import App from "./App";
createRoot(document.getElementById("root")!).render(
<StrictMode>
<AuthonProvider publishableKey="pk_live_your_publishable_key">
<App />
</AuthonProvider>
</StrictMode>
);プロバイダーProps
| Prop | 型 | 説明 |
|---|---|---|
| publishableKey | string | Your pk_live_ or pk_test_ key from the dashboard |
| config.apiUrl | string? | Override the Authon API base URL |
| config.theme | 'light' | 'dark' | 'auto' | Color theme for the modal (default: auto) |
| config.locale | string? | Locale string for the modal UI (default: en) |
| config.appearance | Partial<BrandingConfig>? | Override branding config programmatically |
フック
useAuthon()
認証状態を操作するメインフック。モーダルを開く、トークンを取得する、サインアウトするメソッドを含む完全なコンテキスト値を返します。
tsx
import { useAuthon } from "@authon/react";
function Header() {
const {
isSignedIn,
isLoading,
user,
openSignIn,
openSignUp,
signOut,
getToken,
} = useAuthon();
if (isLoading) return <Spinner />;
return (
<nav>
{isSignedIn ? (
<>
<span>Hello, {user?.displayName}</span>
<button onClick={() => signOut()}>Sign out</button>
</>
) : (
<button onClick={() => openSignIn()}>Sign in</button>
)}
</nav>
);
}useUser()
ユーザーデータとローディング状態のみを返す集中型フック。ユーザー情報の表示のみが必要なコンポーネントで使用。
tsx
import { useUser } from "@authon/react";
function ProfileCard() {
const { user, isLoading } = useUser();
if (isLoading) return <Skeleton />;
if (!user) return null;
return (
<div className="profile-card">
{user.avatarUrl && (
<img src={user.avatarUrl} alt={user.displayName ?? ""} />
)}
<h2>{user.displayName}</h2>
<p>{user.email}</p>
<span>Member since {new Date(user.createdAt).getFullYear()}</span>
</div>
);
}コンポーネント
SignedIn / SignedOut
認証状態に基づいて子コンポーネントを条件付きレンダリング。手動の条件なしに認証対応UIを構築する最も簡単な方法。
tsx
import { SignedIn, SignedOut } from "@authon/react";
function App() {
return (
<>
<SignedIn>
{/* Only rendered when user is signed in */}
<Dashboard />
<UserButton />
</SignedIn>
<SignedOut>
{/* Only rendered when user is NOT signed in */}
<LandingPage />
<SignInButton />
</SignedOut>
</>
);
}SignIn / SignUp
AuthonモーダルUIをトリガー。 popup モードではマウント時にモーダルが開きます。 embedded モードではフォームがインラインにレンダリングされます。
tsx
import { SignIn, SignUp } from "@authon/react";
// Popup — opens the modal when mounted
function LoginPage() {
return <SignIn mode="popup" />;
}
// Embedded — renders a form container in-place
function SignUpPage() {
return (
<div className="flex min-h-screen items-center justify-center">
<SignUp mode="embedded" />
</div>
);
}UserButton
サインアウトドロップダウン付きのドロップインアバターボタン。ユーザーアバターがあれば表示し、なければグラデーション背景のイニシャルにフォールバック。サインインしていない場合は null を返します。
tsx
import { UserButton } from "@authon/react";
function AppHeader() {
return (
<header className="flex items-center justify-between p-4">
<Logo />
<UserButton />
</header>
);
}保護されたルート
useAuthon()とReact Routerを組み合わせてプライベートルートを保護:
components/ProtectedRoute.tsx
import { Navigate } from "react-router-dom";
import { useAuthon } from "@authon/react";
interface ProtectedRouteProps {
children: React.ReactNode;
redirectTo?: string;
}
export function ProtectedRoute({
children,
redirectTo = "/sign-in",
}: ProtectedRouteProps) {
const { isSignedIn, isLoading } = useAuthon();
if (isLoading) return <FullPageSpinner />;
if (!isSignedIn) return <Navigate to={redirectTo} replace />;
return <>{children}</>;
}
// Usage in router
const router = createBrowserRouter([
{ path: "/", element: <Home /> },
{
path: "/dashboard",
element: (
<ProtectedRoute>
<Dashboard />
</ProtectedRoute>
),
},
]);トークンの使用
getToken() でバックエンドAPIコールにJWTを添付:
tsx
import { useAuthon } from "@authon/react";
function useApi() {
const { getToken } = useAuthon();
const apiFetch = async (path: string, init?: RequestInit) => {
const token = getToken();
return fetch(`/api${path}`, {
...init,
headers: {
...init?.headers,
Authorization: token ? `Bearer ${token}` : "",
"Content-Type": "application/json",
},
});
};
return { apiFetch };
}
// In a component
function OrdersList() {
const { apiFetch } = useApi();
const [orders, setOrders] = useState([]);
useEffect(() => {
apiFetch("/orders").then((r) => r.json()).then(setOrders);
}, []);
return <ul>{orders.map((o) => <li key={o.id}>{o.name}</li>)}</ul>;
}