Files
Frontend-Iasis/src/pages/auth/LoginPage.tsx

107 lines
3.6 KiB
TypeScript

import { useState } from 'react';
import { Link, useNavigate } from 'react-router-dom';
import { useForm } from 'react-hook-form';
import { zodResolver } from '@hookform/resolvers/zod';
import { z } from 'zod';
import axios from 'axios';
import { useAuth } from '../../modules/auth';
import { Button } from '../../components/ui';
const loginSchema = z.object({
email: z.string().min(1, 'E-mail é obrigatório').email('E-mail inválido'),
password: z.string().min(1, 'Senha é obrigatória'),
});
type LoginFormData = z.infer<typeof loginSchema>;
export function LoginPage() {
const { login } = useAuth();
const navigate = useNavigate();
const [apiError, setApiError] = useState<string | null>(null);
const {
register,
handleSubmit,
formState: { errors, isSubmitting },
} = useForm<LoginFormData>({
resolver: zodResolver(loginSchema),
});
async function onSubmit(data: LoginFormData) {
setApiError(null);
try {
await login(data.email, data.password);
navigate('/', { replace: true });
} catch (err) {
if (axios.isAxiosError(err)) {
const status = err.response?.status;
if (status === 401) {
setApiError('E-mail ou senha incorretos');
} else if (status === 403) {
setApiError('Usuário inativo');
} else {
setApiError('Erro ao realizar login. Tente novamente.');
}
} else {
setApiError('Erro ao realizar login. Tente novamente.');
}
}
}
return (
<div className="flex min-h-screen items-center justify-center bg-[#0B1F2A]">
<div className="w-full max-w-sm rounded-lg bg-[#0F2A38] p-8">
<div className="mb-8 text-center">
<img src="/logo/logo-full.png" alt="ISIS" className="mx-auto h-12" />
<p className="mt-1 text-small text-text-secondary">Sistema de Gestão</p>
</div>
<form onSubmit={handleSubmit(onSubmit)} noValidate className="space-y-4">
<div>
<label htmlFor="email" className="mb-1 block text-small text-text-secondary">
E-mail
</label>
<input
id="email"
type="email"
placeholder="seu@email.com"
{...register('email')}
className="w-full rounded border border-white/10 bg-[#0B1F2A] px-3 py-2 text-body text-text-primary placeholder:text-text-muted focus:border-isis-blue focus:outline-none"
/>
{errors.email && <p className="mt-1 text-small text-red-400">{errors.email.message}</p>}
</div>
<div>
<label htmlFor="password" className="mb-1 block text-small text-text-secondary">
Senha
</label>
<input
id="password"
type="password"
placeholder="••••••••"
{...register('password')}
className="w-full rounded border border-white/10 bg-[#0B1F2A] px-3 py-2 text-body text-text-primary placeholder:text-text-muted focus:border-isis-blue focus:outline-none"
/>
{errors.password && (
<p className="mt-1 text-small text-red-400">{errors.password.message}</p>
)}
</div>
{apiError && <p className="text-small text-red-400">{apiError}</p>}
<Button type="submit" loading={isSubmitting} className="w-full">
{isSubmitting ? 'Entrando...' : 'Entrar'}
</Button>
</form>
<div className="mt-4 text-center">
<Link to="/esqueci-senha" className="text-small text-isis-blue hover:underline">
Esqueceu sua senha?
</Link>
</div>
</div>
</div>
);
}