pontus-front/app/signup/page.tsx

188 lines
No EOL
6.2 KiB
TypeScript

"use client"
import Altcha from "@/components/core/altcha";
import { Nav } from "@/components/core/nav";
import { TbUserPlus } from "react-icons/tb";
import { useState } from "react";
import { useForm, SubmitHandler } from "react-hook-form";
import { useRouter } from "next/navigation";
import { authClient } from "@/util/auth-client";
interface SignupForm {
email: string;
password: string;
confirmPassword: string;
name?: string;
}
export default function Signup() {
const router = useRouter();
const [altchaState, setAltchaState] = useState<{ status: "success" | "error" | "expired" | "waiting", token: string }>({ status: "waiting", token: "" });
const [isLoading, setIsLoading] = useState(false);
const [apiError, setApiError] = useState<string>("");
const {
register,
handleSubmit,
formState: { errors },
watch,
} = useForm<SignupForm>();
const password = watch("password");
const onSubmit: SubmitHandler<SignupForm> = async (data) => {
if (altchaState.status !== "success") {
setApiError("Please complete the captcha");
return;
}
setIsLoading(true);
setApiError("");
try {
const response = await fetch("/api/signup", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
...data,
token: altchaState.token,
}),
});
const result = await response.json();
if (!response.ok) {
setApiError(result.error || "Failed to create account");
return;
}
try {
await authClient.signIn.email({
email: data.email,
password: data.password,
});
router.push("/");
} catch (signInError) {
console.error("Auto-login failed:", signInError);
router.push("/login?message=Account created successfully. Please sign in.");
}
} catch (error) {
console.error("Signup error:", error);
setApiError("An unexpected error occurred. Please try again.");
} finally {
setIsLoading(false);
}
};
const handleAltchaStateChange = (e: Event | CustomEvent) => {
if ('detail' in e && e.detail?.payload) {
setAltchaState({ status: "success", token: e.detail.payload });
} else {
setAltchaState({ status: "error", token: "" });
}
};
return (
<main>
<Nav />
<div className="flex flex-col items-center justify-center mt-18 sm:my-16 px-4 gap-18">
<div className="flex flex-row items-center justify-between gap-2">
<TbUserPlus size={32} className="sm:w-9 sm:h-9" />
<h1 className="text-3xl sm:text-4xl font-bold">
signup
</h1>
</div>
<form className="flex flex-col bg-foreground/10 rounded-2xl sm:rounded-4xl p-4 gap-4 w-1/4 min-w-80" onSubmit={handleSubmit(onSubmit)}>
<h2 className="text-2xl sm:text-3xl font-light text-center w-full flex flex-wrap">
name (optional)
</h2>
<input
type="text"
placeholder="enter your name"
className="w-full p-2 rounded-md bg-foreground/10"
{...register("name")}
/>
<h2 className="text-2xl sm:text-3xl font-light text-center w-full flex flex-wrap">
email
</h2>
<input
type="email"
placeholder="enter your email"
className="w-full p-2 rounded-md bg-foreground/10"
{...register("email", {
required: "Email is required",
pattern: {
value: /^[^\s@]+@[^\s@]+\.[^\s@]+$/,
message: "Please enter a valid email address"
}
})}
/>
{errors.email && <p className="text-red-500 text-sm">{errors.email.message}</p>}
<h2 className="text-2xl sm:text-3xl font-light text-center w-full flex flex-wrap">
password
</h2>
<input
type="password"
placeholder="enter your password"
className="w-full p-2 rounded-md bg-foreground/10"
{...register("password", {
required: "Password is required",
minLength: {
value: 8,
message: "Password must be at least 8 characters long"
},
pattern: {
value: /^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)/,
message: "Password must contain at least one uppercase letter, one lowercase letter, and one number"
}
})}
/>
{errors.password && <p className="text-red-500 text-sm">{errors.password.message}</p>}
<h2 className="text-2xl sm:text-3xl font-light text-center w-full flex flex-wrap">
confirm password
</h2>
<input
type="password"
placeholder="confirm your password"
className="w-full p-2 rounded-md bg-foreground/10"
{...register("confirmPassword", {
required: "Please confirm your password",
validate: value => value === password || "Passwords do not match"
})}
/>
{errors.confirmPassword && <p className="text-red-500 text-sm">{errors.confirmPassword.message}</p>}
<h2 className="text-2xl sm:text-3xl font-light text-center w-full flex flex-wrap">
captcha
</h2>
<Altcha onStateChange={handleAltchaStateChange} />
{apiError && (
<div className="p-3 bg-red-100 border border-red-400 text-red-700 rounded-md">
{apiError}
</div>
)}
<button
className="bg-blue-400 text-white px-4 py-2 rounded-md disabled:opacity-50 disabled:cursor-not-allowed"
type="submit"
disabled={isLoading || altchaState.status !== "success" || !altchaState.token}
>
{isLoading ? "Creating account..." : altchaState.status === "success" ? "signup" : "waiting for captcha"}
</button>
<p className="text-center text-sm text-gray-600 mt-4">
Already have an account?{" "}
<a href="/login" className="text-blue-400 hover:underline">
Sign in here
</a>
</p>
</form>
</div>
</main>
);
}