feat: sistema de autenticación con login, logout y guard de rutas
- Agrega LoginView con formulario de acceso - Agrega store de auth (Pinia) con estado isAuthenticated - Protege todas las rutas con beforeEach, redirige a /login si no autenticado - App.vue oculta nav/sidebar en rutas públicas - TopAppBar incluye botón de cerrar sesión Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@ -1,9 +1,9 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="bg-canvas min-h-screen text-ink selection:bg-accent/20 selection:text-accent">
|
<div class="bg-canvas min-h-screen text-ink selection:bg-accent/20 selection:text-accent">
|
||||||
|
|
||||||
|
<template v-if="isAuthenticated">
|
||||||
<SideNavBar />
|
<SideNavBar />
|
||||||
<TopAppBar />
|
<TopAppBar />
|
||||||
|
|
||||||
<!-- Área principal -->
|
|
||||||
<main class="ml-60 pt-16 pb-12 px-8 min-h-screen">
|
<main class="ml-60 pt-16 pb-12 px-8 min-h-screen">
|
||||||
<div class="max-w-7xl mx-auto pt-8">
|
<div class="max-w-7xl mx-auto pt-8">
|
||||||
<router-view v-slot="{ Component }">
|
<router-view v-slot="{ Component }">
|
||||||
@ -13,12 +13,23 @@
|
|||||||
</router-view>
|
</router-view>
|
||||||
</div>
|
</div>
|
||||||
</main>
|
</main>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template v-else>
|
||||||
|
<router-view />
|
||||||
|
</template>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
|
import { storeToRefs } from 'pinia'
|
||||||
|
import { useAuthStore } from './stores/auth.js'
|
||||||
import SideNavBar from './components/SideNavBar.vue'
|
import SideNavBar from './components/SideNavBar.vue'
|
||||||
import TopAppBar from './components/TopAppBar.vue'
|
import TopAppBar from './components/TopAppBar.vue'
|
||||||
|
|
||||||
|
const auth = useAuthStore()
|
||||||
|
const { isAuthenticated } = storeToRefs(auth)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
|||||||
@ -26,10 +26,27 @@
|
|||||||
<p class="text-xs font-semibold text-ink leading-none">Marketing Pro</p>
|
<p class="text-xs font-semibold text-ink leading-none">Marketing Pro</p>
|
||||||
<p class="text-[10px] text-ink-3 mt-0.5">Administrador</p>
|
<p class="text-[10px] text-ink-3 mt-0.5">Administrador</p>
|
||||||
</div>
|
</div>
|
||||||
|
<button
|
||||||
|
@click="handleLogout"
|
||||||
|
title="Cerrar sesión"
|
||||||
|
class="ml-1 text-ink-3 hover:text-red-400 transition-colors p-1.5 rounded-lg hover:bg-surface-muted"
|
||||||
|
>
|
||||||
|
<span class="material-symbols-outlined text-[20px]">logout</span>
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
import { useAuthStore } from '../stores/auth.js'
|
||||||
|
|
||||||
|
const router = useRouter()
|
||||||
|
const auth = useAuthStore()
|
||||||
|
|
||||||
|
function handleLogout() {
|
||||||
|
auth.logout()
|
||||||
|
router.push('/login')
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@ -1,12 +1,20 @@
|
|||||||
import { createRouter, createWebHistory } from 'vue-router'
|
import { createRouter, createWebHistory } from 'vue-router'
|
||||||
|
import { useAuthStore } from '../stores/auth.js'
|
||||||
import DashboardView from '../views/DashboardView.vue'
|
import DashboardView from '../views/DashboardView.vue'
|
||||||
import AnalysisCreateView from '../views/AnalysisCreateView.vue'
|
import AnalysisCreateView from '../views/AnalysisCreateView.vue'
|
||||||
import AnalysisDetailView from '../views/AnalysisDetailView.vue'
|
import AnalysisDetailView from '../views/AnalysisDetailView.vue'
|
||||||
import AnalysisListView from '../views/AnalysisListView.vue'
|
import AnalysisListView from '../views/AnalysisListView.vue'
|
||||||
import ScriptsView from '../views/ScriptsView.vue'
|
import ScriptsView from '../views/ScriptsView.vue'
|
||||||
import GenerateView from '../views/GenerateView.vue'
|
import GenerateView from '../views/GenerateView.vue'
|
||||||
|
import LoginView from '../views/LoginView.vue'
|
||||||
|
|
||||||
const routes = [
|
const routes = [
|
||||||
|
{
|
||||||
|
path: '/login',
|
||||||
|
name: 'Login',
|
||||||
|
component: LoginView,
|
||||||
|
meta: { public: true }
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/',
|
path: '/',
|
||||||
name: 'Dashboard',
|
name: 'Dashboard',
|
||||||
@ -39,7 +47,19 @@ const routes = [
|
|||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
export default createRouter({
|
const router = createRouter({
|
||||||
history: createWebHistory(),
|
history: createWebHistory(),
|
||||||
routes
|
routes
|
||||||
})
|
})
|
||||||
|
|
||||||
|
router.beforeEach((to) => {
|
||||||
|
const auth = useAuthStore()
|
||||||
|
if (!to.meta.public && !auth.isAuthenticated) {
|
||||||
|
return { name: 'Login' }
|
||||||
|
}
|
||||||
|
if (to.name === 'Login' && auth.isAuthenticated) {
|
||||||
|
return { name: 'Dashboard' }
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
export default router
|
||||||
|
|||||||
26
frontend/src/stores/auth.js
Normal file
26
frontend/src/stores/auth.js
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import { defineStore } from 'pinia'
|
||||||
|
import { ref } from 'vue'
|
||||||
|
|
||||||
|
const ADMIN_EMAIL = 'admin@gmail.com'
|
||||||
|
const ADMIN_PASSWORD = 'admin123'
|
||||||
|
const STORAGE_KEY = 'auth_session'
|
||||||
|
|
||||||
|
export const useAuthStore = defineStore('auth', () => {
|
||||||
|
const isAuthenticated = ref(!!localStorage.getItem(STORAGE_KEY))
|
||||||
|
|
||||||
|
function login(email, password) {
|
||||||
|
if (email === ADMIN_EMAIL && password === ADMIN_PASSWORD) {
|
||||||
|
localStorage.setItem(STORAGE_KEY, '1')
|
||||||
|
isAuthenticated.value = true
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
function logout() {
|
||||||
|
localStorage.removeItem(STORAGE_KEY)
|
||||||
|
isAuthenticated.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
return { isAuthenticated, login, logout }
|
||||||
|
})
|
||||||
96
frontend/src/views/LoginView.vue
Normal file
96
frontend/src/views/LoginView.vue
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
<template>
|
||||||
|
<div class="min-h-screen bg-canvas flex items-center justify-center px-4">
|
||||||
|
<div class="w-full max-w-sm">
|
||||||
|
|
||||||
|
<!-- Logo / título -->
|
||||||
|
<div class="text-center mb-10">
|
||||||
|
<div class="inline-flex items-center justify-center w-12 h-12 rounded-xl bg-accent/10 border border-accent/20 mb-4">
|
||||||
|
<span class="material-icons text-accent text-2xl">edit_note</span>
|
||||||
|
</div>
|
||||||
|
<h1 class="text-xl font-semibold text-ink">Generador de Guiones</h1>
|
||||||
|
<p class="text-sm text-ink/40 mt-1">Inicia sesión para continuar</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Formulario -->
|
||||||
|
<form @submit.prevent="handleLogin" class="space-y-4">
|
||||||
|
<div>
|
||||||
|
<label class="block text-xs font-medium text-ink/50 mb-1.5 uppercase tracking-wide">
|
||||||
|
Correo electrónico
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
v-model="email"
|
||||||
|
type="email"
|
||||||
|
placeholder="admin@gmail.com"
|
||||||
|
autocomplete="email"
|
||||||
|
class="w-full bg-surface border border-border rounded-lg px-4 py-2.5 text-sm text-ink placeholder-ink/25
|
||||||
|
focus:outline-none focus:border-accent/60 focus:ring-1 focus:ring-accent/30 transition"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<label class="block text-xs font-medium text-ink/50 mb-1.5 uppercase tracking-wide">
|
||||||
|
Contraseña
|
||||||
|
</label>
|
||||||
|
<div class="relative">
|
||||||
|
<input
|
||||||
|
v-model="password"
|
||||||
|
:type="showPassword ? 'text' : 'password'"
|
||||||
|
placeholder="••••••••"
|
||||||
|
autocomplete="current-password"
|
||||||
|
class="w-full bg-surface border border-border rounded-lg px-4 py-2.5 text-sm text-ink placeholder-ink/25
|
||||||
|
focus:outline-none focus:border-accent/60 focus:ring-1 focus:ring-accent/30 transition pr-10"
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
@click="showPassword = !showPassword"
|
||||||
|
class="absolute right-3 top-1/2 -translate-y-1/2 text-ink/30 hover:text-ink/60 transition"
|
||||||
|
>
|
||||||
|
<span class="material-icons text-lg">{{ showPassword ? 'visibility_off' : 'visibility' }}</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Error -->
|
||||||
|
<p v-if="error" class="text-red-400 text-xs text-center">{{ error }}</p>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="submit"
|
||||||
|
:disabled="loading"
|
||||||
|
class="w-full bg-accent hover:bg-accent/90 disabled:opacity-50 text-white font-medium
|
||||||
|
rounded-lg py-2.5 text-sm transition mt-2"
|
||||||
|
>
|
||||||
|
{{ loading ? 'Entrando…' : 'Entrar' }}
|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import { ref } from 'vue'
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
import { useAuthStore } from '../stores/auth.js'
|
||||||
|
|
||||||
|
const router = useRouter()
|
||||||
|
const auth = useAuthStore()
|
||||||
|
const email = ref('')
|
||||||
|
const password = ref('')
|
||||||
|
const error = ref('')
|
||||||
|
const loading = ref(false)
|
||||||
|
const showPassword = ref(false)
|
||||||
|
|
||||||
|
async function handleLogin() {
|
||||||
|
error.value = ''
|
||||||
|
loading.value = true
|
||||||
|
// pequeño delay para feedback visual
|
||||||
|
await new Promise(r => setTimeout(r, 300))
|
||||||
|
const ok = auth.login(email.value.trim(), password.value)
|
||||||
|
loading.value = false
|
||||||
|
if (ok) {
|
||||||
|
router.push('/')
|
||||||
|
} else {
|
||||||
|
error.value = 'Correo o contraseña incorrectos.'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
Reference in New Issue
Block a user