124 lines
4.5 KiB
TypeScript
124 lines
4.5 KiB
TypeScript
import React, { useState } from 'react';
|
|
import apiClient from '../../api/client';
|
|
import { Activity } from 'lucide-react';
|
|
|
|
interface AuthPageProps {
|
|
onLoginSuccess: () => void;
|
|
}
|
|
|
|
export function AuthPage({ onLoginSuccess }: AuthPageProps) {
|
|
const [isLogin, setIsLogin] = useState(true);
|
|
const [userName, setUserName] = useState('');
|
|
const [password, setPassword] = useState('');
|
|
const [error, setError] = useState('');
|
|
const [loading, setLoading] = useState(false);
|
|
|
|
const handleSubmit = async (e: React.FormEvent) => {
|
|
e.preventDefault();
|
|
setError('');
|
|
setLoading(true);
|
|
|
|
try {
|
|
if (isLogin) {
|
|
// Login
|
|
const response = await apiClient.post('/api/v1/auth/login', {
|
|
user_name: userName,
|
|
password: password,
|
|
});
|
|
|
|
if (response.data.token) {
|
|
localStorage.setItem('token', response.data.token);
|
|
onLoginSuccess();
|
|
}
|
|
} else {
|
|
// Register
|
|
const response = await apiClient.post('/api/v1/auth/register', {
|
|
user_name: userName,
|
|
password: password,
|
|
});
|
|
|
|
// After successful register, we can automatically log them in
|
|
// or just show a message and switch to login. Let's switch to login.
|
|
if (response.data.message === 'success') {
|
|
setIsLogin(true);
|
|
setError('Registration successful. Please log in.');
|
|
}
|
|
}
|
|
} catch (err: any) {
|
|
console.error(err);
|
|
setError(err.response?.data?.detail || err.response?.data?.message || 'Authentication failed');
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className="flex min-h-screen w-full items-center justify-center bg-slate-50">
|
|
<div className="w-full max-w-md bg-white rounded-xl shadow-lg p-8 border border-slate-100">
|
|
<div className="flex flex-col items-center mb-8">
|
|
<div className="w-12 h-12 bg-blue-600 rounded-xl flex items-center justify-center text-white mb-4 shadow-md shadow-blue-200">
|
|
<Activity size={24} />
|
|
</div>
|
|
<h2 className="text-2xl font-bold text-slate-800">
|
|
{isLogin ? 'Welcome Back' : 'Create Account'}
|
|
</h2>
|
|
<p className="text-slate-500 mt-2 text-sm">
|
|
{isLogin ? 'Enter your credentials to access your account' : 'Sign up to start using the platform'}
|
|
</p>
|
|
</div>
|
|
|
|
{error && (
|
|
<div className={`mb-4 p-3 rounded-lg text-sm ${error.includes('successful') ? 'bg-green-50 text-green-700 border border-green-200' : 'bg-red-50 text-red-700 border border-red-200'}`}>
|
|
{error}
|
|
</div>
|
|
)}
|
|
|
|
<form onSubmit={handleSubmit} className="space-y-4">
|
|
<div>
|
|
<label className="block text-sm font-medium text-slate-700 mb-1">Username</label>
|
|
<input
|
|
type="text"
|
|
value={userName}
|
|
onChange={(e) => setUserName(e.target.value)}
|
|
className="w-full px-4 py-2 border border-slate-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-transparent transition-shadow"
|
|
required
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label className="block text-sm font-medium text-slate-700 mb-1">Password</label>
|
|
<input
|
|
type="password"
|
|
value={password}
|
|
onChange={(e) => setPassword(e.target.value)}
|
|
className="w-full px-4 py-2 border border-slate-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-transparent transition-shadow"
|
|
required
|
|
/>
|
|
</div>
|
|
|
|
<button
|
|
type="submit"
|
|
disabled={loading}
|
|
className="w-full py-2.5 bg-blue-600 text-white rounded-lg font-medium hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-blue-500 focus:ring-offset-2 transition-colors disabled:opacity-50"
|
|
>
|
|
{loading ? 'Processing...' : (isLogin ? 'Sign In' : 'Sign Up')}
|
|
</button>
|
|
</form>
|
|
|
|
<div className="mt-6 text-center text-sm text-slate-500">
|
|
{isLogin ? "Don't have an account? " : "Already have an account? "}
|
|
<button
|
|
onClick={() => {
|
|
setIsLogin(!isLogin);
|
|
setError('');
|
|
}}
|
|
className="text-blue-600 font-medium hover:text-blue-700 focus:outline-none"
|
|
>
|
|
{isLogin ? 'Sign up' : 'Sign in'}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|