haixunMaster/app/login/page.tsx

113 lines
4.0 KiB
TypeScript

"use client";
import Image from "next/image";
import { FormEvent, useState } from "react";
import { useRouter, useSearchParams } from "next/navigation";
import { Loader2, LogIn } from "lucide-react";
import { Button } from "@/components/ui/button";
import { Card, CardContent, CardHeader } from "@/components/ui/card";
import { Input } from "@/components/ui/input";
import { Label } from "@/components/ui/label";
import { BrandLogo, BrandMark } from "@/components/brand/logo";
import { ThemeToggle } from "@/components/theme-toggle";
import { BRAND_ASSETS } from "@/lib/brand";
import { notify } from "@/lib/notifications/store";
export default function LoginPage() {
const router = useRouter();
const searchParams = useSearchParams();
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const [loading, setLoading] = useState(false);
async function handleSubmit(event: FormEvent) {
event.preventDefault();
setLoading(true);
try {
const res = await fetch("/api/auth/login", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({ email, password }),
});
const data = await res.json();
if (!res.ok) {
notify({ type: "error", title: data.error ?? "登入失敗" });
setLoading(false);
return;
}
const next = searchParams.get("next");
if (data.needsThreadsBind) {
router.replace("/matrix?bind_threads=1");
return;
}
router.replace(next && next.startsWith("/") ? next : "/matrix");
} catch {
notify({ type: "error", title: "連線失敗,請稍後再試" });
setLoading(false);
}
}
return (
<div className="relative flex min-h-[100dvh] flex-col lg:min-h-screen">
<div className="absolute right-4 top-4 z-20 safe-bottom sm:right-5 sm:top-5">
<ThemeToggle compact />
</div>
<div className="relative h-36 shrink-0 sm:h-44 lg:absolute lg:inset-y-0 lg:left-0 lg:z-0 lg:h-full lg:w-[52%]">
<Image
src={BRAND_ASSETS.loginBg}
alt=""
fill
priority
className="object-cover object-center"
sizes="(max-width: 1024px) 100vw, 52vw"
/>
<div className="login-hero-overlay absolute inset-0" />
</div>
<div className="login-form-panel relative z-10 flex flex-1 items-center justify-center px-4 py-6 pb-8 safe-bottom sm:px-5 sm:py-10 lg:ml-[52%]">
<Card className="w-full max-w-md">
<CardHeader className="pb-4">
<div className="flex items-center gap-3">
<BrandLogo size="lg" />
<BrandMark />
</div>
</CardHeader>
<CardContent>
<form className="space-y-4" onSubmit={handleSubmit}>
<div className="space-y-2">
<Label htmlFor="email">Email</Label>
<Input
id="email"
type="email"
value={email}
onChange={(event) => setEmail(event.target.value)}
placeholder="you@example.com"
autoComplete="email"
required
/>
</div>
<div className="space-y-2">
<Label htmlFor="password"></Label>
<Input
id="password"
type="password"
value={password}
onChange={(event) => setPassword(event.target.value)}
placeholder="••••••••"
autoComplete="current-password"
required
/>
</div>
<Button className="w-full" type="submit" disabled={loading} aria-label="登入">
{loading ? <Loader2 className="h-4 w-4 animate-spin" /> : <LogIn className="h-4 w-4" />}
</Button>
</form>
</CardContent>
</Card>
</div>
</div>
);
}