Skip to content

Commit e72e2ed

Browse files
zk020106Charles7c
authored andcommitted
feat(layout): 混合布局和双列布局 补充版权信息组件和弹窗组件
Co-authored-by: kiki1373639299<zkai0106@163.com> # message auto-generated for no-merge-commit merge: !15 merge layout-fix into dev feat(layout): 混合布局和双列布局 补充版权信息组件和弹窗组件 Created-by: kiki1373639299 Commit-by: kiki1373639299 Merged-by: Charles_7c Description: <!-- 非常感谢您的 PR!在提交之前,请务必确保您 PR 的代码经过了完整测试,并且通过了代码规范检查。 --> <!-- 在 [] 中输入 x 来勾选) --> ## PR 类型 <!-- 您的 PR 引入了哪种类型的变更? --> <!-- 只支持选择一种类型,如果有多种类型,可以在更新日志中增加 “类型” 列。 --> - [X] 新 feature - [ ] Bug 修复 - [ ] 功能增强 - [ ] 文档变更 - [ ] 代码样式变更 - [ ] 重构 - [ ] 性能改进 - [ ] 单元测试 - [ ] CI/CD - [ ] 其他 ## PR 目的 <!-- 描述一下您的 PR 解决了什么问题。如果可以,请链接到相关 issues。 --> ## 解决方案 <!-- 详细描述您是如何解决的问题 --> ## PR 测试 <!-- 如果可以,请为您的 PR 添加或更新单元测试。 --> <!-- 请描述一下您是如何测试 PR 的。例如:创建/更新单元测试或添加相关的截图。 --> ## Changelog | 模块 | Changelog | Related issues | |-----|-----------| -------------- | | src/layout | layoutMix 和 layoutColumns 增加 GiFooter 和 NoticePopup | | <!-- 如果有多种类型的变更,可以在变更日志表中增加 “类型” 列,该列的值与上方 “PR 类型” 相同。 --> <!-- Related issues 格式为 Closes #<issue号>,或者 Fixes #<issue号>,或者 Resolves #<issue号>。 --> ## 其他信息 <!-- 请描述一下还有哪些注意事项。例如:如果引入了一个不向下兼容的变更,请描述其影响。 --> ## 提交前确认 - [X] PR 代码经过了完整测试,并且通过了代码规范检查 - [X] 已经完整填写 Changelog,并链接到了相关 issues - [X] PR 代码将要提交到 dev 分支 See merge request: continew/continew-admin-ui!15
1 parent e5c9d2f commit e72e2ed

File tree

4 files changed

+95
-14
lines changed

4 files changed

+95
-14
lines changed

src/layout/LayoutColumns.vue

Lines changed: 63 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -5,21 +5,27 @@
55
<OneLevelMenu :menus="oneLevelMenus" @menu-click="handleMenuItemClickByItem"></OneLevelMenu>
66

77
<!-- 左侧二级菜单区域 -->
8-
<div class="layout-columns__right-menu">
9-
<div class="layout-columns__system-name gi_line_1">{{ appStore.getTitle() }}</div>
8+
<div class="layout-columns__right-menu" :class="{ collapse: appStore.menuCollapse }">
9+
<!-- 系统标题 -->
10+
<div class="layout-columns__title">
11+
<span v-show="!appStore.menuCollapse" class="system-name gi_line_1">{{ appStore.getTitle() }}</span>
12+
</div>
1013
<Menu
1114
v-if="twoLevelMenus.length > 1 || oneLevelMenuActiveRoute?.meta?.alwaysShow === true"
12-
class="layout-columns__menu" :menus="twoLevelMenus" :menu-style="{ width: '180px' }"
15+
class="layout-columns__menu" :menus="twoLevelMenus" :menu-style="menuStyle"
1316
/>
1417
</div>
1518
</div>
1619

17-
<!-- 右侧内容区域 -->
1820
<section class="layout-columns__content">
1921
<Header />
2022
<Tabs v-if="appStore.tab" />
2123
<Main />
24+
<GiFooter v-if="appStore.copyrightDisplay" />
2225
</section>
26+
27+
<!-- 公告弹窗 -->
28+
<NoticePopup ref="noticePopupRef" />
2329
</div>
2430
</template>
2531

@@ -32,6 +38,8 @@ import Tabs from './components/Tabs/index.vue'
3238
import { useAppStore } from '@/stores'
3339
import { useLevelMenu } from '@/layout/hooks/useLevelMenu'
3440
import { useDevice } from '@/hooks'
41+
import NoticePopup from '@/views/user/message/components/NoticePopup.vue'
42+
import { getToken } from '@/utils/auth'
3543
3644
defineOptions({ name: 'LayoutColumns' })
3745
@@ -40,6 +48,29 @@ const { isDesktop } = useDevice()
4048
4149
const { oneLevelMenus, twoLevelMenus, oneLevelMenuActiveRoute, getOneLevelMenus, handleMenuItemClickByItem } = useLevelMenu()
4250
getOneLevelMenus()
51+
// 菜单样式 - 根据折叠状态动态调整宽度
52+
const menuStyle = computed(() => {
53+
return {
54+
width: appStore.menuCollapse ? '48px' : '200px',
55+
}
56+
})
57+
// 公告弹窗引用
58+
const noticePopupRef = ref<InstanceType<typeof NoticePopup>>()
59+
60+
// 检查并显示未读公告
61+
const checkAndShowNotices = () => {
62+
const token = getToken()
63+
64+
// 如果有token,检查未读公告
65+
if (token) {
66+
setTimeout(() => {
67+
noticePopupRef.value?.open()
68+
}, 1000) // 延迟1秒显示,让页面先加载完成
69+
}
70+
}
71+
onMounted(() => {
72+
checkAndShowNotices()
73+
})
4374
</script>
4475

4576
<style lang="scss" scoped>
@@ -59,12 +90,16 @@ getOneLevelMenus()
5990
&__right-menu {
6091
display: flex;
6192
flex-direction: column;
62-
width: 180px;
6393
overflow: hidden;
6494
background-color: var(--color-bg-1);
95+
width: 200px;
96+
transition: width 0.3s ease-in-out;
97+
&.collapse {
98+
width: 48px;
99+
}
65100
}
66101
67-
&__system-name {
102+
&__title {
68103
height: 56px;
69104
padding: 0 12px;
70105
color: var(--color-text-1);
@@ -78,17 +113,35 @@ getOneLevelMenus()
78113
border-bottom: 1px solid var(--color-border);
79114
cursor: pointer;
80115
user-select: none;
81-
transition: padding .2s;
116+
transition: all 0.3s ease-in-out;
117+
118+
.system-name {
119+
padding-left: 6px;
120+
white-space: nowrap;
121+
transition: all 0.3s ease-in-out;
122+
line-height: 1.5;
123+
display: inline-flex;
124+
align-items: center;
125+
opacity: 1;
126+
transform: translateX(0);
127+
128+
&:hover {
129+
color: $color-theme !important;
130+
cursor: pointer;
131+
}
82132
83-
&:hover {
84-
color: $color-theme !important;
85-
cursor: pointer;
133+
.layout-columns__right-menu.collapse & {
134+
opacity: 0;
135+
transform: translateX(-20px);
136+
}
86137
}
87138
}
88139
&__menu {
89140
flex: 1;
90141
overflow: auto;
91142
border-right: 1px solid var(--color-border-2);
143+
transition: all 0.3s ease-in-out;
144+
width: 100%;
92145
}
93146
94147
&__content {

src/layout/LayoutMix.vue

Lines changed: 27 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
>
1212
<Logo :collapsed="appStore.menuCollapse" />
1313
<Menu :menus="twoLevelMenus" :menu-style="{ width: '200px', flex: 1 }" />
14+
<WwAds class="ads" />
1415
</section>
1516

1617
<!-- 右侧内容区域 -->
@@ -32,7 +33,11 @@
3233
</header>
3334
<Tabs v-if="appStore.tab" />
3435
<Main />
36+
<GiFooter v-if="appStore.copyrightDisplay" />
3537
</section>
38+
39+
<!-- 公告弹窗 -->
40+
<NoticePopup ref="noticePopupRef" />
3641
</div>
3742
</template>
3843

@@ -47,6 +52,10 @@ import Tabs from './components/Tabs/index.vue'
4752
import { useAppStore } from '@/stores'
4853
import { useLevelMenu } from '@/layout/hooks/useLevelMenu'
4954
import { useDevice } from '@/hooks'
55+
import { getToken } from '@/utils/auth'
56+
57+
import WwAds from '@/layout/components/WwAds.vue'
58+
import NoticePopup from '@/views/user/message/components/NoticePopup.vue'
5059
5160
/** 组件名称 */
5261
defineOptions({ name: 'LayoutMix' })
@@ -64,9 +73,27 @@ getOneLevelMenus()
6473
6574
const activeMenu = computed(() => [oneLevelMenuActiveRoute.value?.path ?? ''])
6675
76+
// 公告弹窗引用
77+
const noticePopupRef = ref<InstanceType<typeof NoticePopup>>()
78+
79+
// 检查并显示未读公告
80+
const checkAndShowNotices = () => {
81+
const token = getToken()
82+
83+
// 如果有token,检查未读公告
84+
if (token) {
85+
setTimeout(() => {
86+
console.log(noticePopupRef.value)
87+
noticePopupRef.value?.open()
88+
}, 1000) // 延迟1秒显示,让页面先加载完成
89+
}
90+
}
6791
const getMenuIcon = (item: RouteRecordRaw) => {
6892
return item.meta?.icon || item.children?.[0]?.meta?.icon
6993
}
94+
onMounted(() => {
95+
checkAndShowNotices()
96+
})
7097
</script>
7198

7299
<style lang="scss" scoped>

src/layout/hooks/useLevelMenu.ts

Lines changed: 4 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import type { RouteRecordRaw } from 'vue-router'
2-
import { eachTree } from 'xe-utils'
2+
import { eachTree, searchTree } from 'xe-utils'
33
import { useRouteListener } from '@/hooks'
44
import { useRouteStore } from '@/stores'
55
import { filterTree } from '@/utils'
@@ -29,11 +29,12 @@ export function useLevelMenu() {
2929
const oneLevelMenuActiveRoute = ref<RouteRecordRaw | null>(null)
3030

3131
const getOneLevelMenuActiveRoute = (path: string) => {
32-
return oneLevelMenus.value.find((i) => i.path === path) as RouteRecordRaw
32+
const arr = searchTree(showMenuList, (i) => i.path === path)
33+
return arr?.[0]
3334
}
3435

3536
listenerRouteChange(({ to }) => {
36-
oneLevelMenuActiveRoute.value = getOneLevelMenuActiveRoute(to.matched?.[0]?.path)
37+
oneLevelMenuActiveRoute.value = getOneLevelMenuActiveRoute(to.path)
3738
})
3839

3940
// 获取一级菜单

src/views/user/message/components/NoticePopup.vue

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,7 @@ const fetchNoticeDetail = async (index: number) => {
126126
console.error(`获取公告详情失败:`, error)
127127
// 创建一个错误状态的公告对象
128128
noticeCache.value.set(noticeId, {
129-
id: noticeId,
129+
id: noticeId || '',
130130
title: '获取公告失败',
131131
content: '获取公告内容失败,请稍后重试',
132132
createUserString: '',

0 commit comments

Comments
 (0)