195 lines
5.2 KiB
JavaScript
195 lines
5.2 KiB
JavaScript
import { useState, useEffect } from 'react'
|
||
import { useNavigate, useLocation } from 'react-router-dom'
|
||
import {
|
||
DashboardOutlined,
|
||
DesktopOutlined,
|
||
GlobalOutlined,
|
||
CloudServerOutlined,
|
||
UserOutlined,
|
||
AppstoreOutlined,
|
||
SettingOutlined,
|
||
BlockOutlined,
|
||
FolderOutlined,
|
||
FileTextOutlined,
|
||
SafetyOutlined,
|
||
TeamOutlined,
|
||
ProjectOutlined,
|
||
RocketOutlined,
|
||
ReadOutlined,
|
||
BookOutlined,
|
||
} from '@ant-design/icons'
|
||
import { message } from 'antd'
|
||
import { getUserMenus } from '@/api/menu'
|
||
import useUserStore from '@/stores/userStore'
|
||
import ModernSidebar from '../ModernSidebar/ModernSidebar'
|
||
|
||
// 图标映射
|
||
const iconMap = {
|
||
DashboardOutlined: <DashboardOutlined />,
|
||
DesktopOutlined: <DesktopOutlined />,
|
||
GlobalOutlined: <GlobalOutlined />,
|
||
CloudServerOutlined: <CloudServerOutlined />,
|
||
UserOutlined: <UserOutlined />,
|
||
AppstoreOutlined: <AppstoreOutlined />,
|
||
SettingOutlined: <SettingOutlined />,
|
||
BlockOutlined: <BlockOutlined />,
|
||
FolderOutlined: <FolderOutlined />,
|
||
FileTextOutlined: <FileTextOutlined />,
|
||
SafetyOutlined: <SafetyOutlined />,
|
||
TeamOutlined: <TeamOutlined />,
|
||
ProjectOutlined: <ProjectOutlined />,
|
||
ReadOutlined: <ReadOutlined />,
|
||
BookOutlined: <BookOutlined />,
|
||
}
|
||
|
||
function AppSider({ collapsed, onToggle }) {
|
||
const navigate = useNavigate()
|
||
const location = useLocation()
|
||
const { user, logout } = useUserStore()
|
||
const [menuGroups, setMenuGroups] = useState([])
|
||
|
||
// 加载菜单数据
|
||
useEffect(() => {
|
||
loadMenus()
|
||
}, [])
|
||
|
||
const loadMenus = async () => {
|
||
try {
|
||
const res = await getUserMenus()
|
||
if (res.data) {
|
||
// 过滤菜单:只显示 type=1 (目录) 和 type=2 (菜单)
|
||
const validMenus = res.data.filter(item => [1, 2].includes(item.menu_type))
|
||
transformMenuData(validMenus)
|
||
}
|
||
} catch (error) {
|
||
console.error('Load menus error:', error)
|
||
message.error('加载菜单失败')
|
||
}
|
||
}
|
||
|
||
const transformMenuData = (data) => {
|
||
const groups = []
|
||
|
||
// 默认组 (用于存放一级菜单即是叶子节点的情况)
|
||
const defaultGroup = {
|
||
title: '', // 空标题或 '通用'
|
||
items: []
|
||
}
|
||
|
||
data.forEach(item => {
|
||
// 检查是否有子菜单
|
||
const validChildren = item.children ? item.children.filter(child => [1, 2].includes(child.menu_type)) : []
|
||
|
||
if (validChildren.length > 0) {
|
||
// 一级菜单作为组标题
|
||
const groupItems = validChildren.map(child => {
|
||
const icon = typeof child.icon === 'string' ? (iconMap[child.icon] || <AppstoreOutlined />) : child.icon
|
||
return {
|
||
key: child.menu_code,
|
||
label: child.menu_name,
|
||
icon: icon,
|
||
path: child.path
|
||
}
|
||
})
|
||
|
||
groups.push({
|
||
title: item.menu_name, // e.g. "系统管理"
|
||
items: groupItems
|
||
})
|
||
} else {
|
||
// 一级菜单是叶子节点,放入默认组
|
||
const icon = typeof item.icon === 'string' ? (iconMap[item.icon] || <AppstoreOutlined />) : item.icon
|
||
defaultGroup.items.push({
|
||
key: item.menu_code,
|
||
label: item.menu_name,
|
||
icon: icon,
|
||
path: item.path
|
||
})
|
||
}
|
||
})
|
||
|
||
// 如果默认组有内容,放在最前面
|
||
if (defaultGroup.items.length > 0) {
|
||
groups.unshift(defaultGroup)
|
||
}
|
||
|
||
setMenuGroups(groups)
|
||
}
|
||
|
||
const handleNavigate = (key, item) => {
|
||
if (item.path) {
|
||
navigate(item.path)
|
||
}
|
||
}
|
||
|
||
const handleLogout = () => {
|
||
logout()
|
||
navigate('/login')
|
||
}
|
||
|
||
const handleProfileClick = () => {
|
||
navigate('/profile')
|
||
}
|
||
|
||
// 获取当前激活的 key
|
||
// 简单匹配 path
|
||
const getActiveKey = () => {
|
||
const path = location.pathname
|
||
// 遍历所有 items 找匹配
|
||
for (const group of menuGroups) {
|
||
for (const item of group.items) {
|
||
if (item.path === path) return item.key
|
||
}
|
||
}
|
||
return ''
|
||
}
|
||
|
||
const logoNode = (
|
||
<div style={{ display: 'flex', alignItems: 'center', gap: 12, paddingLeft: 8 }}>
|
||
<img src="/favicon.svg" alt="logo" style={{ width: 32, height: 32 }} />
|
||
{!collapsed && (
|
||
<span style={{ fontSize: 18, fontWeight: 'bold', color: 'var(--text-color)' }}>NexDocus</span>
|
||
)}
|
||
</div>
|
||
)
|
||
|
||
// 获取用户头像URL
|
||
const getUserAvatarUrl = () => {
|
||
if (!user?.avatar) return null
|
||
// avatar 字段存储的是相对路径,如:2/avatar/xxx.jpg
|
||
// 需要转换为 API 端点: /api/v1/auth/avatar/{user_id}/{filename}
|
||
// 如果已经是 http 开头(第三方),则直接返回
|
||
if (user.avatar.startsWith('http')) return user.avatar
|
||
|
||
const parts = user.avatar.split('/')
|
||
if (parts.length >= 3) {
|
||
const userId = parts[0]
|
||
const filename = parts[2]
|
||
return `/api/v1/auth/avatar/${userId}/${filename}`
|
||
}
|
||
return null
|
||
}
|
||
|
||
const userObj = user ? {
|
||
name: user.nickname || user.username,
|
||
role: user.role_name || 'Admin',
|
||
avatar: getUserAvatarUrl()
|
||
} : null
|
||
|
||
return (
|
||
<ModernSidebar
|
||
logo={logoNode}
|
||
menuGroups={menuGroups}
|
||
activeKey={getActiveKey()}
|
||
onNavigate={handleNavigate}
|
||
user={userObj}
|
||
onLogout={handleLogout}
|
||
onProfileClick={handleProfileClick}
|
||
collapsed={collapsed}
|
||
onCollapse={onToggle}
|
||
/>
|
||
)
|
||
}
|
||
|
||
export default AppSider
|