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
This commit is contained in:
kiki1373639299
2025-11-12 15:53:36 +08:00
committed by Charles_7c
parent e5c9d2f12f
commit e72e2ed1f8
4 changed files with 95 additions and 14 deletions

View File

@@ -5,21 +5,27 @@
<OneLevelMenu :menus="oneLevelMenus" @menu-click="handleMenuItemClickByItem"></OneLevelMenu> <OneLevelMenu :menus="oneLevelMenus" @menu-click="handleMenuItemClickByItem"></OneLevelMenu>
<!-- 左侧二级菜单区域 --> <!-- 左侧二级菜单区域 -->
<div class="layout-columns__right-menu"> <div class="layout-columns__right-menu" :class="{ collapse: appStore.menuCollapse }">
<div class="layout-columns__system-name gi_line_1">{{ appStore.getTitle() }}</div> <!-- 系统标题 -->
<div class="layout-columns__title">
<span v-show="!appStore.menuCollapse" class="system-name gi_line_1">{{ appStore.getTitle() }}</span>
</div>
<Menu <Menu
v-if="twoLevelMenus.length > 1 || oneLevelMenuActiveRoute?.meta?.alwaysShow === true" v-if="twoLevelMenus.length > 1 || oneLevelMenuActiveRoute?.meta?.alwaysShow === true"
class="layout-columns__menu" :menus="twoLevelMenus" :menu-style="{ width: '180px' }" class="layout-columns__menu" :menus="twoLevelMenus" :menu-style="menuStyle"
/> />
</div> </div>
</div> </div>
<!-- 右侧内容区域 -->
<section class="layout-columns__content"> <section class="layout-columns__content">
<Header /> <Header />
<Tabs v-if="appStore.tab" /> <Tabs v-if="appStore.tab" />
<Main /> <Main />
<GiFooter v-if="appStore.copyrightDisplay" />
</section> </section>
<!-- 公告弹窗 -->
<NoticePopup ref="noticePopupRef" />
</div> </div>
</template> </template>
@@ -32,6 +38,8 @@ import Tabs from './components/Tabs/index.vue'
import { useAppStore } from '@/stores' import { useAppStore } from '@/stores'
import { useLevelMenu } from '@/layout/hooks/useLevelMenu' import { useLevelMenu } from '@/layout/hooks/useLevelMenu'
import { useDevice } from '@/hooks' import { useDevice } from '@/hooks'
import NoticePopup from '@/views/user/message/components/NoticePopup.vue'
import { getToken } from '@/utils/auth'
defineOptions({ name: 'LayoutColumns' }) defineOptions({ name: 'LayoutColumns' })
@@ -40,6 +48,29 @@ const { isDesktop } = useDevice()
const { oneLevelMenus, twoLevelMenus, oneLevelMenuActiveRoute, getOneLevelMenus, handleMenuItemClickByItem } = useLevelMenu() const { oneLevelMenus, twoLevelMenus, oneLevelMenuActiveRoute, getOneLevelMenus, handleMenuItemClickByItem } = useLevelMenu()
getOneLevelMenus() getOneLevelMenus()
// 菜单样式 - 根据折叠状态动态调整宽度
const menuStyle = computed(() => {
return {
width: appStore.menuCollapse ? '48px' : '200px',
}
})
// 公告弹窗引用
const noticePopupRef = ref<InstanceType<typeof NoticePopup>>()
// 检查并显示未读公告
const checkAndShowNotices = () => {
const token = getToken()
// 如果有token检查未读公告
if (token) {
setTimeout(() => {
noticePopupRef.value?.open()
}, 1000) // 延迟1秒显示让页面先加载完成
}
}
onMounted(() => {
checkAndShowNotices()
})
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@@ -59,12 +90,16 @@ getOneLevelMenus()
&__right-menu { &__right-menu {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
width: 180px;
overflow: hidden; overflow: hidden;
background-color: var(--color-bg-1); background-color: var(--color-bg-1);
width: 200px;
transition: width 0.3s ease-in-out;
&.collapse {
width: 48px;
}
} }
&__system-name { &__title {
height: 56px; height: 56px;
padding: 0 12px; padding: 0 12px;
color: var(--color-text-1); color: var(--color-text-1);
@@ -78,17 +113,35 @@ getOneLevelMenus()
border-bottom: 1px solid var(--color-border); border-bottom: 1px solid var(--color-border);
cursor: pointer; cursor: pointer;
user-select: none; user-select: none;
transition: padding .2s; transition: all 0.3s ease-in-out;
&:hover { .system-name {
color: $color-theme !important; padding-left: 6px;
cursor: pointer; white-space: nowrap;
transition: all 0.3s ease-in-out;
line-height: 1.5;
display: inline-flex;
align-items: center;
opacity: 1;
transform: translateX(0);
&:hover {
color: $color-theme !important;
cursor: pointer;
}
.layout-columns__right-menu.collapse & {
opacity: 0;
transform: translateX(-20px);
}
} }
} }
&__menu { &__menu {
flex: 1; flex: 1;
overflow: auto; overflow: auto;
border-right: 1px solid var(--color-border-2); border-right: 1px solid var(--color-border-2);
transition: all 0.3s ease-in-out;
width: 100%;
} }
&__content { &__content {

View File

@@ -11,6 +11,7 @@
> >
<Logo :collapsed="appStore.menuCollapse" /> <Logo :collapsed="appStore.menuCollapse" />
<Menu :menus="twoLevelMenus" :menu-style="{ width: '200px', flex: 1 }" /> <Menu :menus="twoLevelMenus" :menu-style="{ width: '200px', flex: 1 }" />
<WwAds class="ads" />
</section> </section>
<!-- 右侧内容区域 --> <!-- 右侧内容区域 -->
@@ -32,7 +33,11 @@
</header> </header>
<Tabs v-if="appStore.tab" /> <Tabs v-if="appStore.tab" />
<Main /> <Main />
<GiFooter v-if="appStore.copyrightDisplay" />
</section> </section>
<!-- 公告弹窗 -->
<NoticePopup ref="noticePopupRef" />
</div> </div>
</template> </template>
@@ -47,6 +52,10 @@ import Tabs from './components/Tabs/index.vue'
import { useAppStore } from '@/stores' import { useAppStore } from '@/stores'
import { useLevelMenu } from '@/layout/hooks/useLevelMenu' import { useLevelMenu } from '@/layout/hooks/useLevelMenu'
import { useDevice } from '@/hooks' import { useDevice } from '@/hooks'
import { getToken } from '@/utils/auth'
import WwAds from '@/layout/components/WwAds.vue'
import NoticePopup from '@/views/user/message/components/NoticePopup.vue'
/** 组件名称 */ /** 组件名称 */
defineOptions({ name: 'LayoutMix' }) defineOptions({ name: 'LayoutMix' })
@@ -64,9 +73,27 @@ getOneLevelMenus()
const activeMenu = computed(() => [oneLevelMenuActiveRoute.value?.path ?? '']) const activeMenu = computed(() => [oneLevelMenuActiveRoute.value?.path ?? ''])
// 公告弹窗引用
const noticePopupRef = ref<InstanceType<typeof NoticePopup>>()
// 检查并显示未读公告
const checkAndShowNotices = () => {
const token = getToken()
// 如果有token检查未读公告
if (token) {
setTimeout(() => {
console.log(noticePopupRef.value)
noticePopupRef.value?.open()
}, 1000) // 延迟1秒显示让页面先加载完成
}
}
const getMenuIcon = (item: RouteRecordRaw) => { const getMenuIcon = (item: RouteRecordRaw) => {
return item.meta?.icon || item.children?.[0]?.meta?.icon return item.meta?.icon || item.children?.[0]?.meta?.icon
} }
onMounted(() => {
checkAndShowNotices()
})
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>

View File

@@ -1,5 +1,5 @@
import type { RouteRecordRaw } from 'vue-router' import type { RouteRecordRaw } from 'vue-router'
import { eachTree } from 'xe-utils' import { eachTree, searchTree } from 'xe-utils'
import { useRouteListener } from '@/hooks' import { useRouteListener } from '@/hooks'
import { useRouteStore } from '@/stores' import { useRouteStore } from '@/stores'
import { filterTree } from '@/utils' import { filterTree } from '@/utils'
@@ -29,11 +29,12 @@ export function useLevelMenu() {
const oneLevelMenuActiveRoute = ref<RouteRecordRaw | null>(null) const oneLevelMenuActiveRoute = ref<RouteRecordRaw | null>(null)
const getOneLevelMenuActiveRoute = (path: string) => { const getOneLevelMenuActiveRoute = (path: string) => {
return oneLevelMenus.value.find((i) => i.path === path) as RouteRecordRaw const arr = searchTree(showMenuList, (i) => i.path === path)
return arr?.[0]
} }
listenerRouteChange(({ to }) => { listenerRouteChange(({ to }) => {
oneLevelMenuActiveRoute.value = getOneLevelMenuActiveRoute(to.matched?.[0]?.path) oneLevelMenuActiveRoute.value = getOneLevelMenuActiveRoute(to.path)
}) })
// 获取一级菜单 // 获取一级菜单

View File

@@ -126,7 +126,7 @@ const fetchNoticeDetail = async (index: number) => {
console.error(`获取公告详情失败:`, error) console.error(`获取公告详情失败:`, error)
// 创建一个错误状态的公告对象 // 创建一个错误状态的公告对象
noticeCache.value.set(noticeId, { noticeCache.value.set(noticeId, {
id: noticeId, id: noticeId || '',
title: '获取公告失败', title: '获取公告失败',
content: '获取公告内容失败,请稍后重试', content: '获取公告内容失败,请稍后重试',
createUserString: '', createUserString: '',