265 lines
5.9 KiB
Vue
265 lines
5.9 KiB
Vue
<!-- 管理端布局组件 -->
|
|
<template>
|
|
<div class="admin-layout">
|
|
<!-- 侧边栏 -->
|
|
<el-aside width="220px" class="aside">
|
|
<div class="logo">
|
|
<img src="@/assets/logo.png" alt="SAMS" />
|
|
<span>SAMS管理系统</span>
|
|
</div>
|
|
<el-menu
|
|
:default-active="activeMenu"
|
|
class="el-menu-vertical"
|
|
:collapse="isCollapse"
|
|
background-color="#304156"
|
|
text-color="#bfcbd9"
|
|
active-text-color="#409EFF"
|
|
:unique-opened="true"
|
|
:router="true"
|
|
>
|
|
<el-menu-item index="/">
|
|
<el-icon><HomeFilled /></el-icon>
|
|
<template #title>首页</template>
|
|
</el-menu-item>
|
|
|
|
<el-sub-menu index="/activity">
|
|
<template #title>
|
|
<el-icon><Calendar /></el-icon>
|
|
<span>活动管理</span>
|
|
</template>
|
|
<el-menu-item index="/activity/list">活动列表</el-menu-item>
|
|
<el-menu-item index="/activity/approval">待审批</el-menu-item>
|
|
</el-sub-menu>
|
|
|
|
<el-sub-menu index="/club">
|
|
<template #title>
|
|
<el-icon><Collection /></el-icon>
|
|
<span>社团管理</span>
|
|
</template>
|
|
<el-menu-item index="/club/list">社团列表</el-menu-item>
|
|
<el-menu-item index="/club/member">成员管理</el-menu-item>
|
|
</el-sub-menu>
|
|
|
|
<el-menu-item index="/user" v-if="isAdmin">
|
|
<el-icon><User /></el-icon>
|
|
<template #title>人员管理</template>
|
|
</el-menu-item>
|
|
</el-menu>
|
|
</el-aside>
|
|
|
|
<el-container>
|
|
<!-- 顶部导航栏 -->
|
|
<el-header class="header">
|
|
<div class="header-left">
|
|
<el-icon
|
|
class="collapse-btn"
|
|
@click="toggleSidebar"
|
|
:class="{ 'is-active': isCollapse }"
|
|
>
|
|
<Fold v-if="isCollapse" />
|
|
<Expand v-else />
|
|
</el-icon>
|
|
<el-breadcrumb separator="/">
|
|
<el-breadcrumb-item :to="{ path: '/' }">首页</el-breadcrumb-item>
|
|
<el-breadcrumb-item v-if="route.meta?.title">
|
|
{{ route.meta.title }}
|
|
</el-breadcrumb-item>
|
|
</el-breadcrumb>
|
|
</div>
|
|
<div class="header-right">
|
|
<el-dropdown @command="handleCommand">
|
|
<span class="user-dropdown">
|
|
<el-avatar :size="32" :src="userInfo.avatar" />
|
|
<span>{{ userInfo.name }}</span>
|
|
<el-icon><CaretBottom /></el-icon>
|
|
</span>
|
|
<template #dropdown>
|
|
<el-dropdown-menu>
|
|
<el-dropdown-item command="profile">个人信息</el-dropdown-item>
|
|
<el-dropdown-item command="password">修改密码</el-dropdown-item>
|
|
<el-dropdown-item divided command="logout">退出登录</el-dropdown-item>
|
|
</el-dropdown-menu>
|
|
</template>
|
|
</el-dropdown>
|
|
</div>
|
|
</el-header>
|
|
|
|
<!-- 主要内容区域 -->
|
|
<el-main class="main">
|
|
<router-view v-slot="{ Component }">
|
|
<transition name="fade" mode="out-in">
|
|
<component :is="Component" />
|
|
</transition>
|
|
</router-view>
|
|
</el-main>
|
|
</el-container>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { ref, computed, onMounted } from 'vue'
|
|
import { useRouter, useRoute } from 'vue-router'
|
|
import { ElMessage } from 'element-plus'
|
|
import {
|
|
HomeFilled,
|
|
Calendar,
|
|
Collection,
|
|
User,
|
|
Fold,
|
|
Expand,
|
|
CaretBottom
|
|
} from '@element-plus/icons-vue'
|
|
import { removeToken } from '@/utils/auth'
|
|
import { isAdmin } from '@/utils/permission'
|
|
import { useUserStore } from '@/stores/user'
|
|
|
|
const router = useRouter()
|
|
const route = useRoute()
|
|
const userStore = useUserStore()
|
|
|
|
// 侧边栏折叠状态
|
|
const isCollapse = ref(false)
|
|
|
|
// 当前激活的菜单项
|
|
const activeMenu = computed(() => route.path)
|
|
|
|
// 用户信息
|
|
const userInfo = computed(() => userStore.userInfo)
|
|
|
|
// 切换侧边栏
|
|
const toggleSidebar = () => {
|
|
isCollapse.value = !isCollapse.value
|
|
}
|
|
|
|
// 处理下拉菜单命令
|
|
const handleCommand = async (command) => {
|
|
switch (command) {
|
|
case 'profile':
|
|
router.push('/profile')
|
|
break
|
|
case 'password':
|
|
router.push('/user/password')
|
|
break
|
|
case 'logout':
|
|
await handleLogout()
|
|
break
|
|
}
|
|
}
|
|
|
|
// 处理退出登录
|
|
const handleLogout = async () => {
|
|
try {
|
|
await userStore.clearUserInfo()
|
|
removeToken()
|
|
router.push('/login')
|
|
} catch (error) {
|
|
console.error('退出登录失败:', error)
|
|
}
|
|
}
|
|
|
|
// 初始化时获取用户信息
|
|
onMounted(async () => {
|
|
await userStore.fetchUserInfo()
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.admin-layout {
|
|
height: 100vh;
|
|
display: flex;
|
|
}
|
|
|
|
.aside {
|
|
background-color: #304156;
|
|
height: 100vh;
|
|
transition: width 0.3s;
|
|
overflow-x: hidden;
|
|
}
|
|
|
|
.logo {
|
|
height: 60px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
background-color: #2b3649;
|
|
padding: 0 20px;
|
|
}
|
|
|
|
.logo img {
|
|
height: 32px;
|
|
margin-right: 12px;
|
|
}
|
|
|
|
.logo span {
|
|
color: #fff;
|
|
font-size: 16px;
|
|
font-weight: 600;
|
|
white-space: nowrap;
|
|
}
|
|
|
|
.el-menu-vertical {
|
|
border-right: none;
|
|
}
|
|
|
|
.el-menu-vertical:not(.el-menu--collapse) {
|
|
width: 220px;
|
|
}
|
|
|
|
.header {
|
|
background-color: #fff;
|
|
border-bottom: 1px solid #e6e6e6;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
padding: 0 20px;
|
|
}
|
|
|
|
.header-left {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 20px;
|
|
}
|
|
|
|
.collapse-btn {
|
|
font-size: 20px;
|
|
cursor: pointer;
|
|
transition: all 0.3s;
|
|
}
|
|
|
|
.collapse-btn:hover {
|
|
color: #409EFF;
|
|
}
|
|
|
|
.collapse-btn.is-active {
|
|
transform: rotate(180deg);
|
|
}
|
|
|
|
.header-right {
|
|
display: flex;
|
|
align-items: center;
|
|
}
|
|
|
|
.user-dropdown {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 8px;
|
|
cursor: pointer;
|
|
padding: 0 8px;
|
|
}
|
|
|
|
.main {
|
|
background-color: #f0f2f5;
|
|
height: calc(100vh - 60px);
|
|
overflow-y: auto;
|
|
}
|
|
|
|
.fade-enter-active,
|
|
.fade-leave-active {
|
|
transition: opacity 0.3s ease;
|
|
}
|
|
|
|
.fade-enter-from,
|
|
.fade-leave-to {
|
|
opacity: 0;
|
|
}
|
|
</style> |