2025-08-03 12:07:18 +08:00
|
|
|
|
<template>
|
2025-08-09 22:23:24 +08:00
|
|
|
|
<div class="layout-container">
|
|
|
|
|
<!-- 背景装饰元素 -->
|
|
|
|
|
<div class="decoration-circle circle-1"></div>
|
|
|
|
|
<div class="decoration-circle circle-2"></div>
|
|
|
|
|
<div class="decoration-wave"></div>
|
|
|
|
|
|
|
|
|
|
<!-- 固定顶部导航栏 -->
|
|
|
|
|
<div class="fixed-header">
|
|
|
|
|
<div class="header-content">
|
|
|
|
|
<div class="logo-section">
|
|
|
|
|
<h1 class="platform-name">研发交付平台</h1>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="main-nav-section">
|
|
|
|
|
<a-menu mode="horizontal" :default-selected-keys="['1']">
|
|
|
|
|
<a-menu-item key="1">工作台</a-menu-item>
|
|
|
|
|
<a-menu-item key="2">项目管理</a-menu-item>
|
|
|
|
|
<a-menu-item key="3">研发交付</a-menu-item>
|
|
|
|
|
<a-menu-item key="4">制品库</a-menu-item>
|
|
|
|
|
<a-menu-item key="5">测试中心</a-menu-item>
|
|
|
|
|
<a-menu-item key="6">运维中心</a-menu-item>
|
|
|
|
|
</a-menu>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="user-section">
|
|
|
|
|
<a-dropdown position="bottom">
|
|
|
|
|
<a-avatar :size="32" class="user-avatar">
|
|
|
|
|
<icon-user />
|
|
|
|
|
</a-avatar>
|
|
|
|
|
<template #content>
|
|
|
|
|
<a-doption>个人中心</a-doption>
|
|
|
|
|
<a-doption>系统设置</a-doption>
|
|
|
|
|
<a-doption>退出登录</a-doption>
|
|
|
|
|
</template>
|
|
|
|
|
</a-dropdown>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- 主内容区 -->
|
|
|
|
|
<div class="main-content-wrapper">
|
|
|
|
|
<!-- 可收缩侧边栏 -->
|
|
|
|
|
<div class="fixed-sidebar" :class="{ 'collapsed': isSidebarCollapsed }">
|
|
|
|
|
<!-- 使用原生a-menu确保折叠效果 -->
|
|
|
|
|
<a-menu :theme="light" :default-selected-keys="['1']" :collapsed="isSidebarCollapsed" :collapsed-width="64"
|
|
|
|
|
:style="{ width: '100%', height: '100%' }">
|
|
|
|
|
<a-menu-item key="1">
|
|
|
|
|
<template #icon><icon-dashboard /></template>
|
|
|
|
|
<span class="menu-title">流水线列表</span>
|
|
|
|
|
</a-menu-item>
|
|
|
|
|
<a-menu-item key="2">
|
|
|
|
|
<template #icon><icon-branch /></template>
|
|
|
|
|
<span class="menu-title">分支管理</span>
|
|
|
|
|
</a-menu-item>
|
|
|
|
|
<a-menu-item key="3">
|
|
|
|
|
<template #icon><icon-history /></template>
|
|
|
|
|
<span class="menu-title">执行历史</span>
|
|
|
|
|
</a-menu-item>
|
|
|
|
|
<a-menu-item key="4">
|
|
|
|
|
<template #icon><icon-settings /></template>
|
|
|
|
|
<span class="menu-title">流水线模板</span>
|
|
|
|
|
</a-menu-item>
|
|
|
|
|
<a-menu-item key="5">
|
|
|
|
|
<template #icon><icon-monitor /></template>
|
|
|
|
|
<span class="menu-title">监控中心</span>
|
|
|
|
|
</a-menu-item>
|
|
|
|
|
</a-menu>
|
|
|
|
|
|
|
|
|
|
<!-- 自定义收缩按钮 -->
|
|
|
|
|
<div class="collapse-btn-wrapper">
|
|
|
|
|
<a-button class="collapse-btn" @click="toggleSidebar">
|
|
|
|
|
<template #icon>
|
|
|
|
|
<icon-left v-if="!isSidebarCollapsed" />
|
|
|
|
|
<icon-right v-else />
|
|
|
|
|
</template>
|
|
|
|
|
</a-button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- 内容滚动区域(关键修改) -->
|
|
|
|
|
<div class="scrollable-content" :class="{ 'expanded': isSidebarCollapsed }">
|
|
|
|
|
<div class="content-scroll-wrapper">
|
|
|
|
|
<!-- 面包屑(会随内容滚动) -->
|
|
|
|
|
<div class="breadcrumb">
|
|
|
|
|
<a-breadcrumb>
|
|
|
|
|
<a-breadcrumb-item>首页</a-breadcrumb-item>
|
|
|
|
|
<a-breadcrumb-item>流水线系统</a-breadcrumb-item>
|
|
|
|
|
<a-breadcrumb-item>流水线列表</a-breadcrumb-item>
|
|
|
|
|
</a-breadcrumb>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- 主内容区 -->
|
|
|
|
|
<main class="router-view-wrapper">
|
|
|
|
|
<router-view />
|
|
|
|
|
</main>
|
|
|
|
|
|
|
|
|
|
<!-- 页脚(会随内容滚动) -->
|
|
|
|
|
<footer class="layout-footer">
|
|
|
|
|
<p>© 2025 研发交付平台 · 让软件交付更高效</p>
|
|
|
|
|
</footer>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2025-08-03 12:07:18 +08:00
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script setup>
|
2025-08-09 22:23:24 +08:00
|
|
|
|
import { ref } from 'vue';
|
|
|
|
|
|
|
|
|
|
const isSidebarCollapsed = ref(false);
|
2025-08-03 12:07:18 +08:00
|
|
|
|
|
2025-08-09 22:23:24 +08:00
|
|
|
|
const toggleSidebar = () => {
|
|
|
|
|
isSidebarCollapsed.value = !isSidebarCollapsed.value;
|
|
|
|
|
};
|
2025-08-03 12:07:18 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
2025-08-09 22:23:24 +08:00
|
|
|
|
<style lang="less" scoped>
|
|
|
|
|
/* 基础布局 */
|
|
|
|
|
.layout-container {
|
|
|
|
|
min-height: 100vh;
|
|
|
|
|
background-color: #f5f7fa;
|
|
|
|
|
position: relative;
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
|
|
|
|
.decoration-circle {
|
|
|
|
|
position: fixed;
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
background: linear-gradient(135deg, rgba(24, 144, 255, 0.1) 0%, rgba(24, 144, 255, 0.05) 100%);
|
|
|
|
|
z-index: 0;
|
|
|
|
|
|
|
|
|
|
&.circle-1 {
|
|
|
|
|
width: 600px;
|
|
|
|
|
height: 600px;
|
|
|
|
|
top: -200px;
|
|
|
|
|
left: -200px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&.circle-2 {
|
|
|
|
|
width: 400px;
|
|
|
|
|
height: 400px;
|
|
|
|
|
bottom: -100px;
|
|
|
|
|
right: -100px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.decoration-wave {
|
|
|
|
|
position: fixed;
|
|
|
|
|
bottom: 0;
|
|
|
|
|
left: 0;
|
|
|
|
|
width: 100%;
|
|
|
|
|
height: 70px;
|
|
|
|
|
background: url('@/assets/svg/wave.svg') repeat-x;
|
|
|
|
|
background-size: 1000px 100px;
|
|
|
|
|
opacity: 0.1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 顶部导航 */
|
|
|
|
|
.fixed-header {
|
|
|
|
|
position: fixed;
|
|
|
|
|
top: 0;
|
|
|
|
|
left: 0;
|
|
|
|
|
right: 0;
|
|
|
|
|
height: 60px;
|
|
|
|
|
background-color: #fff;
|
|
|
|
|
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.05);
|
|
|
|
|
z-index: 100;
|
|
|
|
|
|
|
|
|
|
.header-content {
|
|
|
|
|
max-width: 100%;
|
|
|
|
|
height: 100%;
|
|
|
|
|
padding: 0 24px;
|
|
|
|
|
margin: 0 auto;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
|
|
|
|
.logo-section {
|
|
|
|
|
.platform-name {
|
|
|
|
|
color: var(--color-text-1);
|
|
|
|
|
font-size: 18px;
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
margin: 0;
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.main-nav-section {
|
|
|
|
|
flex: 1;
|
|
|
|
|
margin: 0 40px;
|
|
|
|
|
overflow-x: auto;
|
|
|
|
|
overflow-y: hidden;
|
|
|
|
|
|
|
|
|
|
:deep(.arco-menu-horizontal) {
|
|
|
|
|
height: 60px;
|
|
|
|
|
background: transparent;
|
|
|
|
|
border-bottom: none;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.user-avatar {
|
|
|
|
|
background-color: var(--color-primary-light-3);
|
|
|
|
|
color: var(--color-primary);
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
|
background-color: var(--color-primary-light-2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 主内容区 */
|
|
|
|
|
.main-content-wrapper {
|
|
|
|
|
display: flex;
|
|
|
|
|
padding-top: 60px;
|
|
|
|
|
height: calc(100vh - 60px);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 侧边栏 */
|
|
|
|
|
.fixed-sidebar {
|
|
|
|
|
position: fixed;
|
|
|
|
|
left: 0;
|
|
|
|
|
top: 60px;
|
|
|
|
|
padding: 12px 8px;
|
|
|
|
|
bottom: 0;
|
|
|
|
|
width: 220px;
|
|
|
|
|
background-color: var(--color-bg-2);
|
|
|
|
|
z-index: 90;
|
|
|
|
|
transition: width 0.3s ease;
|
|
|
|
|
|
|
|
|
|
&.collapsed {
|
|
|
|
|
width: 64px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.sidebar-header {
|
|
|
|
|
padding: 16px;
|
|
|
|
|
background-color: var(--color-primary-light-1);
|
|
|
|
|
height: 60px;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
position: relative;
|
|
|
|
|
|
|
|
|
|
h3 {
|
|
|
|
|
color: var(--color-text-1);
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
font-weight: 500;
|
|
|
|
|
margin: 0;
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 强制修正折叠菜单样式 */
|
|
|
|
|
:deep(.arco-menu) {
|
|
|
|
|
height: 100%;
|
|
|
|
|
border-right: none;
|
|
|
|
|
|
|
|
|
|
&.arco-menu-collapsed {
|
|
|
|
|
.arco-menu-item {
|
|
|
|
|
justify-content: center !important;
|
|
|
|
|
padding: 0 12px !important;
|
|
|
|
|
|
|
|
|
|
.arco-menu-icon {
|
|
|
|
|
margin: 0 auto !important;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.arco-menu-title {
|
|
|
|
|
display: inline-block;
|
|
|
|
|
width: 0;
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 自定义收缩按钮 */
|
|
|
|
|
.collapse-btn-wrapper {
|
|
|
|
|
position: absolute;
|
|
|
|
|
right: -12px;
|
|
|
|
|
top: 50%;
|
|
|
|
|
transform: translateY(-50%);
|
|
|
|
|
z-index: 110;
|
|
|
|
|
width: 24px;
|
|
|
|
|
height: 32px;
|
|
|
|
|
|
|
|
|
|
.collapse-btn {
|
|
|
|
|
width: 100%;
|
|
|
|
|
height: 100%;
|
|
|
|
|
display: flex;
|
|
|
|
|
align-items: center;
|
|
|
|
|
justify-content: center;
|
|
|
|
|
background: #fff;
|
|
|
|
|
border: 1px solid var(--color-border-2);
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
padding: 0;
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
|
|
|
|
// &:hover {
|
|
|
|
|
// background: var(--color-fill-2);
|
|
|
|
|
// border-color: var(--color-border-3);
|
|
|
|
|
// box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2);
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
&> :deep(svg) {
|
|
|
|
|
font-size: 14px;
|
|
|
|
|
color: var(--color-text-2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 内容滚动区域(关键修改) */
|
|
|
|
|
.scrollable-content {
|
|
|
|
|
flex: 1;
|
|
|
|
|
margin-left: 220px;
|
|
|
|
|
height: 100%;
|
|
|
|
|
overflow-y: scroll;
|
|
|
|
|
scrollbar-width: none;
|
|
|
|
|
/* Firefox */
|
|
|
|
|
-ms-overflow-style: none;
|
|
|
|
|
/* IE/Edge */
|
|
|
|
|
transition: margin-left 0.3s ease;
|
|
|
|
|
|
|
|
|
|
&::-webkit-scrollbar {
|
|
|
|
|
display: none;
|
|
|
|
|
/* Chrome/Safari */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
&.expanded {
|
|
|
|
|
margin-left: 64px;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.content-scroll-wrapper {
|
|
|
|
|
min-height: 100%;
|
|
|
|
|
display: flex;
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 面包屑 */
|
|
|
|
|
.breadcrumb {
|
|
|
|
|
padding: 16px 24px 8px;
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 路由视图容器 */
|
|
|
|
|
.router-view-wrapper {
|
|
|
|
|
flex: 1;
|
|
|
|
|
padding: 0 24px;
|
|
|
|
|
min-height: calc(100vh - 180px);
|
|
|
|
|
/* 动态计算最小高度 */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* 页脚 */
|
|
|
|
|
.layout-footer {
|
|
|
|
|
padding: 16px 24px;
|
|
|
|
|
margin-top: auto;
|
|
|
|
|
/* 关键:使页脚始终在底部 */
|
|
|
|
|
text-align: center;
|
|
|
|
|
color: var(--color-text-3);
|
|
|
|
|
font-size: 12px;
|
|
|
|
|
border-top: 1px solid var(--color-border-2);
|
|
|
|
|
background: var(--color-bg-2);
|
|
|
|
|
}
|
|
|
|
|
</style>
|