Files
geekai/web/src/components/admin/AdminSidebar.vue
2025-04-11 15:05:21 +08:00

308 lines
6.3 KiB
Vue

<template>
<div :class="'sidebar ' + theme">
<a class="logo w-full flex items-center" href="/" target="_blank">
<img :src="logo" />
<span class="text" v-show="!sidebar.collapse">{{ title }}</span>
</a>
<el-menu
class="sidebar-el-menu"
:default-active="onRoutes"
:collapse="sidebar.collapse"
background-color="#324157"
text-color="#bfcbd9"
active-text-color="#20a0ff"
unique-opened
router
>
<template v-for="item in items">
<template v-if="item.subs">
<el-sub-menu :index="item.index" :key="item.index">
<template #title>
<i :class="'iconfont icon-' + item.icon"></i>
<span>{{ item.title }}</span>
</template>
<template v-for="subItem in item.subs">
<el-sub-menu v-if="subItem.subs" :index="subItem.index" :key="subItem.index">
<template #title>{{ subItem.title }}</template>
<el-menu-item
v-for="(threeItem, i) in subItem.subs"
:key="i"
:index="threeItem.index"
>
{{ threeItem.title }}
</el-menu-item>
</el-sub-menu>
<el-menu-item v-else :index="subItem.index" :key="subItem.index">
<i v-if="subItem.icon" :class="'iconfont icon-' + subItem.icon"></i>
{{ subItem.title }}
</el-menu-item>
</template>
</el-sub-menu>
</template>
<template v-else>
<el-menu-item :index="item.index" :key="item.index">
<i :class="'iconfont icon-' + item.icon"></i>
<template #title>{{ item.title }}</template>
</el-menu-item>
</template>
</template>
</el-menu>
</div>
</template>
<script setup>
import { useSharedStore } from '@/store/sharedata'
import { setMenuItems, useSidebarStore } from '@/store/sidebar'
import { httpGet } from '@/utils/http'
import { ElMessage } from 'element-plus'
import { computed, ref, watch } from 'vue'
import { useRoute } from 'vue-router'
const title = ref('')
const logo = ref('')
// 加载系统配置
httpGet('/api/admin/config/get?key=system')
.then((res) => {
title.value = res.data.admin_title
logo.value = res.data.logo
})
.catch((e) => {
ElMessage.error('加载系统配置失败: ' + e.message)
})
const store = useSharedStore()
const theme = ref(store.theme)
watch(
() => store.theme,
(val) => {
theme.value = val
}
)
const items = [
{
icon: 'home',
index: '/admin/dashboard',
title: '仪表盘',
},
{
icon: 'user-fill',
index: '/admin/user',
title: '用户管理',
},
{
icon: 'menu',
index: '1',
title: '应用管理',
subs: [
{
index: '/admin/app',
title: '应用列表',
icon: 'sub-menu',
},
{
index: '/admin/app/type',
title: '应用分类',
icon: 'chuangzuo',
},
],
},
{
icon: 'api-key',
index: '/admin/apikey',
title: 'API-KEY',
},
{
icon: 'model',
index: '/admin/chat/model',
title: '模型管理',
},
{
icon: 'recharge',
index: '/admin/product',
title: '充值产品',
},
{
icon: 'order',
index: '/admin/order',
title: '充值订单',
},
{
icon: 'reward',
index: '/admin/redeem',
title: '兑换码',
},
{
icon: 'control',
index: '/admin/functions',
title: '函数管理',
},
{
icon: 'menu',
index: '2',
title: '创作记录',
subs: [
{
icon: 'prompt',
index: '/admin/chats',
title: '对话记录',
},
{
icon: 'image',
index: '/admin/images',
title: '绘图记录',
},
{
icon: 'mp3',
index: '/admin/medias',
title: '音视频记录',
},
],
},
{
icon: 'role',
index: '/admin/manger',
title: '管理员',
},
{
icon: 'config',
index: '/admin/system',
title: '系统设置',
},
{
icon: 'log',
index: '/admin/powerLog',
title: '用户算力日志',
},
{
icon: 'log',
index: '/admin/loginLog',
title: '用户登录日志',
},
// {
// icon: 'menu',
// index: '1',
// title: '常用模板页面',
// subs: [
// {
// index: '/admin/demo/form',
// title: '表单页面',
// },
// {
// index: '/admin/demo/table',
// title: '常用表格',
// },
// {
// index: '/admin/demo/import',
// title: '导入Excel',
// },
// {
// index: '/admin/demo/editor',
// title: '富文本编辑器',
// },
// ],
// },
]
const route = useRoute()
const onRoutes = computed(() => {
return route.path
})
const sidebar = useSidebarStore()
setMenuItems(items)
</script>
<style scoped lang="stylus">
.sidebar {
display: block;
position: absolute;
left: 0;
top: 0;
bottom: 0;
overflow-y: scroll;
.logo {
display flex
padding 6px 15px;
cursor pointer
background-color: #324157
img {
height 36px;
padding-top 5px;
border-radius 100%
background #fff
border 2px solid #754ff6
padding 2px
}
.text {
color #ffffff
font-weight bold
padding 12px 0 12px 10px;
transition: width 2s ease;
}
}
ul {
height: 100%;
.el-menu-item, .el-sub-menu {
.iconfont {
font-size 16px;
margin-right 5px;
}
}
.el-menu-item.is-active {
background-color rgb(40, 52, 70)
}
}
.sidebar-el-menu:not(.el-menu--collapse) {
width: 250px;
}
}
.sidebar::-webkit-scrollbar {
width: 0;
}
.sidebar.dark {
border-right 1px solid var(--el-border-color-dark)
.logo {
background var(--el-bg-color)
border-right 1px solid var(--el-border-color)
.text {
color var(--el-text-color-regular)
}
}
ul {
background var(--el-bg-color)
.el-menu-item.is-active {
background-color var(--el-menu-bg-color-dark)
}
.el-menu-item:hover {
background-color var(--el-menu-bg-color-darker)
}
}
.sidebar-el-menu:not(.el-menu--collapse) {
width: 250px;
}
.el-menu {
border-color var(--el-border-color)
}
}
</style>