|
|
|
|
<template>
|
|
|
|
|
<div class="main-layout">
|
|
|
|
|
<!-- 顶部导航栏 -->
|
|
|
|
|
<header class="header">
|
|
|
|
|
<div class="header-left">
|
|
|
|
|
<div class="logo">
|
|
|
|
|
<h1>🚀 GoFaster</h1>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="breadcrumb">
|
|
|
|
|
<span v-for="(item, index) in breadcrumbs" :key="index">
|
|
|
|
|
<span v-if="index > 0" class="separator">/</span>
|
|
|
|
|
<span class="breadcrumb-item">{{ item }}</span>
|
|
|
|
|
</span>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="header-right">
|
|
|
|
|
<!-- 消息通知 -->
|
|
|
|
|
<div class="message-center">
|
|
|
|
|
<button class="message-btn" @click="toggleMessagePanel">
|
|
|
|
|
<i class="icon">📢</i>
|
|
|
|
|
<span v-if="unreadCount > 0" class="badge">{{ unreadCount }}</span>
|
|
|
|
|
</button>
|
|
|
|
|
|
|
|
|
|
<!-- 消息面板 -->
|
|
|
|
|
<div v-if="showMessagePanel" class="message-panel">
|
|
|
|
|
<div class="message-header">
|
|
|
|
|
<h3>消息</h3>
|
|
|
|
|
<button class="close-btn" @click="showMessagePanel = false">×</button>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="message-list">
|
|
|
|
|
<div v-for="message in messages" :key="message.id" class="message-item">
|
|
|
|
|
<div class="message-icon">{{ message.icon }}</div>
|
|
|
|
|
<div class="message-content">
|
|
|
|
|
<div class="message-title">{{ message.title }}</div>
|
|
|
|
|
<div class="message-time">{{ formatTime(message.time) }}</div>
|
|
|
|
|
</div>
|
|
|
|
|
<button class="mark-read-btn" @click="markAsRead(message.id)">
|
|
|
|
|
✓
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- 用户信息 -->
|
|
|
|
|
<div class="user-info">
|
|
|
|
|
<div class="user-avatar" @click="toggleUserMenu">
|
|
|
|
|
<img v-if="currentUser.avatar" :src="currentUser.avatar" :alt="currentUser.name" />
|
|
|
|
|
<span v-else class="avatar-placeholder">{{ currentUser.name?.charAt(0) || 'U' }}</span>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- 用户菜单 -->
|
|
|
|
|
<div v-if="showUserMenu" class="user-menu">
|
|
|
|
|
<div class="user-menu-header">
|
|
|
|
|
<div class="user-details">
|
|
|
|
|
<div class="user-name">{{ currentUser.name || '用户' }}</div>
|
|
|
|
|
<div class="user-email">{{ currentUser.email || 'user@example.com' }}</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="user-menu-items">
|
|
|
|
|
<button class="menu-item" @click="openProfile">
|
|
|
|
|
<i class="icon">👤</i> 个人资料
|
|
|
|
|
</button>
|
|
|
|
|
<button class="menu-item" @click="openSettings">
|
|
|
|
|
<i class="icon">⚙️</i> 设置
|
|
|
|
|
</button>
|
|
|
|
|
<button class="menu-item" @click="logout">
|
|
|
|
|
<i class="icon">🚪</i> 退出登录
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</header>
|
|
|
|
|
|
|
|
|
|
<!-- 主要内容区域 -->
|
|
|
|
|
<div class="main-content">
|
|
|
|
|
<!-- 左侧菜单 -->
|
|
|
|
|
<aside class="sidebar">
|
|
|
|
|
<nav class="sidebar-nav">
|
|
|
|
|
<div class="nav-section">
|
|
|
|
|
<ul class="nav-list">
|
|
|
|
|
<li v-for="item in mainMenuItems" :key="item.id">
|
|
|
|
|
<router-link
|
|
|
|
|
:to="item.path"
|
|
|
|
|
class="nav-item"
|
|
|
|
|
:class="{ active: currentRoute === item.path }"
|
|
|
|
|
@click="handleMenuClick(item)"
|
|
|
|
|
>
|
|
|
|
|
<i class="nav-icon">{{ item.icon }}</i>
|
|
|
|
|
<span class="nav-text">{{ item.name }}</span>
|
|
|
|
|
<button
|
|
|
|
|
v-if="item.favorite"
|
|
|
|
|
class="favorite-btn"
|
|
|
|
|
@click.stop="toggleFavorite(item.id)"
|
|
|
|
|
>
|
|
|
|
|
⭐
|
|
|
|
|
</button>
|
|
|
|
|
</router-link>
|
|
|
|
|
</li>
|
|
|
|
|
</ul>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="nav-section">
|
|
|
|
|
<ul class="nav-list">
|
|
|
|
|
<li v-for="item in favoriteMenuItems" :key="item.id">
|
|
|
|
|
<router-link
|
|
|
|
|
:to="item.path"
|
|
|
|
|
class="nav-item"
|
|
|
|
|
:class="{ active: currentRoute === item.path }"
|
|
|
|
|
>
|
|
|
|
|
<i class="nav-icon">{{ item.icon }}</i>
|
|
|
|
|
<span class="nav-text">{{ item.name }}</span>
|
|
|
|
|
<button
|
|
|
|
|
class="favorite-btn active"
|
|
|
|
|
@click.stop="toggleFavorite(item.id)"
|
|
|
|
|
>
|
|
|
|
|
⭐
|
|
|
|
|
</button>
|
|
|
|
|
</router-link>
|
|
|
|
|
</li>
|
|
|
|
|
</ul>
|
|
|
|
|
</div>
|
|
|
|
|
</nav>
|
|
|
|
|
</aside>
|
|
|
|
|
|
|
|
|
|
<!-- 右侧内容区域 -->
|
|
|
|
|
<div class="content-area">
|
|
|
|
|
<!-- 内容选项卡 -->
|
|
|
|
|
<div class="content-tabs">
|
|
|
|
|
<div class="tab-list">
|
|
|
|
|
<div
|
|
|
|
|
v-for="tab in openTabs"
|
|
|
|
|
:key="tab.id"
|
|
|
|
|
:class="['tab-item', { active: tab.id === currentTab }]"
|
|
|
|
|
@click="switchTab(tab.id)"
|
|
|
|
|
>
|
|
|
|
|
<span class="tab-title">{{ tab.title }}</span>
|
|
|
|
|
<button
|
|
|
|
|
v-if="openTabs.length > 1"
|
|
|
|
|
class="tab-close"
|
|
|
|
|
@click.stop="closeTab(tab.id)"
|
|
|
|
|
title="关闭标签页"
|
|
|
|
|
>
|
|
|
|
|
×
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="tab-actions">
|
|
|
|
|
<button
|
|
|
|
|
v-if="openTabs.length > 1"
|
|
|
|
|
class="close-all-btn"
|
|
|
|
|
@click="closeAllTabs"
|
|
|
|
|
title="关闭所有标签页"
|
|
|
|
|
>
|
|
|
|
|
<span class="close-all-icon">⊗</span>
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- 功能内容区 -->
|
|
|
|
|
<div class="content-body">
|
|
|
|
|
<router-view />
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script>
|
|
|
|
|
import { ref, reactive, computed, onMounted, watch } from 'vue'
|
|
|
|
|
import { useRouter, useRoute } from 'vue-router'
|
|
|
|
|
import { userService } from '@/services/userService'
|
|
|
|
|
|
|
|
|
|
export default {
|
|
|
|
|
name: 'MainLayout',
|
|
|
|
|
setup() {
|
|
|
|
|
const router = useRouter()
|
|
|
|
|
const route = useRoute()
|
|
|
|
|
|
|
|
|
|
// 响应式数据
|
|
|
|
|
const showMessagePanel = ref(false)
|
|
|
|
|
const showUserMenu = ref(false)
|
|
|
|
|
const currentTab = ref('home')
|
|
|
|
|
const openTabs = ref([
|
|
|
|
|
{ id: 'home', title: '欢迎', path: '/', closable: false }
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
const currentUser = reactive({
|
|
|
|
|
name: '管理员',
|
|
|
|
|
email: 'admin@gofaster.com',
|
|
|
|
|
avatar: null
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const messages = ref([
|
|
|
|
|
{
|
|
|
|
|
id: 1,
|
|
|
|
|
title: '系统更新完成',
|
|
|
|
|
icon: '🔄',
|
|
|
|
|
time: new Date(Date.now() - 1000 * 60 * 30),
|
|
|
|
|
read: false
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
id: 2,
|
|
|
|
|
title: '新用户注册',
|
|
|
|
|
icon: '👤',
|
|
|
|
|
time: new Date(Date.now() - 1000 * 60 * 60),
|
|
|
|
|
read: false
|
|
|
|
|
}
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
const mainMenuItems = ref([
|
|
|
|
|
{ id: 'home', name: '欢迎', path: '/', icon: '🏠', favorite: false },
|
|
|
|
|
{ id: 'speed-test', name: '速度测试', path: '/speed-test', icon: '⚡', favorite: false },
|
|
|
|
|
{ id: 'user-management', name: '用户管理', path: '/user-management', icon: '👥', favorite: false },
|
|
|
|
|
{ id: 'history', name: '历史记录', path: '/history', icon: '📊', favorite: false },
|
|
|
|
|
{ id: 'settings', name: '系统设置', path: '/settings', icon: '⚙️', favorite: false }
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
const favoriteMenuItems = ref([])
|
|
|
|
|
|
|
|
|
|
// 计算属性
|
|
|
|
|
const unreadCount = computed(() => messages.value.filter(m => !m.read).length)
|
|
|
|
|
const currentRoute = computed(() => route.path)
|
|
|
|
|
const breadcrumbs = computed(() => {
|
|
|
|
|
const path = route.path
|
|
|
|
|
if (path === '/') return ['欢迎']
|
|
|
|
|
if (path === '/user-management') return ['欢迎', '用户管理']
|
|
|
|
|
if (path === '/speed-test') return ['欢迎', '速度测试']
|
|
|
|
|
if (path === '/history') return ['欢迎', '历史记录']
|
|
|
|
|
if (path === '/settings') return ['欢迎', '系统设置']
|
|
|
|
|
return ['欢迎']
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// 方法
|
|
|
|
|
const toggleMessagePanel = () => {
|
|
|
|
|
showMessagePanel.value = !showMessagePanel.value
|
|
|
|
|
showUserMenu.value = false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const toggleUserMenu = () => {
|
|
|
|
|
showUserMenu.value = !showUserMenu.value
|
|
|
|
|
showMessagePanel.value = false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleMenuClick = (item) => {
|
|
|
|
|
// 检查标签页是否已存在
|
|
|
|
|
const existingTab = openTabs.value.find(tab => tab.id === item.id)
|
|
|
|
|
if (!existingTab) {
|
|
|
|
|
openTabs.value.push({
|
|
|
|
|
id: item.id,
|
|
|
|
|
title: item.name,
|
|
|
|
|
path: item.path,
|
|
|
|
|
closable: true
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
currentTab.value = item.id
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const switchTab = (tabId) => {
|
|
|
|
|
currentTab.value = tabId
|
|
|
|
|
const tab = openTabs.value.find(t => t.id === tabId)
|
|
|
|
|
if (tab) {
|
|
|
|
|
router.push(tab.path)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const closeTab = (tabId) => {
|
|
|
|
|
const index = openTabs.value.findIndex(tab => tab.id === tabId)
|
|
|
|
|
if (index > -1) {
|
|
|
|
|
openTabs.value.splice(index, 1)
|
|
|
|
|
// 如果关闭的是当前标签页,切换到前一个标签页
|
|
|
|
|
if (currentTab.value === tabId) {
|
|
|
|
|
const newTab = openTabs.value[index - 1] || openTabs.value[0]
|
|
|
|
|
if (newTab) {
|
|
|
|
|
currentTab.value = newTab.id
|
|
|
|
|
router.push(newTab.path)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const closeAllTabs = () => {
|
|
|
|
|
openTabs.value = openTabs.value.filter(tab => !tab.closable)
|
|
|
|
|
currentTab.value = 'home'
|
|
|
|
|
router.push('/')
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const toggleFavorite = (itemId) => {
|
|
|
|
|
const item = mainMenuItems.value.find(i => i.id === itemId)
|
|
|
|
|
if (item) {
|
|
|
|
|
item.favorite = !item.favorite
|
|
|
|
|
updateFavoriteMenu()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const updateFavoriteMenu = () => {
|
|
|
|
|
favoriteMenuItems.value = mainMenuItems.value.filter(item => item.favorite)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const markAsRead = (messageId) => {
|
|
|
|
|
const message = messages.value.find(m => m.id === messageId)
|
|
|
|
|
if (message) {
|
|
|
|
|
message.read = true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const formatTime = (time) => {
|
|
|
|
|
const now = new Date()
|
|
|
|
|
const diff = now - time
|
|
|
|
|
const minutes = Math.floor(diff / (1000 * 60))
|
|
|
|
|
const hours = Math.floor(diff / (1000 * 60 * 60))
|
|
|
|
|
const days = Math.floor(diff / (1000 * 60 * 60 * 24))
|
|
|
|
|
|
|
|
|
|
if (minutes < 60) return `${minutes}分钟前`
|
|
|
|
|
if (hours < 24) return `${hours}小时前`
|
|
|
|
|
return `${days}天前`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const openProfile = () => {
|
|
|
|
|
showUserMenu.value = false
|
|
|
|
|
// 跳转到个人资料页面
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const openSettings = () => {
|
|
|
|
|
showUserMenu.value = false
|
|
|
|
|
router.push('/settings')
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const logout = async () => {
|
|
|
|
|
try {
|
|
|
|
|
await userService.logout()
|
|
|
|
|
router.push('/login')
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error('退出登录失败:', error)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 监听路由变化
|
|
|
|
|
watch(() => route.path, (newPath) => {
|
|
|
|
|
const tab = openTabs.value.find(t => t.path === newPath)
|
|
|
|
|
if (tab) {
|
|
|
|
|
currentTab.value = tab.id
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
onMounted(() => {
|
|
|
|
|
// 初始化收藏菜单
|
|
|
|
|
updateFavoriteMenu()
|
|
|
|
|
|
|
|
|
|
// 加载当前用户信息
|
|
|
|
|
const savedUser = localStorage.getItem('user')
|
|
|
|
|
if (savedUser) {
|
|
|
|
|
Object.assign(currentUser, JSON.parse(savedUser))
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
showMessagePanel,
|
|
|
|
|
showUserMenu,
|
|
|
|
|
currentTab,
|
|
|
|
|
openTabs,
|
|
|
|
|
currentUser,
|
|
|
|
|
messages,
|
|
|
|
|
mainMenuItems,
|
|
|
|
|
favoriteMenuItems,
|
|
|
|
|
unreadCount,
|
|
|
|
|
currentRoute,
|
|
|
|
|
breadcrumbs,
|
|
|
|
|
toggleMessagePanel,
|
|
|
|
|
toggleUserMenu,
|
|
|
|
|
handleMenuClick,
|
|
|
|
|
switchTab,
|
|
|
|
|
closeTab,
|
|
|
|
|
closeAllTabs,
|
|
|
|
|
toggleFavorite,
|
|
|
|
|
markAsRead,
|
|
|
|
|
formatTime,
|
|
|
|
|
openProfile,
|
|
|
|
|
openSettings,
|
|
|
|
|
logout
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
.main-layout {
|
|
|
|
|
height: calc(100vh - 24px); /* 减去状态条高度 */
|
|
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
background: #f5f5f5;
|
|
|
|
|
/* 添加中文字体支持 */
|
|
|
|
|
font-family: 'Microsoft YaHei', 'PingFang SC', 'Hiragino Sans GB', 'WenQuanYi Micro Hei', Avenir, Helvetica, Arial, sans-serif;
|
|
|
|
|
/* 确保不会溢出 */
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 顶部导航栏 */
|
|
|
|
|
.header {
|
|
|
|
|
height: 60px;
|
|
|
|
|
background: white;
|
|
|
|
|
border-bottom: 1px solid #e0e0e0;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
padding: 0 20px;
|
|
|
|
|
box-shadow: 0 2px 4px rgba(0,0,0,0.1);
|
|
|
|
|
z-index: 100;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.header-left {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 20px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo h1 {
|
|
|
|
|
margin: 0;
|
|
|
|
|
font-size: 24px;
|
|
|
|
|
color: #1976d2;
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breadcrumb {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 8px;
|
|
|
|
|
color: #666;
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breadcrumb-item {
|
|
|
|
|
color: #333;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.separator {
|
|
|
|
|
color: #ccc;
|
|
|
|
|
margin: 0 4px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.header-right {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 20px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 消息中心 */
|
|
|
|
|
.message-center {
|
|
|
|
|
position: relative;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-btn {
|
|
|
|
|
background: none;
|
|
|
|
|
border: none;
|
|
|
|
|
font-size: 16px; /* 从20px缩小到16px(缩小一半) */
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
position: relative;
|
|
|
|
|
padding: 8px;
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
transition: background-color 0.2s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-btn:hover {
|
|
|
|
|
background: #f5f5f5;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.badge {
|
|
|
|
|
position: absolute;
|
|
|
|
|
top: 0;
|
|
|
|
|
right: 0;
|
|
|
|
|
background: #f44336;
|
|
|
|
|
color: white;
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
width: 18px;
|
|
|
|
|
height: 18px;
|
|
|
|
|
font-size: 11px;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-panel {
|
|
|
|
|
position: absolute;
|
|
|
|
|
top: 100%;
|
|
|
|
|
right: 0;
|
|
|
|
|
width: 350px;
|
|
|
|
|
background: white;
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
box-shadow: 0 4px 12px rgba(0,0,0,0.15);
|
|
|
|
|
z-index: 1000;
|
|
|
|
|
margin-top: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-header {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
align-items: center;
|
|
|
|
|
padding: 16px;
|
|
|
|
|
border-bottom: 1px solid #eee;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-header h3 {
|
|
|
|
|
margin: 0;
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.close-btn {
|
|
|
|
|
background: none;
|
|
|
|
|
border: none;
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
color: #999;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-list {
|
|
|
|
|
max-height: 400px;
|
|
|
|
|
overflow-y: auto;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-item {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
padding: 12px 16px;
|
|
|
|
|
border-bottom: 1px solid #f5f5f5;
|
|
|
|
|
transition: background-color 0.2s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-item:hover {
|
|
|
|
|
background: #f9f9f9;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-icon {
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
margin-right: 12px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-content {
|
|
|
|
|
flex: 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-title {
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
margin-bottom: 4px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.message-time {
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
color: #999;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.mark-read-btn {
|
|
|
|
|
background: none;
|
|
|
|
|
border: none;
|
|
|
|
|
color: #4caf50;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
padding: 4px;
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.mark-read-btn:hover {
|
|
|
|
|
background: #f0f8f0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 用户信息 */
|
|
|
|
|
.user-info {
|
|
|
|
|
position: relative;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-avatar {
|
|
|
|
|
width: 40px;
|
|
|
|
|
height: 40px;
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
background: #1976d2;
|
|
|
|
|
color: white;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
transition: background-color 0.2s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-avatar:hover {
|
|
|
|
|
background: #1565c0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-avatar img {
|
|
|
|
|
width: 100%;
|
|
|
|
|
height: 100%;
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
object-fit: cover;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.avatar-placeholder {
|
|
|
|
|
font-size: 18px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-menu {
|
|
|
|
|
position: absolute;
|
|
|
|
|
top: 100%;
|
|
|
|
|
right: 0;
|
|
|
|
|
width: 250px;
|
|
|
|
|
background: white;
|
|
|
|
|
border-radius: 8px;
|
|
|
|
|
box-shadow: 0 4px 12px rgba(0,0,0,0.15);
|
|
|
|
|
z-index: 1000;
|
|
|
|
|
margin-top: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-menu-header {
|
|
|
|
|
padding: 16px;
|
|
|
|
|
border-bottom: 1px solid #eee;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-name {
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
margin-bottom: 4px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-email {
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
color: #666;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-menu-items {
|
|
|
|
|
padding: 8px 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-item {
|
|
|
|
|
width: 100%;
|
|
|
|
|
background: none;
|
|
|
|
|
border: none;
|
|
|
|
|
padding: 12px 16px;
|
|
|
|
|
text-align: left;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 12px;
|
|
|
|
|
transition: background-color 0.2s;
|
|
|
|
|
font-size: 13px; /* 从默认14px缩小到13px */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.menu-item:hover {
|
|
|
|
|
background: #f5f5f5;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 主要内容区域 */
|
|
|
|
|
.main-content {
|
|
|
|
|
flex: 1;
|
|
|
|
|
display: flex;
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
/* 高度由flex自动计算 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 左侧菜单 */
|
|
|
|
|
.sidebar {
|
|
|
|
|
width: 250px;
|
|
|
|
|
background: white;
|
|
|
|
|
border-right: 1px solid #e0e0e0;
|
|
|
|
|
overflow-y: auto;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.sidebar-nav {
|
|
|
|
|
padding: 20px 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-section {
|
|
|
|
|
margin-bottom: 20px; /* 从30px减少到20px,因为删除了标题 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-section:last-child {
|
|
|
|
|
margin-bottom: 0; /* 最后一个section不需要底部间距 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-title {
|
|
|
|
|
padding: 0 20px;
|
|
|
|
|
margin-bottom: 12px;
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
color: #999;
|
|
|
|
|
text-transform: none; /* 取消大写转换 */
|
|
|
|
|
letter-spacing: normal; /* 取消字母间距 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-list {
|
|
|
|
|
list-style: none;
|
|
|
|
|
padding: 0;
|
|
|
|
|
margin: 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-item {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
padding: 10px 20px; /* 从12px减少到10px,因为字体变小了 */
|
|
|
|
|
color: #333;
|
|
|
|
|
text-decoration: none;
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
position: relative;
|
|
|
|
|
font-size: 13px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-item:hover {
|
|
|
|
|
background: #f5f5f5;
|
|
|
|
|
color: #1976d2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-item.active {
|
|
|
|
|
background: #e3f2fd;
|
|
|
|
|
color: #1976d2;
|
|
|
|
|
border-right: 3px solid #1976d2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-icon {
|
|
|
|
|
font-size: 12px; /* 从18px缩小到12px(缩小三分之一) */
|
|
|
|
|
margin-right: 12px;
|
|
|
|
|
width: 16px; /* 从20px缩小到16px */
|
|
|
|
|
text-align: center;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.nav-text {
|
|
|
|
|
flex: 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.favorite-btn {
|
|
|
|
|
background: none;
|
|
|
|
|
border: none;
|
|
|
|
|
font-size: 12px; /* 从14px缩小到12px,与图标保持一致 */
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
opacity: 0.3;
|
|
|
|
|
transition: opacity 0.2s;
|
|
|
|
|
padding: 2px; /* 添加内边距 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.favorite-btn:hover,
|
|
|
|
|
.favorite-btn.active {
|
|
|
|
|
opacity: 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 右侧内容区域 */
|
|
|
|
|
.content-area {
|
|
|
|
|
flex: 1;
|
|
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
overflow: hidden; /* 隐藏所有滚动条 */
|
|
|
|
|
/* 确保高度正确计算 */
|
|
|
|
|
height: 100%;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 内容选项卡 */
|
|
|
|
|
.content-tabs {
|
|
|
|
|
background: #f8f9fa;
|
|
|
|
|
border-bottom: 1px solid #dee2e6;
|
|
|
|
|
padding: 0;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
height: 32px;
|
|
|
|
|
overflow-x: auto;
|
|
|
|
|
overflow-y: hidden; /* 确保垂直方向不滚动 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-list {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
flex: 1;
|
|
|
|
|
overflow-x: auto;
|
|
|
|
|
overflow-y: hidden; /* 确保垂直方向不滚动 */
|
|
|
|
|
height: 100%; /* 限制高度 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-item {
|
|
|
|
|
padding: 6px 16px;
|
|
|
|
|
background: transparent;
|
|
|
|
|
border: none;
|
|
|
|
|
border-right: 1px solid #dee2e6;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
font-size: 13px;
|
|
|
|
|
color: #6c757d;
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
gap: 6px;
|
|
|
|
|
height: 100%; /* 确保高度填满容器 */
|
|
|
|
|
min-width: 120px;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
position: relative;
|
|
|
|
|
box-sizing: border-box; /* 确保padding不会增加总高度 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-item:hover {
|
|
|
|
|
background: #e9ecef;
|
|
|
|
|
color: #495057;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-item.active {
|
|
|
|
|
background: white;
|
|
|
|
|
color: #007bff;
|
|
|
|
|
border-bottom: 2px solid #007bff;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-item .tab-close {
|
|
|
|
|
width: 16px;
|
|
|
|
|
height: 16px;
|
|
|
|
|
border: none;
|
|
|
|
|
background: transparent;
|
|
|
|
|
color: #6c757d;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
margin-left: 4px;
|
|
|
|
|
border-radius: 0;
|
|
|
|
|
line-height: 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-item .tab-close:hover {
|
|
|
|
|
color: #dc3545;
|
|
|
|
|
background: transparent;
|
|
|
|
|
transform: none;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-title {
|
|
|
|
|
flex: 1;
|
|
|
|
|
text-align: center;
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-close {
|
|
|
|
|
background: none;
|
|
|
|
|
border: none;
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
color: #999;
|
|
|
|
|
padding: 2px;
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-close:hover {
|
|
|
|
|
background: #f0f0f0;
|
|
|
|
|
color: #666;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.tab-actions {
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: flex-end;
|
|
|
|
|
padding-right: 10px;
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
height: 100%; /* 确保高度一致 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.close-all-btn {
|
|
|
|
|
background: none;
|
|
|
|
|
border: none;
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
color: #999;
|
|
|
|
|
padding: 4px;
|
|
|
|
|
border-radius: 3px;
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
width: 24px;
|
|
|
|
|
height: 24px; /* 确保按钮高度合适 */
|
|
|
|
|
margin-left: 8px;
|
|
|
|
|
box-sizing: border-box; /* 确保padding不会增加总高度 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.close-all-btn:hover {
|
|
|
|
|
background: #e9ecef;
|
|
|
|
|
color: #495057;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.close-all-icon {
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
line-height: 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 功能内容区 */
|
|
|
|
|
.content-body {
|
|
|
|
|
flex: 1;
|
|
|
|
|
overflow-y: auto; /* 只有内容区有垂直滚动条 */
|
|
|
|
|
overflow-x: hidden; /* 隐藏水平滚动条 */
|
|
|
|
|
background: white;
|
|
|
|
|
position: relative;
|
|
|
|
|
/* 高度由flex自动计算 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 动态滚动条样式 */
|
|
|
|
|
.content-body::-webkit-scrollbar {
|
|
|
|
|
width: 8px;
|
|
|
|
|
height: 8px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-body::-webkit-scrollbar-track {
|
|
|
|
|
background: transparent;
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-body::-webkit-scrollbar-thumb {
|
|
|
|
|
background: rgba(0, 0, 0, 0.2);
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
transition: background 0.3s ease;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-body::-webkit-scrollbar-thumb:hover {
|
|
|
|
|
background: rgba(0, 0, 0, 0.4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-body::-webkit-scrollbar-corner {
|
|
|
|
|
background: transparent;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 选项卡区域的滚动条样式 */
|
|
|
|
|
.content-tabs::-webkit-scrollbar {
|
|
|
|
|
height: 4px; /* 只显示水平滚动条 */
|
|
|
|
|
width: 0; /* 隐藏垂直滚动条 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-tabs::-webkit-scrollbar-track {
|
|
|
|
|
background: transparent;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-tabs::-webkit-scrollbar-thumb {
|
|
|
|
|
background: rgba(0, 0, 0, 0.1);
|
|
|
|
|
border-radius: 2px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-tabs::-webkit-scrollbar-thumb:hover {
|
|
|
|
|
background: rgba(0, 0, 0, 0.2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 确保内容可以滚动 */
|
|
|
|
|
.content-body > * {
|
|
|
|
|
min-height: 100%;
|
|
|
|
|
/* 确保子元素不会创建额外的滚动条 */
|
|
|
|
|
overflow: visible;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 响应式设计 */
|
|
|
|
|
@media (max-width: 768px) {
|
|
|
|
|
.sidebar {
|
|
|
|
|
width: 200px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.header {
|
|
|
|
|
padding: 0 15px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.logo h1 {
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.breadcrumb {
|
|
|
|
|
display: none;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
</style>
|