文件
csp/frontend/src/app/auth/page.tsx

218 行
8.8 KiB
TypeScript
原始文件 Blame 文件历史

此文件含有模棱两可的 Unicode 字符
此文件含有可能会与其他字符混淆的 Unicode 字符。 如果您是想特意这样的,可以安全地忽略该警告。 使用 Escape 按钮显示他们。
"use client";
import Link from "next/link";
import { useEffect, useMemo, useState } from "react";
import { useRouter } from "next/navigation";
import { API_BASE, apiFetch } from "@/lib/api";
import { readToken, saveToken } from "@/lib/auth";
import { useI18nText } from "@/lib/i18n";
type AuthOk = { ok: true; user_id: number; token: string; expires_at: number };
type AuthErr = { ok: false; error: string };
type AuthResp = AuthOk | AuthErr;
function passwordScore(password: string, isZh: boolean): { label: string; color: string } {
if (password.length >= 12) return { label: isZh ? "强" : "Strong", color: "text-emerald-600" };
if (password.length >= 8) return { label: isZh ? "中" : "Medium", color: "text-blue-600" };
return { label: isZh ? "弱" : "Weak", color: "text-orange-600" };
}
export default function AuthPage() {
const { isZh, tx } = useI18nText();
const router = useRouter();
const apiBase = useMemo(() => API_BASE, []);
const [checkingAuth, setCheckingAuth] = useState(true);
const [mode, setMode] = useState<"register" | "login">("login");
const [username, setUsername] = useState(process.env.NEXT_PUBLIC_TEST_USERNAME ?? "");
const [password, setPassword] = useState(process.env.NEXT_PUBLIC_TEST_PASSWORD ?? "");
const [confirmPassword, setConfirmPassword] = useState("");
const [showPassword, setShowPassword] = useState(false);
const [loading, setLoading] = useState(false);
const [resp, setResp] = useState<AuthResp | null>(null);
useEffect(() => {
if (readToken()) {
router.replace("/problems");
return;
}
setCheckingAuth(false);
}, [router]);
const usernameErr = username.trim().length < 3 ? tx("用户名至少 3 位", "Username must be at least 3 chars") : "";
const passwordErr = password.length < 6 ? tx("密码至少 6 位", "Password must be at least 6 chars") : "";
const confirmErr =
mode === "register" && password !== confirmPassword ? tx("两次密码不一致", "Passwords do not match") : "";
const canSubmit = !loading && !usernameErr && !passwordErr && !confirmErr;
async function submit() {
if (!canSubmit) return;
setLoading(true);
setResp(null);
try {
const j = await apiFetch<AuthResp>(`/api/v1/auth/${mode}`, {
method: "POST",
body: JSON.stringify({ username: username.trim(), password }),
});
setResp(j);
if (j.ok) {
saveToken(j.token);
setTimeout(() => {
router.replace("/problems");
}, 350);
}
} catch (e: unknown) {
setResp({ ok: false, error: String(e) });
} finally {
setLoading(false);
}
}
const strength = passwordScore(password, isZh);
if (checkingAuth) {
return (
<main className="mx-auto max-w-4xl px-3 py-12 text-sm text-zinc-500">
{tx("已登录,正在跳转...", "Already signed in, redirecting...")}
</main>
);
}
return (
<main className="mx-auto max-w-4xl px-3 py-6 max-[390px]:px-2 sm:px-4 md:px-6 md:py-10">
<div className="grid gap-6 md:grid-cols-[1.1fr,1fr]">
<section className="rounded-2xl border bg-zinc-900 p-6 text-zinc-100">
<h1 className="text-2xl font-semibold">{tx("欢迎回来,开始刷题", "Welcome back, let's practice")}</h1>
<p className="mt-3 text-sm text-zinc-300">
{tx("登录后可提交评测、保存草稿、查看错题本和个人进度。", "After sign-in you can submit, save drafts, review wrong-book, and track your progress.")}
</p>
<div className="mt-6 space-y-2 text-sm text-zinc-300">
<p>{tx("• 题库按 CSP-J / CSP-S / NOIP 入门组织", "• Problem sets are organized by CSP-J / CSP-S / NOIP junior")}</p>
<p>{tx("• 题目页支持本地草稿与试运行", "• Problem page supports local draft and run")}</p>
<p>{tx("• 生成式题解会异步入库,支持多解法", "• Generated solutions are queued asynchronously with multiple methods")}</p>
</div>
<p className="mt-6 text-xs text-zinc-400">
API Base: <span className="font-mono">{apiBase}</span>
</p>
</section>
<section className="rounded-2xl border bg-white p-6">
<div className="grid grid-cols-2 gap-2 rounded-lg bg-zinc-100 p-1 text-sm">
<button
type="button"
className={`rounded-md px-3 py-2 ${
mode === "login" ? "bg-white shadow-sm" : "text-zinc-600"
}`}
onClick={() => {
setMode("login");
setResp(null);
}}
disabled={loading}
>
{tx("登录", "Sign In")}
</button>
<button
type="button"
className={`rounded-md px-3 py-2 ${
mode === "register" ? "bg-white shadow-sm" : "text-zinc-600"
}`}
onClick={() => {
setMode("register");
setResp(null);
}}
disabled={loading}
>
{tx("注册", "Register")}
</button>
</div>
<div className="mt-5 space-y-4">
<div>
<label className="text-sm font-medium">{tx("用户名", "Username")}</label>
<input
className="mt-1 w-full rounded-lg border px-3 py-2"
value={username}
onChange={(e) => setUsername(e.target.value)}
placeholder={tx("例如csp_student", "e.g. csp_student")}
/>
{usernameErr && <p className="mt-1 text-xs text-red-600">{usernameErr}</p>}
</div>
<div>
<div className="flex items-center justify-between">
<label className="text-sm font-medium">{tx("密码", "Password")}</label>
<span className={`text-xs ${strength.color}`}>{tx("强度", "Strength")}: {strength.label}</span>
</div>
<input
type={showPassword ? "text" : "password"}
className="mt-1 w-full rounded-lg border px-3 py-2"
value={password}
onChange={(e) => setPassword(e.target.value)}
placeholder={tx("至少 6 位", "At least 6 chars")}
/>
{passwordErr && <p className="mt-1 text-xs text-red-600">{passwordErr}</p>}
</div>
{mode === "register" && (
<div>
<label className="text-sm font-medium">{tx("确认密码", "Confirm Password")}</label>
<input
type={showPassword ? "text" : "password"}
className="mt-1 w-full rounded-lg border px-3 py-2"
value={confirmPassword}
onChange={(e) => setConfirmPassword(e.target.value)}
placeholder={tx("再输入一次密码", "Enter password again")}
/>
{confirmErr && <p className="mt-1 text-xs text-red-600">{confirmErr}</p>}
</div>
)}
<label className="flex items-center gap-2 text-xs text-zinc-600">
<input
type="checkbox"
checked={showPassword}
onChange={(e) => setShowPassword(e.target.checked)}
/>
{tx("显示密码", "Show password")}
</label>
<button
className="w-full rounded-lg bg-zinc-900 px-4 py-2 text-white hover:bg-zinc-800 disabled:opacity-50"
onClick={() => void submit()}
disabled={!canSubmit}
>
{loading ? tx("提交中...", "Submitting...") : mode === "register" ? tx("注册并登录", "Register & Sign In") : tx("登录", "Sign In")}
</button>
</div>
{resp && (
<div
className={`mt-4 rounded-lg border px-3 py-2 text-sm ${
resp.ok ? "border-emerald-300 bg-emerald-50 text-emerald-700" : "border-red-300 bg-red-50 text-red-700"
}`}
>
{resp.ok
? tx("登录成功,正在跳转到题库...", "Signed in. Redirecting to problem set...")
: `${tx("操作失败:", "Action failed: ")}${resp.error}`}
</div>
)}
<p className="mt-4 text-xs text-zinc-500">
{tx("登录后 Token 自动保存在浏览器 localStorage,可直接前往", "Token is stored in browser localStorage after sign-in. You can go to")}
<Link className="mx-1 underline" href="/problems">
{tx("题库", "Problems")}
</Link>
{tx("与", "and")}
<Link className="mx-1 underline" href="/me">
{tx("我的", "My Account")}
</Link>
{tx("页面。", ".")}
</p>
</section>
</div>
</main>
);
}