kanban-app/frontend/src/pages/Register.tsx

187 lines
5.7 KiB
TypeScript
Raw Normal View History

2026-02-25 09:29:45 +00:00
import { useState, FormEvent, ChangeEvent } from 'react';
import { useNavigate, Link } from 'react-router-dom';
import { useApi } from '../hooks/useApi';
2026-02-24 08:52:57 +00:00
interface FormData {
2026-02-25 09:29:45 +00:00
email: string;
username: string;
password: string;
confirmPassword: string;
first_name: string;
last_name: string;
2026-02-24 08:52:57 +00:00
}
export function Register() {
const [formData, setFormData] = useState<FormData>({
email: '',
username: '',
password: '',
confirmPassword: '',
first_name: '',
last_name: '',
2026-02-25 09:29:45 +00:00
});
const [error, setError] = useState<string>('');
const [loading, setLoading] = useState<boolean>(false);
const navigate = useNavigate();
const { register } = useApi();
2026-02-24 08:52:57 +00:00
const handleChange = (e: ChangeEvent<HTMLInputElement>) => {
setFormData({
...formData,
[e.target.name]: e.target.value,
2026-02-25 09:29:45 +00:00
});
};
2026-02-24 08:52:57 +00:00
const handleSubmit = async (e: FormEvent<HTMLFormElement>) => {
2026-02-25 09:29:45 +00:00
e.preventDefault();
setError('');
2026-02-24 08:52:57 +00:00
if (formData.password !== formData.confirmPassword) {
2026-02-25 09:29:45 +00:00
setError('Passwords do not match');
return;
2026-02-24 08:52:57 +00:00
}
if (formData.password.length < 6) {
2026-02-25 09:29:45 +00:00
setError('Password must be at least 6 characters');
return;
2026-02-24 08:52:57 +00:00
}
2026-02-25 09:29:45 +00:00
setLoading(true);
2026-02-24 08:52:57 +00:00
try {
await register({
email: formData.email,
username: formData.username,
password: formData.password,
first_name: formData.first_name,
last_name: formData.last_name,
2026-02-25 09:29:45 +00:00
});
navigate('/login');
2026-02-24 08:52:57 +00:00
} catch (err: any) {
2026-02-25 09:29:45 +00:00
setError(err.response?.data?.error || 'Registration failed. Please try again.');
2026-02-24 08:52:57 +00:00
} finally {
2026-02-25 09:29:45 +00:00
setLoading(false);
2026-02-24 08:52:57 +00:00
}
2026-02-25 09:29:45 +00:00
};
2026-02-24 08:52:57 +00:00
return (
<div className="max-w-md mx-auto">
<h1 className="text-3xl font-bold text-white mb-8 text-center">Register</h1>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<form onSubmit={handleSubmit} className="space-y-6">
{error && (
<div className="bg-red-900 border border-red-700 text-red-100 px-4 py-3 rounded">
{error}
</div>
)}
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<div className="grid grid-cols-2 gap-4">
<div>
<label htmlFor="first_name" className="block text-sm font-medium text-gray-300 mb-2">
First Name
</label>
<input
id="first_name"
name="first_name"
type="text"
value={formData.first_name}
onChange={handleChange}
className="w-full px-4 py-2 bg-gray-800 border border-gray-700 rounded-lg text-white focus:ring-2 focus:ring-blue-500 focus:border-transparent"
/>
</div>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<div>
<label htmlFor="last_name" className="block text-sm font-medium text-gray-300 mb-2">
Last Name
</label>
<input
id="last_name"
name="last_name"
type="text"
value={formData.last_name}
onChange={handleChange}
className="w-full px-4 py-2 bg-gray-800 border border-gray-700 rounded-lg text-white focus:ring-2 focus:ring-blue-500 focus:border-transparent"
/>
</div>
</div>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<div>
<label htmlFor="username" className="block text-sm font-medium text-gray-300 mb-2">
Username
</label>
<input
id="username"
name="username"
type="text"
value={formData.username}
onChange={handleChange}
required
className="w-full px-4 py-2 bg-gray-800 border border-gray-700 rounded-lg text-white focus:ring-2 focus:ring-blue-500 focus:border-transparent"
/>
</div>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<div>
<label htmlFor="email" className="block text-sm font-medium text-gray-300 mb-2">
Email
</label>
<input
id="email"
name="email"
type="email"
value={formData.email}
onChange={handleChange}
required
className="w-full px-4 py-2 bg-gray-800 border border-gray-700 rounded-lg text-white focus:ring-2 focus:ring-blue-500 focus:border-transparent"
/>
</div>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<div>
<label htmlFor="password" className="block text-sm font-medium text-gray-300 mb-2">
Password
</label>
<input
id="password"
name="password"
type="password"
value={formData.password}
onChange={handleChange}
required
className="w-full px-4 py-2 bg-gray-800 border border-gray-700 rounded-lg text-white focus:ring-2 focus:ring-blue-500 focus:border-transparent"
/>
</div>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<div>
<label htmlFor="confirmPassword" className="block text-sm font-medium text-gray-300 mb-2">
Confirm Password
</label>
<input
id="confirmPassword"
name="confirmPassword"
type="password"
value={formData.confirmPassword}
onChange={handleChange}
required
className="w-full px-4 py-2 bg-gray-800 border border-gray-700 rounded-lg text-white focus:ring-2 focus:ring-blue-500 focus:border-transparent"
/>
</div>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<button
type="submit"
disabled={loading}
className="w-full bg-blue-600 hover:bg-blue-700 text-white font-medium py-2 px-4 rounded-lg transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
>
{loading ? 'Registering...' : 'Register'}
</button>
</form>
2026-02-25 09:29:45 +00:00
2026-02-24 08:52:57 +00:00
<p className="mt-6 text-center text-gray-400">
Already have an account?{' '}
<Link to="/login" className="text-blue-400 hover:text-blue-300">
Login
</Link>
</p>
</div>
2026-02-25 09:29:45 +00:00
);
}