SAMS/sams-ui/src/layout/admin/index.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>