mirror of
https://github.com/pacnpal/thrilltrack-explorer.git
synced 2025-12-20 08:11:13 -05:00
198 lines
6.3 KiB
TypeScript
198 lines
6.3 KiB
TypeScript
'use client';
|
|
|
|
import { useState } from 'react';
|
|
import { useForm } from 'react-hook-form';
|
|
import { zodResolver } from '@hookform/resolvers/zod';
|
|
import { z } from 'zod';
|
|
import { useAuth } from '@/lib/contexts/AuthContext';
|
|
import { Button } from '@/components/ui/button';
|
|
import { Input } from '@/components/ui/input';
|
|
import { Label } from '@/components/ui/label';
|
|
import { Alert, AlertDescription } from '@/components/ui/alert';
|
|
import { AlertCircle, Loader2, CheckCircle2 } from 'lucide-react';
|
|
|
|
const registerSchema = z
|
|
.object({
|
|
email: z.string().email('Invalid email address'),
|
|
password: z
|
|
.string()
|
|
.min(8, 'Password must be at least 8 characters')
|
|
.regex(/[A-Z]/, 'Password must contain at least one uppercase letter')
|
|
.regex(/[a-z]/, 'Password must contain at least one lowercase letter')
|
|
.regex(/[0-9]/, 'Password must contain at least one number'),
|
|
confirmPassword: z.string(),
|
|
username: z
|
|
.string()
|
|
.min(3, 'Username must be at least 3 characters')
|
|
.max(30, 'Username must be at most 30 characters')
|
|
.regex(/^[a-zA-Z0-9_-]+$/, 'Username can only contain letters, numbers, hyphens, and underscores'),
|
|
})
|
|
.refine((data) => data.password === data.confirmPassword, {
|
|
message: "Passwords don't match",
|
|
path: ['confirmPassword'],
|
|
});
|
|
|
|
type RegisterFormData = z.infer<typeof registerSchema>;
|
|
|
|
interface RegisterFormProps {
|
|
onSuccess?: () => void;
|
|
onSwitchToLogin?: () => void;
|
|
}
|
|
|
|
export function RegisterForm({ onSuccess, onSwitchToLogin }: RegisterFormProps) {
|
|
const { register: registerUser } = useAuth();
|
|
const [error, setError] = useState<string>('');
|
|
const [isLoading, setIsLoading] = useState(false);
|
|
const [success, setSuccess] = useState(false);
|
|
|
|
const {
|
|
register,
|
|
handleSubmit,
|
|
formState: { errors },
|
|
watch,
|
|
} = useForm<RegisterFormData>({
|
|
resolver: zodResolver(registerSchema),
|
|
});
|
|
|
|
const password = watch('password', '');
|
|
|
|
const onSubmit = async (data: RegisterFormData) => {
|
|
setError('');
|
|
setIsLoading(true);
|
|
|
|
try {
|
|
await registerUser({
|
|
email: data.email,
|
|
password: data.password,
|
|
username: data.username,
|
|
});
|
|
setSuccess(true);
|
|
setTimeout(() => {
|
|
onSuccess?.();
|
|
}, 2000);
|
|
} catch (err: any) {
|
|
const errorMessage = err.response?.data?.detail || err.message || 'Registration failed';
|
|
setError(errorMessage);
|
|
} finally {
|
|
setIsLoading(false);
|
|
}
|
|
};
|
|
|
|
if (success) {
|
|
return (
|
|
<div className="text-center space-y-4">
|
|
<div className="mx-auto w-12 h-12 rounded-full bg-green-100 dark:bg-green-900 flex items-center justify-center">
|
|
<CheckCircle2 className="h-6 w-6 text-green-600 dark:text-green-400" />
|
|
</div>
|
|
<div>
|
|
<h3 className="text-lg font-semibold">Account Created!</h3>
|
|
<p className="text-sm text-muted-foreground mt-2">
|
|
Your account has been successfully created. Redirecting to dashboard...
|
|
</p>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<form onSubmit={handleSubmit(onSubmit)} className="space-y-4">
|
|
{error && (
|
|
<Alert variant="destructive">
|
|
<AlertCircle className="h-4 w-4" />
|
|
<AlertDescription>{error}</AlertDescription>
|
|
</Alert>
|
|
)}
|
|
|
|
<div className="space-y-2">
|
|
<Label htmlFor="username">Username</Label>
|
|
<Input
|
|
id="username"
|
|
type="text"
|
|
placeholder="johndoe"
|
|
{...register('username')}
|
|
disabled={isLoading}
|
|
/>
|
|
{errors.username && (
|
|
<p className="text-sm text-destructive">{errors.username.message}</p>
|
|
)}
|
|
</div>
|
|
|
|
<div className="space-y-2">
|
|
<Label htmlFor="email">Email</Label>
|
|
<Input
|
|
id="email"
|
|
type="email"
|
|
placeholder="you@example.com"
|
|
{...register('email')}
|
|
disabled={isLoading}
|
|
/>
|
|
{errors.email && (
|
|
<p className="text-sm text-destructive">{errors.email.message}</p>
|
|
)}
|
|
</div>
|
|
|
|
<div className="space-y-2">
|
|
<Label htmlFor="password">Password</Label>
|
|
<Input
|
|
id="password"
|
|
type="password"
|
|
placeholder="••••••••"
|
|
{...register('password')}
|
|
disabled={isLoading}
|
|
/>
|
|
{errors.password && (
|
|
<p className="text-sm text-destructive">{errors.password.message}</p>
|
|
)}
|
|
{password && (
|
|
<div className="space-y-1 text-xs">
|
|
<p className={password.length >= 8 ? 'text-green-600' : 'text-muted-foreground'}>
|
|
{password.length >= 8 ? '✓' : '○'} At least 8 characters
|
|
</p>
|
|
<p className={/[A-Z]/.test(password) ? 'text-green-600' : 'text-muted-foreground'}>
|
|
{/[A-Z]/.test(password) ? '✓' : '○'} One uppercase letter
|
|
</p>
|
|
<p className={/[a-z]/.test(password) ? 'text-green-600' : 'text-muted-foreground'}>
|
|
{/[a-z]/.test(password) ? '✓' : '○'} One lowercase letter
|
|
</p>
|
|
<p className={/[0-9]/.test(password) ? 'text-green-600' : 'text-muted-foreground'}>
|
|
{/[0-9]/.test(password) ? '✓' : '○'} One number
|
|
</p>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
<div className="space-y-2">
|
|
<Label htmlFor="confirmPassword">Confirm Password</Label>
|
|
<Input
|
|
id="confirmPassword"
|
|
type="password"
|
|
placeholder="••••••••"
|
|
{...register('confirmPassword')}
|
|
disabled={isLoading}
|
|
/>
|
|
{errors.confirmPassword && (
|
|
<p className="text-sm text-destructive">{errors.confirmPassword.message}</p>
|
|
)}
|
|
</div>
|
|
|
|
<Button type="submit" className="w-full" disabled={isLoading}>
|
|
{isLoading && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
|
|
Create Account
|
|
</Button>
|
|
|
|
{onSwitchToLogin && (
|
|
<div className="text-sm text-center">
|
|
<button
|
|
type="button"
|
|
onClick={onSwitchToLogin}
|
|
className="text-muted-foreground hover:text-foreground"
|
|
disabled={isLoading}
|
|
>
|
|
Already have an account? <span className="text-primary">Sign in</span>
|
|
</button>
|
|
</div>
|
|
)}
|
|
</form>
|
|
);
|
|
}
|