2025-12-13 14:23:40 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="sidebar-layout">
|
|
|
|
|
|
<!-- 侧边栏 -->
|
|
|
|
|
|
<aside class="sidebar" :class="{ collapsed: collapsed }">
|
|
|
|
|
|
<div class="sidebar-header">
|
|
|
|
|
|
<div class="logo">
|
|
|
|
|
|
<img src="/logo.jpg" alt="Logo" class="logo-img" />
|
|
|
|
|
|
<span v-if="!collapsed" class="logo-text">城市生命线</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="collapse-btn" @click="toggleSidebar">
|
|
|
|
|
|
<el-icon>
|
|
|
|
|
|
<DArrowLeft v-if="!collapsed" />
|
|
|
|
|
|
<DArrowRight v-else />
|
|
|
|
|
|
</el-icon>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<nav class="nav-menu">
|
|
|
|
|
|
<div class="nav-section">
|
|
|
|
|
|
<div
|
|
|
|
|
|
v-for="item in menuItems"
|
|
|
|
|
|
:key="item.key"
|
|
|
|
|
|
class="nav-item"
|
|
|
|
|
|
:class="{ active: activeMenu === item.key }"
|
|
|
|
|
|
@click="handleMenuClick(item)"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-icon><component :is="item.icon" /></el-icon>
|
|
|
|
|
|
<span v-if="!collapsed">{{ item.label }}</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</nav>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 用户信息和返回按钮 -->
|
|
|
|
|
|
<div class="user-section">
|
|
|
|
|
|
<div class="user-avatar">
|
|
|
|
|
|
<el-avatar :size="36" src="/avatar.svg" @error="handleAvatarError" />
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<span v-if="!collapsed" class="user-name">{{ userName }}</span>
|
|
|
|
|
|
<el-tooltip content="返回主系统" placement="top">
|
|
|
|
|
|
<div class="back-icon" @click="backToMain">
|
|
|
|
|
|
<el-icon><Back /></el-icon>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</el-tooltip>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</aside>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 主内容区 -->
|
|
|
|
|
|
<main class="main-content">
|
|
|
|
|
|
<!-- iframe 模式 -->
|
|
|
|
|
|
<div v-if="currentIframeUrl" class="iframe-container">
|
|
|
|
|
|
<div class="iframe-header">
|
|
|
|
|
|
<span class="iframe-title">{{ currentMenuItem?.label }}</span>
|
|
|
|
|
|
<el-button
|
|
|
|
|
|
text
|
|
|
|
|
|
@click="handleRefreshIframe"
|
|
|
|
|
|
:icon="Refresh"
|
|
|
|
|
|
>
|
|
|
|
|
|
刷新
|
|
|
|
|
|
</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<iframe
|
|
|
|
|
|
ref="iframeRef"
|
|
|
|
|
|
:src="currentIframeUrl"
|
|
|
|
|
|
class="content-iframe"
|
|
|
|
|
|
frameborder="0"
|
|
|
|
|
|
@load="handleIframeLoad"
|
|
|
|
|
|
/>
|
|
|
|
|
|
<div v-if="iframeLoading" class="iframe-loading">
|
|
|
|
|
|
<el-icon class="is-loading"><Loading /></el-icon>
|
|
|
|
|
|
<span>加载中...</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 路由模式 -->
|
|
|
|
|
|
<router-view v-else />
|
|
|
|
|
|
</main>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
|
|
|
|
|
import { ref, computed, watch } from 'vue'
|
|
|
|
|
|
import { useRouter, useRoute } from 'vue-router'
|
|
|
|
|
|
import {
|
|
|
|
|
|
ChatDotRound,
|
|
|
|
|
|
Grid,
|
|
|
|
|
|
Connection,
|
|
|
|
|
|
Document,
|
|
|
|
|
|
Service,
|
|
|
|
|
|
DArrowLeft,
|
|
|
|
|
|
DArrowRight,
|
|
|
|
|
|
User,
|
|
|
|
|
|
Setting,
|
|
|
|
|
|
SwitchButton,
|
|
|
|
|
|
Refresh,
|
|
|
|
|
|
Loading,
|
|
|
|
|
|
Back
|
|
|
|
|
|
} from '@element-plus/icons-vue'
|
|
|
|
|
|
import { ElMessage } from 'element-plus'
|
2025-12-13 16:46:04 +08:00
|
|
|
|
import type { MenuItem } from 'shared/types'
|
2025-12-13 14:23:40 +08:00
|
|
|
|
|
|
|
|
|
|
const router = useRouter()
|
|
|
|
|
|
const route = useRoute()
|
|
|
|
|
|
|
|
|
|
|
|
// 状态管理
|
|
|
|
|
|
const collapsed = ref(false)
|
|
|
|
|
|
const activeMenu = ref('home')
|
|
|
|
|
|
const iframeLoading = ref(false)
|
|
|
|
|
|
const iframeRef = ref<HTMLIFrameElement>()
|
|
|
|
|
|
|
|
|
|
|
|
// 从 LocalStorage 获取用户名
|
|
|
|
|
|
function getUserName(): string {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const loginDomainStr = localStorage.getItem('loginDomain')
|
|
|
|
|
|
if (loginDomainStr) {
|
|
|
|
|
|
const loginDomain = JSON.parse(loginDomainStr)
|
|
|
|
|
|
return loginDomain.user?.username || loginDomain.userInfo?.username || '管理员'
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
}
|
|
|
|
|
|
return '管理员'
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const userName = ref(getUserName())
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* 从 LocalStorage 加载菜单
|
|
|
|
|
|
*/
|
|
|
|
|
|
function loadMenuFromStorage(): MenuItem[] {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const loginDomainStr = localStorage.getItem('loginDomain')
|
|
|
|
|
|
if (!loginDomainStr) {
|
|
|
|
|
|
return []
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const loginDomain = JSON.parse(loginDomainStr)
|
|
|
|
|
|
const userViews = loginDomain.userViews || []
|
|
|
|
|
|
|
|
|
|
|
|
|
2025-12-13 15:56:12 +08:00
|
|
|
|
// 过滤出 AdminIframeSidebarLayout 的菜单(3个 iframe 管理后台入口)
|
2025-12-13 14:23:40 +08:00
|
|
|
|
const sidebarViews = userViews.filter((view: any) =>
|
2025-12-13 15:56:12 +08:00
|
|
|
|
view.layout === 'AdminIframeSidebarLayout' && // AdminIframeSidebarLayout 布局
|
|
|
|
|
|
view.viewType === 'iframe' && // iframe 类型
|
2025-12-13 14:23:40 +08:00
|
|
|
|
view.type === 1 && // type 1 是侧边栏菜单
|
2025-12-13 15:56:12 +08:00
|
|
|
|
view.service === 'platform' // platform 服务
|
2025-12-13 14:23:40 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// 按 orderNum 排序
|
|
|
|
|
|
sidebarViews.sort((a: any, b: any) => (a.orderNum || 0) - (b.orderNum || 0))
|
|
|
|
|
|
|
|
|
|
|
|
// 转换为 MenuItem 格式
|
|
|
|
|
|
const menuItems: MenuItem[] = sidebarViews.map((view: any) => {
|
|
|
|
|
|
// 根据 viewType 或 iframeUrl 判断是 route 还是 iframe
|
|
|
|
|
|
const isIframe = view.viewType === 'iframe' || !!view.iframeUrl
|
|
|
|
|
|
|
|
|
|
|
|
// 确定菜单的路由路径
|
|
|
|
|
|
let menuUrl = view.url
|
|
|
|
|
|
if (isIframe && view.url && (view.url.startsWith('http://') || view.url.startsWith('https://'))) {
|
|
|
|
|
|
// iframe 类型且 url 是外部链接,使用 viewId 作为路由路径
|
|
|
|
|
|
menuUrl = `/${view.viewId}`
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
key: view.viewId || view.name,
|
|
|
|
|
|
label: view.name,
|
|
|
|
|
|
icon: view.icon || 'Grid',
|
|
|
|
|
|
url: menuUrl,
|
|
|
|
|
|
type: isIframe ? 'iframe' : 'route'
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
return menuItems
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
return []
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 菜单配置(从 LocalStorage 加载)
|
|
|
|
|
|
const menuItems = ref<MenuItem[]>(loadMenuFromStorage())
|
|
|
|
|
|
|
|
|
|
|
|
// 当前菜单项
|
|
|
|
|
|
const currentMenuItem = computed(() => {
|
|
|
|
|
|
return menuItems.value.find(item => item.key === activeMenu.value)
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 当前 iframe URL(从路由 meta 读取)
|
|
|
|
|
|
const currentIframeUrl = computed(() => {
|
|
|
|
|
|
const meta = route.meta as any
|
|
|
|
|
|
return meta?.iframeUrl || null
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
// 切换侧边栏
|
|
|
|
|
|
const toggleSidebar = () => {
|
|
|
|
|
|
collapsed.value = !collapsed.value
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 处理菜单点击
|
|
|
|
|
|
const handleMenuClick = (item: MenuItem) => {
|
|
|
|
|
|
activeMenu.value = item.key
|
|
|
|
|
|
|
|
|
|
|
|
// 所有菜单都通过路由跳转
|
|
|
|
|
|
if (item.url) {
|
|
|
|
|
|
router.push(item.url)
|
|
|
|
|
|
if (item.type === 'iframe') {
|
|
|
|
|
|
iframeLoading.value = true
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// iframe 加载完成
|
|
|
|
|
|
const handleIframeLoad = () => {
|
|
|
|
|
|
iframeLoading.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 刷新 iframe
|
|
|
|
|
|
const handleRefreshIframe = () => {
|
|
|
|
|
|
if (iframeRef.value) {
|
|
|
|
|
|
iframeLoading.value = true
|
|
|
|
|
|
iframeRef.value.src = iframeRef.value.src
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 用户头像加载错误
|
|
|
|
|
|
const handleAvatarError = () => {
|
|
|
|
|
|
return true
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 返回主系统(SidebarLayout)
|
|
|
|
|
|
const backToMain = () => {
|
|
|
|
|
|
// 查找第一个非admin路由
|
|
|
|
|
|
const loginDomainStr = localStorage.getItem('loginDomain')
|
|
|
|
|
|
if (loginDomainStr) {
|
|
|
|
|
|
const loginDomain = JSON.parse(loginDomainStr)
|
|
|
|
|
|
const userViews = loginDomain.userViews || []
|
|
|
|
|
|
const mainViews = userViews.filter((view: any) =>
|
|
|
|
|
|
view.service === 'platform' && !view.url?.startsWith('/admin')
|
|
|
|
|
|
)
|
|
|
|
|
|
if (mainViews.length > 0) {
|
|
|
|
|
|
// 按 orderNum 排序,跳转到第一个
|
|
|
|
|
|
mainViews.sort((a: any, b: any) => (a.orderNum || 0) - (b.orderNum || 0))
|
|
|
|
|
|
router.push(mainViews[0].url)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 用户操作
|
|
|
|
|
|
const handleUserCommand = (command: string) => {
|
|
|
|
|
|
switch (command) {
|
|
|
|
|
|
case 'profile':
|
|
|
|
|
|
router.push('/profile')
|
|
|
|
|
|
break
|
|
|
|
|
|
case 'settings':
|
|
|
|
|
|
router.push('/settings')
|
|
|
|
|
|
break
|
|
|
|
|
|
case 'logout':
|
|
|
|
|
|
localStorage.clear()
|
|
|
|
|
|
ElMessage.success('退出成功')
|
|
|
|
|
|
router.push('/login')
|
|
|
|
|
|
break
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 监听路由变化,同步激活菜单
|
|
|
|
|
|
watch(
|
|
|
|
|
|
() => route.path,
|
|
|
|
|
|
(newPath) => {
|
|
|
|
|
|
// 查找匹配的菜单项(route 或 iframe 类型)
|
|
|
|
|
|
const menuItem = menuItems.value.find((item: MenuItem) => item.url === newPath)
|
|
|
|
|
|
if (menuItem) {
|
|
|
|
|
|
activeMenu.value = menuItem.key
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// 如果路径不匹配,尝试通过 route.name 匹配 viewId
|
|
|
|
|
|
const menuByName = menuItems.value.find((item: MenuItem) => item.key === route.name)
|
|
|
|
|
|
if (menuByName) {
|
|
|
|
|
|
activeMenu.value = menuByName.key
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
{ immediate: true }
|
|
|
|
|
|
)
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
2025-12-13 15:56:12 +08:00
|
|
|
|
@import url("./AdminIframeSidebarLayout.scss");
|
2025-12-13 14:23:40 +08:00
|
|
|
|
</style>
|