Files
lic/front-end/app/pages/login.vue
2026-04-21 18:05:15 -03:00

177 lines
5.1 KiB
Vue

<!-- front-end/app/pages/login.vue -->
<script setup lang="ts">
import type { TenantOption } from '~/composables/useAuth'
definePageMeta({ layout: 'auth' })
const { login, selectTenant } = useAuth()
const email = ref('')
const password = ref('')
const error = ref('')
const loading = ref(false)
// Estado do modal multi-tenant
const tenants = ref<TenantOption[]>([])
const pendingEmail = ref('')
const pendingPassword = ref('')
const showTenantModal = ref(false)
const tenantError = ref('')
const tenantLoading = ref(false)
async function handleSubmit() {
error.value = ''
loading.value = true
const result = await login(email.value, password.value)
loading.value = false
if ('success' in result && result.success) {
navigateTo('/')
} else if ('needsTenantSelect' in result) {
tenants.value = result.tenants
pendingEmail.value = result.email
pendingPassword.value = result.password
showTenantModal.value = true
} else if ('error' in result) {
error.value = result.error ?? 'Erro ao autenticar.'
}
}
async function handleSelectTenant(tenantId: string) {
tenantError.value = ''
tenantLoading.value = true
const result = await selectTenant(pendingEmail.value, pendingPassword.value, tenantId)
tenantLoading.value = false
if (result.success) {
navigateTo('/')
} else {
tenantError.value = result.error ?? 'Erro ao selecionar empresa.'
}
}
</script>
<template>
<div class="login-card">
<div class="login-header">
<h1>Bem-vindo de volta</h1>
<p>Informe suas credenciais para acessar o sistema</p>
</div>
<form @submit.prevent="handleSubmit">
<UFormField label="E-mail" class="field">
<UInput
v-model="email"
type="email"
placeholder="usuario@orgao.gov.br"
size="md"
class="w-full"
required
/>
</UFormField>
<UFormField label="Senha" class="field">
<UInput
v-model="password"
type="password"
placeholder="••••••••"
size="md"
class="w-full"
required
/>
</UFormField>
<div class="forgot">
<a href="#">Esqueceu a senha?</a>
</div>
<p v-if="error" class="error-msg">{{ error }}</p>
<UButton
type="submit"
block
size="md"
:loading="loading"
class="btn-login"
>
Entrar
</UButton>
<p class="register-link">
Não tem conta? <NuxtLink to="/register">Criar conta</NuxtLink>
</p>
</form>
</div>
<!-- Modal seleção de empresa -->
<div v-if="showTenantModal" class="modal-overlay">
<div class="modal-card">
<h2>Selecione a empresa</h2>
<p>Seu acesso está vinculado a mais de uma empresa.</p>
<div class="tenant-list">
<button
v-for="t in tenants"
:key="t.id"
class="tenant-btn"
:disabled="tenantLoading"
@click="handleSelectTenant(t.id)"
>
<span class="tenant-name">{{ t.name }}</span>
<span class="tenant-slug">{{ t.slug }}</span>
</button>
</div>
<p v-if="tenantError" class="error-msg">{{ tenantError }}</p>
</div>
</div>
</template>
<style scoped>
.login-card {
background: white;
border-radius: 20px;
padding: 48px 40px;
width: 100%;
max-width: 420px;
box-shadow: 0 25px 60px rgba(0, 0, 0, 0.25);
}
.login-header { margin-bottom: 32px; }
.login-header h1 { font-size: 24px; font-weight: 700; color: #0f172a; letter-spacing: -0.5px; }
.login-header p { font-size: 14px; color: #64748b; margin-top: 4px; }
.field { margin-bottom: 18px; }
.forgot { text-align: right; margin-top: -10px; margin-bottom: 20px; }
.forgot a { font-size: 12px; color: #667eea; text-decoration: none; font-weight: 500; }
.error-msg { color: #dc2626; font-size: 13px; margin-bottom: 12px; }
.btn-login {
background: linear-gradient(135deg, #667eea, #764ba2) !important;
box-shadow: 0 4px 15px rgba(102, 126, 234, 0.4);
}
.register-link { text-align: center; font-size: 13px; color: #64748b; margin-top: 16px; }
.register-link a { color: #667eea; font-weight: 600; text-decoration: none; }
/* Modal */
.modal-overlay {
position: fixed; inset: 0;
background: rgba(0,0,0,0.5);
display: flex; align-items: center; justify-content: center;
z-index: 100;
}
.modal-card {
background: white; border-radius: 16px; padding: 32px;
width: 100%; max-width: 380px;
box-shadow: 0 20px 50px rgba(0,0,0,0.3);
}
.modal-card h2 { font-size: 18px; font-weight: 700; color: #0f172a; margin-bottom: 6px; }
.modal-card p { font-size: 13px; color: #64748b; margin-bottom: 20px; }
.tenant-list { display: flex; flex-direction: column; gap: 10px; }
.tenant-btn {
display: flex; flex-direction: column; align-items: flex-start;
padding: 14px 16px; border-radius: 10px;
border: 1.5px solid #e2e8f0; background: white;
cursor: pointer; transition: all 0.15s; text-align: left;
}
.tenant-btn:hover { border-color: #667eea; background: #f0f3ff; }
.tenant-btn:disabled { opacity: 0.6; cursor: not-allowed; }
.tenant-name { font-size: 14px; font-weight: 600; color: #0f172a; }
.tenant-slug { font-size: 12px; color: #94a3b8; margin-top: 2px; }
</style>