195 lines
6.5 KiB
TypeScript
195 lines
6.5 KiB
TypeScript
"use client";
|
|
|
|
import { Suspense, useState } from "react";
|
|
import { useRouter, useSearchParams } from "next/navigation";
|
|
|
|
function LoginForm() {
|
|
const router = useRouter();
|
|
const searchParams = useSearchParams();
|
|
const next = searchParams.get("next") || "/dashboard";
|
|
|
|
const [username, setUsername] = useState("");
|
|
const [password, setPassword] = useState("");
|
|
const [error, setError] = useState("");
|
|
const [loading, setLoading] = useState(false);
|
|
|
|
// #region agent log
|
|
function debugLog(hypothesisId: string, message: string, data: Record<string, unknown>) {
|
|
fetch("http://localhost:7701/ingest/5073259c-ddcc-441a-a087-e13a2cf7ac9e", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
"X-Debug-Session-Id": "ca90b5",
|
|
},
|
|
body: JSON.stringify({
|
|
sessionId: "ca90b5",
|
|
runId: "login-debug-01",
|
|
hypothesisId,
|
|
location: "frontend/app/tela_login/page.tsx:debugLog",
|
|
message,
|
|
data,
|
|
timestamp: Date.now(),
|
|
}),
|
|
}).catch(() => {});
|
|
}
|
|
// #endregion
|
|
|
|
async function handleSubmit(e: React.FormEvent) {
|
|
e.preventDefault();
|
|
setError("");
|
|
if (!username.trim() || !password.trim()) {
|
|
setError("Informe usuário e senha.");
|
|
return;
|
|
}
|
|
setLoading(true);
|
|
try {
|
|
// #region agent log
|
|
debugLog("H4", "login_submit_start", {
|
|
endpoint: "/api/api/auth/login/",
|
|
next,
|
|
usernameLength: username.trim().length,
|
|
});
|
|
// #endregion
|
|
const res = await fetch("/api/api/auth/login/", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
credentials: "include",
|
|
body: JSON.stringify({ username: username.trim(), password, next }),
|
|
});
|
|
const data = await res.json().catch(() => ({}));
|
|
// #region agent log
|
|
debugLog("H5", "login_submit_response_body", {
|
|
status: res.status,
|
|
ok: res.ok,
|
|
success: (data as { success?: boolean }).success,
|
|
hasSuccessKey: Object.prototype.hasOwnProperty.call(data, "success"),
|
|
redirect: (data as { redirect?: string })?.redirect || null,
|
|
message: (data as { message?: string })?.message || null,
|
|
});
|
|
debugLog("H1_H3", "login_submit_response", {
|
|
status: res.status,
|
|
ok: res.ok,
|
|
redirect: (data as { redirect?: string })?.redirect || null,
|
|
message: (data as { message?: string })?.message || null,
|
|
});
|
|
// #endregion
|
|
if (!res.ok) {
|
|
setError(data.message || "Erro ao autenticar. Tente novamente.");
|
|
return;
|
|
}
|
|
if (data.success && data.redirect) {
|
|
router.push(data.redirect);
|
|
} else {
|
|
router.push("/dashboard");
|
|
}
|
|
} catch {
|
|
// #region agent log
|
|
debugLog("H2", "login_submit_fetch_exception", {
|
|
apiBaseHint: "proxied_by_next_api_route",
|
|
});
|
|
// #endregion
|
|
setError("Erro de conexão. Verifique se o backend está rodando.");
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div className="flex w-full h-screen overflow-hidden">
|
|
{/* Painel esquerdo - visual */}
|
|
<div className="hidden md:flex flex-1 flex-col justify-center items-center text-white text-center p-8 bg-gradient-to-br from-[#1e3a8a] to-[#3b82f6] relative">
|
|
<div className="absolute top-5 right-5 bg-white/15 backdrop-blur-sm py-1.5 px-3 rounded-full text-xs font-semibold tracking-wide border border-white/20">
|
|
PROD
|
|
</div>
|
|
<h1 className="text-5xl font-bold mb-4 tracking-tight">SGMP</h1>
|
|
<p className="text-xl opacity-90 max-w-md">
|
|
Sistema de Gestão e Movimentações de Pessoas
|
|
</p>
|
|
</div>
|
|
|
|
{/* Painel direito - formulário */}
|
|
<div className="flex-1 flex justify-center items-center bg-white p-8">
|
|
<div className="w-full max-w-md">
|
|
<div className="mb-8">
|
|
<h2 className="text-2xl font-semibold text-slate-800 mb-1">
|
|
Bem-vindo
|
|
</h2>
|
|
<p className="text-slate-500">Insira suas credenciais para acessar.</p>
|
|
</div>
|
|
|
|
{error && (
|
|
<div className="mb-6 p-3 rounded-lg bg-red-50 text-red-800 text-sm border border-red-200">
|
|
{error}
|
|
</div>
|
|
)}
|
|
|
|
<form onSubmit={handleSubmit} className="space-y-5">
|
|
<div>
|
|
<label
|
|
htmlFor="username"
|
|
className="block mb-2 font-medium text-slate-700 text-sm"
|
|
>
|
|
Usuário
|
|
</label>
|
|
<input
|
|
id="username"
|
|
type="text"
|
|
name="username"
|
|
value={username}
|
|
onChange={(e) => setUsername(e.target.value)}
|
|
placeholder="Matrícula Winthor"
|
|
required
|
|
autoFocus
|
|
className="w-full px-4 py-3 border border-slate-200 rounded-lg text-base focus:outline-none focus:border-blue-500 focus:ring-2 focus:ring-blue-500/20 transition-colors"
|
|
/>
|
|
</div>
|
|
|
|
<div>
|
|
<label
|
|
htmlFor="password"
|
|
className="block mb-2 font-medium text-slate-700 text-sm"
|
|
>
|
|
Senha
|
|
</label>
|
|
<input
|
|
id="password"
|
|
type="password"
|
|
name="password"
|
|
value={password}
|
|
onChange={(e) => setPassword(e.target.value)}
|
|
placeholder="Senha Winthor"
|
|
required
|
|
className="w-full px-4 py-3 border border-slate-200 rounded-lg text-base focus:outline-none focus:border-blue-500 focus:ring-2 focus:ring-blue-500/20 transition-colors"
|
|
/>
|
|
</div>
|
|
|
|
<button
|
|
type="submit"
|
|
disabled={loading}
|
|
className="w-full py-3.5 px-4 bg-blue-600 hover:bg-blue-700 disabled:bg-blue-400 disabled:cursor-not-allowed text-white font-semibold rounded-lg text-base transition-colors"
|
|
>
|
|
{loading ? "Entrando…" : "Entrar"}
|
|
</button>
|
|
</form>
|
|
|
|
<p className="mt-6 text-center text-sm text-slate-500">
|
|
Use suas credenciais do Winthor (matrícula e senha).
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default function TelaLoginPage() {
|
|
return (
|
|
<Suspense fallback={
|
|
<div className="flex items-center justify-center min-h-screen bg-white">
|
|
<p className="text-slate-500">Carregando…</p>
|
|
</div>
|
|
}>
|
|
<LoginForm />
|
|
</Suspense>
|
|
);
|
|
}
|