feat: 完成 #3 登录页面和认证功能

This commit is contained in:
root
2026-03-09 19:43:05 +08:00
parent 135d2f92b2
commit 0894833b8f
5 changed files with 53 additions and 12 deletions
+1 -2
View File
@@ -1,14 +1,13 @@
import { createApp } from 'vue' import { createApp } from 'vue'
import { createPinia } from 'pinia'
import ElementPlus from 'element-plus' import ElementPlus from 'element-plus'
import 'element-plus/dist/index.css' import 'element-plus/dist/index.css'
import * as ElementPlusIconsVue from '@element-plus/icons-vue' import * as ElementPlusIconsVue from '@element-plus/icons-vue'
import router from './router' import router from './router'
import App from './App.vue' import App from './App.vue'
import { pinia } from './stores'
import './style.css' import './style.css'
const app = createApp(App) const app = createApp(App)
const pinia = createPinia()
// 注册所有 Element Plus 图标 // 注册所有 Element Plus 图标
for (const [key, component] of Object.entries(ElementPlusIconsVue)) { for (const [key, component] of Object.entries(ElementPlusIconsVue)) {
+29 -7
View File
@@ -1,4 +1,6 @@
import { createRouter, createWebHistory } from 'vue-router' import { createRouter, createWebHistory } from 'vue-router'
import { useUserStore } from '../stores/user'
import { pinia } from '../stores'
const routes = [ const routes = [
{ {
@@ -83,15 +85,35 @@ const router = createRouter({
}) })
// 路由守卫 // 路由守卫
router.beforeEach((to, from, next) => { router.beforeEach(async (to, from, next) => {
const token = localStorage.getItem('admin_token') const userStore = useUserStore(pinia)
const token = userStore.token
if (to.meta.requiresAuth && !token) { if (!to.meta.requiresAuth) {
next('/login') if (to.path === '/login' && token) {
} else if (to.path === '/login' && token) { try {
next('/') await userStore.ensureProfileLoaded()
} else { next('/')
} catch (error) {
next()
}
return
}
next() next()
return
}
if (!token) {
next(`/login?redirect=${encodeURIComponent(to.fullPath)}`)
return
}
try {
await userStore.ensureProfileLoaded()
next()
} catch (error) {
userStore.logout()
next('/login')
} }
}) })
+3
View File
@@ -0,0 +1,3 @@
import { createPinia } from 'pinia'
export const pinia = createPinia()
+15 -1
View File
@@ -1,5 +1,5 @@
import { defineStore } from 'pinia' import { defineStore } from 'pinia'
import { login as loginApi, getProfile } from '../api/auth' import { getProfile, login as loginApi, logout as logoutApi } from '../api/auth'
export const useUserStore = defineStore('user', { export const useUserStore = defineStore('user', {
state: () => ({ state: () => ({
@@ -35,7 +35,21 @@ export const useUserStore = defineStore('user', {
} }
}, },
async ensureProfileLoaded() {
if (!this.token) {
throw new Error('未登录')
}
if (this.userInfo) {
return this.userInfo
}
const res = await this.fetchUserInfo()
return res.data
},
logout() { logout() {
if (this.token) {
logoutApi().catch(() => {})
}
this.token = '' this.token = ''
this.userInfo = null this.userInfo = null
localStorage.removeItem('admin_token') localStorage.removeItem('admin_token')
+5 -2
View File
@@ -59,12 +59,13 @@
<script setup> <script setup>
import { ref, reactive } from 'vue' import { ref, reactive } from 'vue'
import { useRouter } from 'vue-router' import { useRoute, useRouter } from 'vue-router'
import { ElMessage } from 'element-plus' import { ElMessage } from 'element-plus'
import { User, Lock } from '@element-plus/icons-vue' import { User, Lock } from '@element-plus/icons-vue'
import { useUserStore } from '../../stores/user' import { useUserStore } from '../../stores/user'
const router = useRouter() const router = useRouter()
const route = useRoute()
const userStore = useUserStore() const userStore = useUserStore()
const loginFormRef = ref(null) const loginFormRef = ref(null)
@@ -95,7 +96,9 @@ const handleLogin = async () => {
try { try {
await userStore.login(loginForm.username, loginForm.password) await userStore.login(loginForm.username, loginForm.password)
ElMessage.success('登录成功') ElMessage.success('登录成功')
router.push('/') const redirectPath =
typeof route.query.redirect === 'string' ? route.query.redirect : '/'
router.push(redirectPath)
} catch (error) { } catch (error) {
console.error('登录失败:', error) console.error('登录失败:', error)
} finally { } finally {