188 lines
No EOL
6.2 KiB
TypeScript
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>
|
|
);
|
|
} |