style(web): update app styles and AppShellRouteView

- app.css: update app-level styles
- AppShellRouteView.vue: update app shell route view
This commit is contained in:
caoxiaozhu
2026-05-14 03:00:56 +00:00
parent b9cb6d9253
commit 00b72c3d43
2 changed files with 303 additions and 106 deletions

View File

@@ -18,7 +18,7 @@
height: var(--desktop-stage-height, 100dvh); height: var(--desktop-stage-height, 100dvh);
min-height: var(--desktop-stage-height, 100dvh); min-height: var(--desktop-stage-height, 100dvh);
display: grid; display: grid;
grid-template-columns: 220px minmax(0, 1fr); grid-template-rows: auto auto minmax(0, 1fr);
background: var(--bg); background: var(--bg);
} }
@@ -85,17 +85,10 @@
.main { min-width: 0; min-height: 0; display: grid; grid-template-rows: auto auto minmax(0, 1fr); } .main { min-width: 0; min-height: 0; display: grid; grid-template-rows: auto auto minmax(0, 1fr); }
.main.overview-main { .main.overview-main {
height: var(--desktop-stage-height, 100dvh);
grid-template-rows: auto minmax(0, 1fr); grid-template-rows: auto minmax(0, 1fr);
overflow: hidden; overflow: hidden;
} }
.main.workbench-main { .main.workbench-main {
height: var(--desktop-stage-height, 100dvh);
grid-template-rows: auto minmax(0, 1fr);
overflow: hidden;
}
.main.chat-main {
height: var(--desktop-stage-height, 100dvh);
grid-template-rows: auto minmax(0, 1fr); grid-template-rows: auto minmax(0, 1fr);
overflow: hidden; overflow: hidden;
} }
@@ -105,7 +98,6 @@
.main.audit-main, .main.audit-main,
.main.employees-main, .main.employees-main,
.main.settings-main { .main.settings-main {
height: var(--desktop-stage-height, 100dvh);
grid-template-rows: auto minmax(0, 1fr); grid-template-rows: auto minmax(0, 1fr);
overflow: hidden; overflow: hidden;
} }
@@ -116,10 +108,6 @@
grid-template-rows: minmax(0, 1fr); grid-template-rows: minmax(0, 1fr);
} }
.workarea { min-height: 0; overflow: auto; padding: 24px; } .workarea { min-height: 0; overflow: auto; padding: 24px; }
.workarea.chat-workarea {
min-height: 0;
overflow: hidden;
}
.workarea.requests-workarea, .workarea.requests-workarea,
.workarea.approval-workarea, .workarea.approval-workarea,
.workarea.policies-workarea, .workarea.policies-workarea,
@@ -135,10 +123,10 @@
background: #fff; background: #fff;
} }
@media (max-width: 1180px) {
.app { grid-template-columns: 220px minmax(0, 1fr); }
}
@media (max-width: 760px) { @media (max-width: 760px) {
.app { display: block; } .app {
height: auto;
min-height: 100dvh;
}
.workarea { padding: 18px 16px 28px; } .workarea { padding: 18px 16px 28px; }
} }

View File

@@ -1,19 +1,54 @@
<template> <template>
<div class="app"> <div class="app">
<SidebarRail <header class="shell-header">
:nav-items="filteredNavItems" <div class="shell-brand">
:active-view="activeView" <span class="shell-brand-mark" aria-hidden="true">
:company-name="companyProfile.name" <svg viewBox="0 0 36 36">
:current-user="currentUser" <path d="M19.8 4.5c5.7 1.1 9.9 5.7 10.5 11.6-2.8-.9-5.5-.7-7.9.6-2.8 1.5-4.5 4.3-5.2 8.2-4.4-2.8-6.5-6.5-6.3-11.1.2-4.2 3.5-7.8 8.9-9.3Z" />
@navigate="handleNavigate" <path d="M9 7.6c-3 3.5-4 7.3-2.9 11.2 1.2 4.2 4.6 7 10.1 8.5-2 1.8-4.6 2.6-7.6 2.3C5.1 26.7 3.5 23.1 3.7 19 4 14.4 5.7 10.6 9 7.6Z" />
@open-chat="handleOpenChat" </svg>
@logout="handleLogout" </span>
/> <div class="shell-brand-copy">
<strong>{{ companyProfile.name || 'X-Financial' }}</strong>
<span>费用与报销运营平台</span>
</div>
</div>
<div class="shell-user-actions">
<div class="shell-user" aria-label="当前登录用户">
<span class="shell-user-avatar">{{ currentUser.avatar || '用' }}</span>
<span class="shell-user-copy">
<strong>{{ currentUser.name || '当前用户' }}</strong>
<span>{{ currentUser.role || '系统角色' }}</span>
</span>
</div>
<button class="shell-logout" type="button" @click="handleLogout">
<i class="mdi mdi-logout-variant"></i>
<span>退出登录</span>
</button>
</div>
</header>
<nav class="shell-nav" aria-label="主导航">
<div class="shell-nav-scroll">
<button
v-for="item in filteredNavItems"
:key="item.id"
class="shell-nav-btn"
:class="{ active: activeView === item.id }"
type="button"
@click="handleNavigate(item.id)"
>
<span class="shell-nav-icon" v-html="item.icon"></span>
<span>{{ item.label }}</span>
</button>
</div>
</nav>
<main <main
class="main" class="main"
:class="{ :class="{
'chat-main': activeView === 'chat',
'overview-main': activeView === 'overview', 'overview-main': activeView === 'overview',
'workbench-main': activeView === 'workbench', 'workbench-main': activeView === 'workbench',
'requests-main': activeView === 'requests', 'requests-main': activeView === 'requests',
@@ -40,12 +75,11 @@
@update:active-range="activeRange = $event" @update:active-range="activeRange = $event"
@update:custom-range="customRange = $event" @update:custom-range="customRange = $event"
@batch-approve="toast('已批量通过 23 条审批任务')" @batch-approve="toast('已批量通过 23 条审批任务')"
@open-chat="handleOpenChat"
@new-application="openTravelCreate" @new-application="openTravelCreate"
/> />
<FilterBar <FilterBar
v-if="activeView !== 'chat' && activeView !== 'overview' && activeView !== 'workbench' && activeView !== 'requests' && activeView !== 'approval' && activeView !== 'policies' && activeView !== 'audit' && activeView !== 'employees' && activeView !== 'settings'" v-if="activeView !== 'overview' && activeView !== 'workbench' && activeView !== 'requests' && activeView !== 'approval' && activeView !== 'policies' && activeView !== 'audit' && activeView !== 'employees' && activeView !== 'settings'"
:compact="activeView === 'overview'" :compact="activeView === 'overview'"
:filters="filters" :filters="filters"
:ranges="ranges" :ranges="ranges"
@@ -56,7 +90,6 @@
<section <section
class="workarea" class="workarea"
:class="{ :class="{
'chat-workarea': activeView === 'chat',
'requests-workarea': activeView === 'requests', 'requests-workarea': activeView === 'requests',
'approval-workarea': activeView === 'approval', 'approval-workarea': activeView === 'approval',
'policies-workarea': activeView === 'policies', 'policies-workarea': activeView === 'policies',
@@ -68,7 +101,6 @@
<OverviewView <OverviewView
v-if="activeView === 'overview'" v-if="activeView === 'overview'"
:filtered-requests="filteredRequests" :filtered-requests="filteredRequests"
@ask="handleOpenChat"
@approve="handleApprove" @approve="handleApprove"
@reject="handleReject" @reject="handleReject"
/> />
@@ -79,25 +111,6 @@
@open-assistant="openSmartEntry" @open-assistant="openSmartEntry"
/> />
<ChatView
v-else-if="activeView === 'chat'"
:documents="filteredDocuments"
:doc-search="docSearch"
:messages="messages"
:uploaded-files="uploadedFiles"
:active-case="activeCase"
:quick-prompts="travelPrompts"
:draft="draft"
:sending="sending"
:message-list="messageList"
@send="sendMessage"
@upload="handleUpload"
@draft="draft = $event"
@select-case="handleOpenChat"
@approve-case="toast(`${activeCase?.id || '当前单据'} 已标记为通过`)"
@reject-case="toast(`${activeCase?.id || '当前单据'} 已标记为驳回`)"
/>
<TravelRequestDetailView <TravelRequestDetailView
v-else-if="activeView === 'requests' && detailMode && selectedRequest" v-else-if="activeView === 'requests' && detailMode && selectedRequest"
:request="selectedRequest" :request="selectedRequest"
@@ -145,12 +158,10 @@
<script setup> <script setup>
import { computed, ref } from 'vue' import { computed, ref } from 'vue'
import SidebarRail from '../components/layout/SidebarRail.vue'
import TopBar from '../components/layout/TopBar.vue' import TopBar from '../components/layout/TopBar.vue'
import FilterBar from '../components/layout/FilterBar.vue' import FilterBar from '../components/layout/FilterBar.vue'
import OverviewView from './OverviewView.vue' import OverviewView from './OverviewView.vue'
import PersonalWorkbenchView from './PersonalWorkbenchView.vue' import PersonalWorkbenchView from './PersonalWorkbenchView.vue'
import ChatView from './ChatView.vue'
import TravelReimbursementCreateView from './TravelReimbursementCreateView.vue' import TravelReimbursementCreateView from './TravelReimbursementCreateView.vue'
import TravelRequestDetailView from './TravelRequestDetailView.vue' import TravelRequestDetailView from './TravelRequestDetailView.vue'
import RequestsView from './RequestsView.vue' import RequestsView from './RequestsView.vue'
@@ -184,13 +195,9 @@ const {
handleApprove, handleApprove,
handleDraftSaved, handleDraftSaved,
handleNavigate, handleNavigate,
handleOpenChat,
handleReject, handleReject,
handleRequestDeleted, handleRequestDeleted,
handleRequestUpdated, handleRequestUpdated,
handleUpload,
messageList,
messages,
navItems, navItems,
openRequestDetail, openRequestDetail,
openSmartEntry, openSmartEntry,
@@ -203,15 +210,11 @@ const {
requests, requests,
search, search,
selectedRequest, selectedRequest,
sendMessage,
sending,
smartEntryContext, smartEntryContext,
smartEntryOpen, smartEntryOpen,
smartEntrySessionId, smartEntrySessionId,
toast, toast,
topBarView, topBarView
travelPrompts,
uploadedFiles
} = useAppShell() } = useAppShell()
const { companyProfile, currentUser, logout } = useSystemState() const { companyProfile, currentUser, logout } = useSystemState()
@@ -221,3 +224,209 @@ function handleLogout() {
logout('manual') logout('manual')
} }
</script> </script>
<style scoped>
.shell-header {
display: flex;
align-items: center;
justify-content: space-between;
gap: 20px;
padding: 18px 24px 14px;
border-bottom: 1px solid #e2e8f0;
background:
linear-gradient(135deg, rgba(240, 253, 250, 0.9), rgba(255, 255, 255, 0.98)),
#fff;
}
.shell-brand {
display: flex;
align-items: center;
gap: 14px;
min-width: 0;
}
.shell-brand-mark {
width: 42px;
height: 42px;
display: grid;
place-items: center;
border-radius: 12px;
background: rgba(16, 185, 129, 0.1);
color: #059669;
flex: 0 0 auto;
}
.shell-brand-mark svg {
width: 24px;
height: 24px;
fill: currentColor;
}
.shell-brand-copy {
display: grid;
gap: 2px;
min-width: 0;
}
.shell-brand-copy strong {
color: #0f172a;
font-size: 18px;
font-weight: 800;
line-height: 1.1;
}
.shell-brand-copy span {
color: #64748b;
font-size: 13px;
font-weight: 600;
}
.shell-user-actions {
display: flex;
align-items: center;
gap: 12px;
flex-wrap: wrap;
justify-content: flex-end;
}
.shell-user {
display: inline-flex;
align-items: center;
gap: 10px;
min-height: 48px;
padding: 0 14px;
border: 1px solid #dbe4ee;
border-radius: 14px;
background: rgba(255, 255, 255, 0.88);
}
.shell-user-avatar {
width: 30px;
height: 30px;
display: grid;
place-items: center;
border-radius: 999px;
background: linear-gradient(135deg, #0f9f78, #34d399);
color: #fff;
font-size: 13px;
font-weight: 800;
}
.shell-user-copy {
display: grid;
gap: 1px;
}
.shell-user-copy strong {
color: #0f172a;
font-size: 13px;
font-weight: 750;
}
.shell-user-copy span {
color: #64748b;
font-size: 12px;
}
.shell-logout {
min-height: 44px;
display: inline-flex;
align-items: center;
gap: 8px;
padding: 0 14px;
border: 1px solid #dbe4ee;
border-radius: 12px;
background: #fff;
color: #334155;
font-size: 13px;
font-weight: 700;
transition: border-color 180ms ease, background 180ms ease, color 180ms ease;
}
.shell-logout:hover {
border-color: rgba(239, 68, 68, 0.2);
background: rgba(254, 242, 242, 0.9);
color: #dc2626;
}
.shell-nav {
padding: 12px 24px 0;
background: #fff;
}
.shell-nav-scroll {
display: flex;
align-items: center;
gap: 10px;
overflow-x: auto;
padding-bottom: 12px;
scrollbar-width: thin;
}
.shell-nav-btn {
min-height: 44px;
display: inline-flex;
align-items: center;
gap: 10px;
padding: 0 16px;
border: 1px solid #dbe4ee;
border-radius: 999px;
background: #fff;
color: #475569;
font-size: 13px;
font-weight: 700;
white-space: nowrap;
transition:
border-color 180ms ease,
background 180ms ease,
color 180ms ease,
box-shadow 180ms ease;
}
.shell-nav-btn:hover {
border-color: rgba(16, 185, 129, 0.25);
background: rgba(240, 253, 250, 0.96);
color: #059669;
}
.shell-nav-btn.active {
border-color: rgba(16, 185, 129, 0.2);
background: linear-gradient(135deg, rgba(16, 185, 129, 0.14), rgba(16, 185, 129, 0.06));
color: #047857;
box-shadow: 0 10px 24px rgba(16, 185, 129, 0.12);
}
.shell-nav-icon {
width: 18px;
height: 18px;
display: inline-grid;
place-items: center;
color: currentColor;
}
.shell-nav-btn :deep(svg) {
width: 18px;
height: 18px;
stroke: currentColor;
stroke-width: 2;
fill: none;
stroke-linecap: round;
stroke-linejoin: round;
}
@media (max-width: 760px) {
.shell-header {
flex-direction: column;
align-items: stretch;
padding: 16px 16px 12px;
}
.shell-user-actions {
justify-content: space-between;
}
.shell-nav {
padding: 10px 16px 0;
}
}
</style>