122 lines
4.8 KiB
TypeScript
122 lines
4.8 KiB
TypeScript
"use client";
|
|
|
|
import Link from "next/link";
|
|
import { useSearchParams } from "next/navigation";
|
|
import { useEffect, useState, Suspense } from "react";
|
|
import { SiteFooter } from "../../components/site-footer";
|
|
import { SiteHeader } from "../../components/site-header";
|
|
|
|
type ApiResponse<T> = {
|
|
data: T;
|
|
meta: { timestamp: string; version: "v1" };
|
|
error: null | { message: string; code?: string };
|
|
};
|
|
|
|
function VerifyEmailContent() {
|
|
const searchParams = useSearchParams();
|
|
const token = searchParams.get("token");
|
|
const [status, setStatus] = useState<"loading" | "success" | "error">("loading");
|
|
const [message, setMessage] = useState("");
|
|
|
|
useEffect(() => {
|
|
if (!token) {
|
|
setStatus("error");
|
|
setMessage("No verification token provided.");
|
|
return;
|
|
}
|
|
fetch(`/api/auth/verify-email?token=${encodeURIComponent(token)}`)
|
|
.then((res) => res.json() as Promise<ApiResponse<{ message: string }>>)
|
|
.then((payload) => {
|
|
if (payload.error) {
|
|
setStatus("error");
|
|
setMessage(payload.error.message ?? "Verification failed.");
|
|
} else {
|
|
setStatus("success");
|
|
setMessage(payload.data?.message ?? "Email verified successfully.");
|
|
}
|
|
})
|
|
.catch(() => {
|
|
setStatus("error");
|
|
setMessage("Something went wrong. Please try again.");
|
|
});
|
|
}, [token]);
|
|
|
|
return (
|
|
<div className="text-center space-y-4">
|
|
{status === "loading" && (
|
|
<>
|
|
<div className="mx-auto h-12 w-12 rounded-full border-2 border-primary border-t-transparent animate-spin" />
|
|
<p className="text-sm text-muted-foreground">Verifying your email...</p>
|
|
</>
|
|
)}
|
|
{status === "success" && (
|
|
<>
|
|
<div className="mx-auto h-12 w-12 rounded-full bg-primary/10 flex items-center justify-center">
|
|
<svg className="h-6 w-6 text-primary" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 13l4 4L19 7" />
|
|
</svg>
|
|
</div>
|
|
<p className="text-lg font-semibold text-foreground">Email Verified!</p>
|
|
<p className="text-sm text-muted-foreground">{message}</p>
|
|
<Link
|
|
href="/login"
|
|
className="inline-flex justify-center rounded-lg bg-primary py-2 px-6 text-sm font-bold text-primary-foreground hover:bg-primary/90 transition-all"
|
|
>
|
|
Sign in
|
|
</Link>
|
|
</>
|
|
)}
|
|
{status === "error" && (
|
|
<>
|
|
<div className="mx-auto h-12 w-12 rounded-full bg-red-500/10 flex items-center justify-center">
|
|
<svg className="h-6 w-6 text-red-500" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
</div>
|
|
<p className="text-lg font-semibold text-foreground">Verification Failed</p>
|
|
<p className="text-sm text-muted-foreground">{message}</p>
|
|
<p className="text-xs text-muted-foreground">The link may have expired. Please register again or contact support.</p>
|
|
<Link
|
|
href="/login"
|
|
className="inline-flex justify-center rounded-lg bg-primary py-2 px-6 text-sm font-bold text-primary-foreground hover:bg-primary/90 transition-all"
|
|
>
|
|
Back to sign in
|
|
</Link>
|
|
</>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default function VerifyEmailPage() {
|
|
return (
|
|
<div className="min-h-screen bg-background font-sans text-foreground">
|
|
<SiteHeader />
|
|
<div className="flex min-h-full flex-col justify-center py-12 sm:px-6 lg:px-8 mt-16 relative">
|
|
<div className="absolute inset-0 mesh-gradient -z-10 opacity-30" />
|
|
<div className="sm:mx-auto sm:w-full sm:max-w-md">
|
|
<div className="flex justify-center">
|
|
<div
|
|
className="h-12 w-12 rounded-full flex items-center justify-center text-white font-bold text-xl shadow-[0_12px_30px_var(--gradient-glow)]"
|
|
style={{ background: "linear-gradient(to right, var(--gradient-start), var(--gradient-mid), var(--gradient-end))" }}
|
|
>
|
|
L1
|
|
</div>
|
|
</div>
|
|
<h2 className="mt-6 text-center text-3xl font-bold tracking-tight text-foreground">
|
|
Email Verification
|
|
</h2>
|
|
</div>
|
|
<div className="mt-8 sm:mx-auto sm:w-full sm:max-w-md">
|
|
<div className="glass-panel py-8 px-4 shadow-xl sm:rounded-xl sm:px-10">
|
|
<Suspense fallback={<div className="text-center text-sm text-muted-foreground">Loading...</div>}>
|
|
<VerifyEmailContent />
|
|
</Suspense>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<SiteFooter />
|
|
</div>
|
|
);
|
|
}
|