mirror of
https://github.com/chansee97/nova-admin.git
synced 2025-04-05 12:44:27 +08:00
feat: add mix layout
This commit is contained in:
parent
8f5f11f4d3
commit
9ce6bd3b86
63
.vscode/settings.json
vendored
63
.vscode/settings.json
vendored
@ -1,6 +1,4 @@
|
|||||||
{
|
{
|
||||||
// Enable the ESlint flat config support
|
|
||||||
"eslint.experimental.useFlatConfig": true,
|
|
||||||
// Disable the default formatter, use eslint instead
|
// Disable the default formatter, use eslint instead
|
||||||
"prettier.enable": false,
|
"prettier.enable": false,
|
||||||
"editor.formatOnSave": false,
|
"editor.formatOnSave": false,
|
||||||
@ -11,46 +9,16 @@
|
|||||||
},
|
},
|
||||||
// Silent the stylistic rules in you IDE, but still auto fix them
|
// Silent the stylistic rules in you IDE, but still auto fix them
|
||||||
"eslint.rules.customizations": [
|
"eslint.rules.customizations": [
|
||||||
{
|
{ "rule": "style/*", "severity": "off" },
|
||||||
"rule": "style/*",
|
{ "rule": "format/*", "severity": "off" },
|
||||||
"severity": "off"
|
{ "rule": "*-indent", "severity": "off" },
|
||||||
},
|
{ "rule": "*-spacing", "severity": "off" },
|
||||||
{
|
{ "rule": "*-spaces", "severity": "off" },
|
||||||
"rule": "format/*",
|
{ "rule": "*-order", "severity": "off" },
|
||||||
"severity": "off"
|
{ "rule": "*-dangle", "severity": "off" },
|
||||||
},
|
{ "rule": "*-newline", "severity": "off" },
|
||||||
{
|
{ "rule": "*quotes", "severity": "off" },
|
||||||
"rule": "*-indent",
|
{ "rule": "*semi", "severity": "off" }
|
||||||
"severity": "off"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"rule": "*-spacing",
|
|
||||||
"severity": "off"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"rule": "*-spaces",
|
|
||||||
"severity": "off"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"rule": "*-order",
|
|
||||||
"severity": "off"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"rule": "*-dangle",
|
|
||||||
"severity": "off"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"rule": "*-newline",
|
|
||||||
"severity": "off"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"rule": "*quotes",
|
|
||||||
"severity": "off"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"rule": "*semi",
|
|
||||||
"severity": "off"
|
|
||||||
}
|
|
||||||
],
|
],
|
||||||
// Enable eslint for all supported languages
|
// Enable eslint for all supported languages
|
||||||
"eslint.validate": [
|
"eslint.validate": [
|
||||||
@ -64,7 +32,16 @@
|
|||||||
"json",
|
"json",
|
||||||
"jsonc",
|
"jsonc",
|
||||||
"yaml",
|
"yaml",
|
||||||
"toml"
|
"toml",
|
||||||
|
"xml",
|
||||||
|
"gql",
|
||||||
|
"graphql",
|
||||||
|
"astro",
|
||||||
|
"css",
|
||||||
|
"less",
|
||||||
|
"scss",
|
||||||
|
"pcss",
|
||||||
|
"postcss"
|
||||||
],
|
],
|
||||||
"i18n-ally.sourceLanguage": "zh_CN",
|
"i18n-ally.sourceLanguage": "zh_CN",
|
||||||
"i18n-ally.displayLanguage": "zh_CN",
|
"i18n-ally.displayLanguage": "zh_CN",
|
||||||
|
@ -22,12 +22,14 @@ const isLocal = computed(() => {
|
|||||||
|
|
||||||
function getLocalIcon(icon: string) {
|
function getLocalIcon(icon: string) {
|
||||||
const svgName = icon.replace('local:', '')
|
const svgName = icon.replace('local:', '')
|
||||||
const svg = import.meta.glob('@/assets/svg-icons/*.svg', {
|
const svg = import.meta.glob<string>('@/assets/svg-icons/*.svg', {
|
||||||
query: '?raw',
|
query: '?raw',
|
||||||
import: 'default',
|
import: 'default',
|
||||||
eager: true,
|
eager: true,
|
||||||
})
|
})
|
||||||
return svg[`/src/assets/svg-icons/${svgName}.svg`]
|
|
||||||
|
const domparser = new DOMParser()
|
||||||
|
return domparser.parseFromString(svg[`/src/assets/svg-icons/${svgName}.svg`], 'image/svg+xml')
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@ -38,15 +40,13 @@ function getLocalIcon(icon: string) {
|
|||||||
:depth="depth"
|
:depth="depth"
|
||||||
:color="color"
|
:color="color"
|
||||||
>
|
>
|
||||||
<Icon :icon="icon" />
|
<template v-if="isLocal">
|
||||||
</n-icon>
|
{{ getLocalIcon(icon) }}
|
||||||
<n-icon
|
</template>
|
||||||
v-if="icon && isLocal"
|
<template v-else>
|
||||||
:size="size"
|
<Icon :icon="icon" />
|
||||||
:depth="depth"
|
</template>
|
||||||
:color="color"
|
</n-icon>
|
||||||
v-html="getLocalIcon(icon)"
|
|
||||||
/>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped></style>
|
<style scoped></style>
|
||||||
|
@ -29,7 +29,7 @@ const value = defineModel<LayoutMode>('value', { required: true })
|
|||||||
:class="{
|
:class="{
|
||||||
'outline outline-2': value === 'topMenu',
|
'outline outline-2': value === 'topMenu',
|
||||||
}"
|
}"
|
||||||
class="grid grid-rows-[30%_1fr] outline-[var(--primary-color)] hover:(outline outline-2) cursor-pointer"
|
class="grid grid-rows-[30%_1fr] outline-[var(--primary-color)] hover:(outline outline-2) cursor-pointer"
|
||||||
@click="value = 'topMenu'"
|
@click="value = 'topMenu'"
|
||||||
>
|
>
|
||||||
<div class="bg-[var(--primary-color)]" />
|
<div class="bg-[var(--primary-color)]" />
|
||||||
@ -38,6 +38,24 @@ const value = defineModel<LayoutMode>('value', { required: true })
|
|||||||
</template>
|
</template>
|
||||||
<span> {{ $t('app.topMenu') }} </span>
|
<span> {{ $t('app.topMenu') }} </span>
|
||||||
</n-tooltip>
|
</n-tooltip>
|
||||||
|
|
||||||
|
<n-tooltip placement="bottom" trigger="hover">
|
||||||
|
<template #trigger>
|
||||||
|
<n-el
|
||||||
|
:class="{
|
||||||
|
'outline outline-2': value === 'mixMenu',
|
||||||
|
}"
|
||||||
|
class="grid grid-cols-[20%_1fr] grid-rows-[15%_15%_1fr] outline-[var(--primary-color)] hover:(outline outline-2) cursor-pointer"
|
||||||
|
@click="value = 'mixMenu'"
|
||||||
|
>
|
||||||
|
<div class="bg-[var(--primary-color)] row-span-3" />
|
||||||
|
<div class="bg-[var(--primary-color)]" />
|
||||||
|
<div class="bg-[var(--primary-color-suppl)]" />
|
||||||
|
<div class="bg-[var(--divider-color)]" />
|
||||||
|
</n-el>
|
||||||
|
</template>
|
||||||
|
<span> {{ $t('app.topMenu') }} </span>
|
||||||
|
</n-tooltip>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
@ -4,13 +4,13 @@ import { useAppStore, useRouteStore } from '@/store'
|
|||||||
|
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const appStore = useAppStore()
|
const appStore = useAppStore()
|
||||||
const routesStore = useRouteStore()
|
const routeStore = useRouteStore()
|
||||||
|
|
||||||
const menuInstRef = ref<MenuInst | null>(null)
|
const menuInstRef = ref<MenuInst | null>(null)
|
||||||
watch(
|
watch(
|
||||||
() => route.path,
|
() => route.path,
|
||||||
() => {
|
() => {
|
||||||
menuInstRef.value?.showOption(routesStore.activeMenu as string)
|
menuInstRef.value?.showOption(routeStore.activeMenu as string)
|
||||||
},
|
},
|
||||||
{ immediate: true },
|
{ immediate: true },
|
||||||
)
|
)
|
||||||
@ -22,7 +22,7 @@ watch(
|
|||||||
:collapsed="appStore.collapsed"
|
:collapsed="appStore.collapsed"
|
||||||
:indent="20"
|
:indent="20"
|
||||||
:collapsed-width="64"
|
:collapsed-width="64"
|
||||||
:options="routesStore.menus"
|
:options="routeStore.menus"
|
||||||
:value="routesStore.activeMenu"
|
:value="routeStore.activeMenu"
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import leftMenu from './leftMenu.layout.vue'
|
import leftMenu from './leftMenu.layout.vue'
|
||||||
import topMenu from './topMenu.layout.vue'
|
import topMenu from './topMenu.layout.vue'
|
||||||
|
import mixMenu from './mixMenu.layout.vue'
|
||||||
import { SettingDrawer } from './components'
|
import { SettingDrawer } from './components'
|
||||||
import { useAppStore } from '@/store/app'
|
import { useAppStore } from '@/store/app'
|
||||||
|
|
||||||
@ -8,6 +9,7 @@ const appStore = useAppStore()
|
|||||||
const layoutMap = {
|
const layoutMap = {
|
||||||
leftMenu,
|
leftMenu,
|
||||||
topMenu,
|
topMenu,
|
||||||
|
mixMenu,
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
152
src/layouts/mixMenu.layout.vue
Normal file
152
src/layouts/mixMenu.layout.vue
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import type { MenuInst, MenuOption } from 'naive-ui'
|
||||||
|
import {
|
||||||
|
BackTop,
|
||||||
|
CollapaseButton,
|
||||||
|
FullScreen,
|
||||||
|
Logo,
|
||||||
|
Notices,
|
||||||
|
Search,
|
||||||
|
Setting,
|
||||||
|
TabBar,
|
||||||
|
UserCenter,
|
||||||
|
} from './components'
|
||||||
|
import { useAppStore, useRouteStore } from '@/store'
|
||||||
|
|
||||||
|
const routeStore = useRouteStore()
|
||||||
|
const appStore = useAppStore()
|
||||||
|
const pageRoute = useRoute()
|
||||||
|
const router = useRouter()
|
||||||
|
|
||||||
|
const menuInstRef = ref<MenuInst | null>(null)
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => pageRoute.path,
|
||||||
|
() => {
|
||||||
|
menuInstRef.value?.showOption(routeStore.activeMenu as string)
|
||||||
|
},
|
||||||
|
{ immediate: true },
|
||||||
|
)
|
||||||
|
|
||||||
|
const topMenu = ref<MenuOption[]>([])
|
||||||
|
const activeTopMenu = ref<string>('')
|
||||||
|
function handleTopMenu(rowMenu: MenuOption[]) {
|
||||||
|
topMenu.value = rowMenu.map((i) => {
|
||||||
|
const { icon, label, key } = i
|
||||||
|
return {
|
||||||
|
icon,
|
||||||
|
label,
|
||||||
|
key,
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
handleTopMenu(routeStore.menus)
|
||||||
|
|
||||||
|
// 根据当前页面获取选中菜单和对应侧边菜单
|
||||||
|
const currentMenuKey = pageRoute.matched[1].path
|
||||||
|
updateTopMenu(currentMenuKey)
|
||||||
|
})
|
||||||
|
|
||||||
|
const sideMenu = ref<MenuOption[]>([])
|
||||||
|
function handleSideMenu(key: string) {
|
||||||
|
const targetMenu = routeStore.menus.find(i => i.key === key)
|
||||||
|
if (targetMenu) {
|
||||||
|
sideMenu.value = targetMenu.children ? targetMenu.children : [targetMenu]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateTopMenu(key: string) {
|
||||||
|
handleSideMenu(key)
|
||||||
|
activeTopMenu.value = key
|
||||||
|
router.push(key)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<n-layout
|
||||||
|
has-sider
|
||||||
|
class="wh-full"
|
||||||
|
embedded
|
||||||
|
>
|
||||||
|
<n-layout-sider
|
||||||
|
bordered
|
||||||
|
:collapsed="appStore.collapsed"
|
||||||
|
collapse-mode="width"
|
||||||
|
:collapsed-width="64"
|
||||||
|
:width="240"
|
||||||
|
content-style="display: flex;flex-direction: column;min-height:100%;"
|
||||||
|
>
|
||||||
|
<Logo v-if="appStore.showLogo" />
|
||||||
|
<n-scrollbar class="flex-1">
|
||||||
|
<n-menu
|
||||||
|
ref="menuInstRef"
|
||||||
|
:collapsed="appStore.collapsed"
|
||||||
|
:indent="20"
|
||||||
|
:collapsed-width="64"
|
||||||
|
:options="sideMenu"
|
||||||
|
:value="routeStore.activeMenu"
|
||||||
|
/>
|
||||||
|
</n-scrollbar>
|
||||||
|
</n-layout-sider>
|
||||||
|
<n-layout
|
||||||
|
class="h-full flex flex-col"
|
||||||
|
content-style="display: flex;flex-direction: column;min-height:100%;"
|
||||||
|
embedded
|
||||||
|
:native-scrollbar="false"
|
||||||
|
>
|
||||||
|
<n-layout-header bordered position="absolute" class="z-1">
|
||||||
|
<div class="h-60px flex-y-center justify-between">
|
||||||
|
<CollapaseButton />
|
||||||
|
<n-menu
|
||||||
|
ref="menuInstRef"
|
||||||
|
mode="horizontal"
|
||||||
|
responsive
|
||||||
|
:options="topMenu"
|
||||||
|
:value="activeTopMenu"
|
||||||
|
@update:value="updateTopMenu"
|
||||||
|
/>
|
||||||
|
<div class="flex-y-center gap-1 h-full p-x-xl">
|
||||||
|
<Search />
|
||||||
|
<Notices />
|
||||||
|
<FullScreen />
|
||||||
|
<DarkModeSwitch />
|
||||||
|
<LangsSwitch />
|
||||||
|
<Setting />
|
||||||
|
<UserCenter />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<TabBar v-if="appStore.showTabs" class="h-45px" />
|
||||||
|
</n-layout-header>
|
||||||
|
<div class="flex-1 p-16px flex flex-col">
|
||||||
|
<div class="h-60px" />
|
||||||
|
<div v-if="appStore.showTabs" class="h-45px" />
|
||||||
|
<router-view v-slot="{ Component, route }" class="flex-1">
|
||||||
|
<transition
|
||||||
|
:name="appStore.transitionAnimation"
|
||||||
|
mode="out-in"
|
||||||
|
>
|
||||||
|
<keep-alive :include="routeStore.cacheRoutes">
|
||||||
|
<component
|
||||||
|
:is="Component"
|
||||||
|
v-if="appStore.loadFlag"
|
||||||
|
:key="route.fullPath"
|
||||||
|
/>
|
||||||
|
</keep-alive>
|
||||||
|
</transition>
|
||||||
|
</router-view>
|
||||||
|
<div v-if="appStore.showFooter" class="h-40px" />
|
||||||
|
</div>
|
||||||
|
<n-layout-footer
|
||||||
|
v-if="appStore.showFooter"
|
||||||
|
bordered
|
||||||
|
position="absolute"
|
||||||
|
class="h-40px flex-center"
|
||||||
|
>
|
||||||
|
{{ appStore.footerText }}
|
||||||
|
</n-layout-footer>
|
||||||
|
<BackTop />
|
||||||
|
</n-layout>
|
||||||
|
</n-layout>
|
||||||
|
</template>
|
@ -5,7 +5,7 @@ import themeConfig from './theme.json'
|
|||||||
import { local, setLocale } from '@/utils'
|
import { local, setLocale } from '@/utils'
|
||||||
|
|
||||||
type TransitionAnimation = '' | 'fade-slide' | 'fade-bottom' | 'fade-scale' | 'zoom-fade' | 'zoom-out'
|
type TransitionAnimation = '' | 'fade-slide' | 'fade-bottom' | 'fade-scale' | 'zoom-fade' | 'zoom-out'
|
||||||
export type LayoutMode = 'leftMenu' | 'topMenu'
|
export type LayoutMode = 'leftMenu' | 'topMenu' | 'mixMenu'
|
||||||
|
|
||||||
const docEle = ref(document.documentElement)
|
const docEle = ref(document.documentElement)
|
||||||
|
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import type { MenuOption } from 'naive-ui'
|
||||||
import { createMenus, createRoutes, generateCacheRoutes } from './helper'
|
import { createMenus, createRoutes, generateCacheRoutes } from './helper'
|
||||||
import { $t, local } from '@/utils'
|
import { $t, local } from '@/utils'
|
||||||
import { router } from '@/router'
|
import { router } from '@/router'
|
||||||
@ -7,7 +8,7 @@ import { useAuthStore } from '@/store/auth'
|
|||||||
|
|
||||||
interface RoutesStatus {
|
interface RoutesStatus {
|
||||||
isInitAuthRoute: boolean
|
isInitAuthRoute: boolean
|
||||||
menus: AppRoute.Route[]
|
menus: MenuOption[]
|
||||||
rowRoutes: AppRoute.RowRoute[]
|
rowRoutes: AppRoute.RowRoute[]
|
||||||
activeMenu: string | null
|
activeMenu: string | null
|
||||||
cacheRoutes: string[]
|
cacheRoutes: string[]
|
||||||
|
Loading…
x
Reference in New Issue
Block a user