Files
ao3-mirror-ssr/src/App.vue
UnknownMp 2aea1eae7f
All checks were successful
Node.js CI / build-and-test (push) Successful in 24s
[Feature]
为 SSR 的导航菜单添加链接
2025-05-16 09:31:13 +08:00

128 lines
4.4 KiB
Vue

<script setup>
import { onMounted, onBeforeMount, onServerPrefetch, nextTick, ref, watch } from 'vue'
import { useRouter, useRoute, RouterView } from 'vue-router'
import { useApiStore } from '@/stores/api.js'
import { useClientOnlyStore } from './ssr/clientOnlyStore.js'
import { useThemeStore } from './stores/themeScheme.js'
import { useMobileScreen } from './stores/device.js'
import { useRouteStore } from './stores/route.js'
import 'mdui/components/top-app-bar.js'
import 'mdui/components/top-app-bar-title.js'
import 'mdui/components/navigation-drawer.js'
import 'mdui/components/list.js'
import 'mdui/components/list-item.js'
import 'mdui/components/circular-progress.js'
import 'mdui/components/button-icon.js'
import 'mdui/components/switch.js'
import '@mdui/icons/arrow-back.js'
import '@mdui/icons/light-mode.js'
import '@mdui/icons/menu.js'
const router = useRouter()
const route = useRoute()
const clientOnlyStore = useClientOnlyStore()
const api = useApiStore()
const themeScheme = useThemeStore()
const mobileScreen = useMobileScreen()
const routeStore = useRouteStore()
const drawerOpen = ref(false)
const drawer = ref(null)
const closeDrawer = ref(true)
let progressTimer = null
onServerPrefetch(async () => {
await api.init()
})
onBeforeMount(() => {
mobileScreen.reCal()
if(!mobileScreen.isMobile) drawerOpen.value = true
})
onMounted(async () => {
themeScheme.applyTheme()
await api.init()
clientOnlyStore.setClient()
new MutationObserver(() => {
if (document.documentElement.style.width.includes('calc')) {
document.documentElement.style.width = '';
}
}).observe(document.documentElement, { attributes: true, attributeFilter: ['style'] });
watch(() => mobileScreen.isMobile, (newV, oldV) => {
if( oldV && !newV ) nextTick(() => drawer.value.open = true)
if( !oldV && newV ) nextTick(() => drawer.value.open = false)
})
})
</script>
<template>
<header><ClientOnly>
<mdui-top-app-bar style="background-color: rgb(var(--mdui-color-primary-container));" scroll-behavior="shrink elevate">
<mdui-button-icon @click="drawer.open = !drawer.open">
<mdui-icon-menu></mdui-icon-menu>
</mdui-button-icon>
<!--<mdui-button-icon @click="$router.back()" v-if="$route.path != '/'">
<mdui-icon-arrow-back></mdui-icon-arrow-back>
</mdui-button-icon>-->
<mdui-top-app-bar-title style="color: rgb(var(--mdui-color-on-surface-variant))">{{ routeStore.title }}</mdui-top-app-bar-title>
<mdui-circular-progress v-if="routeStore.showProgress" :value='routeStore.progress' :max='routeStore.progressMax'></mdui-circular-progress>
<div style="flex-grow: 1"></div>
<mdui-button-icon @click="themeScheme.switchMode()">
<mdui-icon-light-mode style="color: rgb(var(--mdui-color-on-surface-variant))"></mdui-icon-light-mode>
</mdui-button-icon>
</mdui-top-app-bar>
<template #ssr>
<h1>{{ routeStore.title }}</h1>
</template></ClientOnly></header>
<nav><ClientOnly>
<mdui-navigation-drawer ref='drawer' :open="drawerOpen" close-on-overlay-click close-on-esc style="margin-top: 64px;">
<mdui-list style="height: 100%; background-color: rgb(var(--mdui-color-surface-variant));">
<KeepAlive><mdui-list-item
v-for="item in routeStore.allRoutes"
:key="item.path"
@click="routeStore.drawerPress(item.path); if (mobileScreen.isMobile && closeDrawer ) drawer.open = false"
:class="{ 'active-item' : item.path == $route.path }"
>{{ item.name }}</mdui-list-item></KeepAlive>
<div v-if="mobileScreen.isMobile" class="bottom"><mdui-switch @change="closeDrawer = $event.target.checked" :checked="closeDrawer"></mdui-switch><div style="margin-left: 8px">切换页面时关闭菜单</div></div>
</mdui-list>
</mdui-navigation-drawer>
<template #ssr>
<ul>
<li v-for="item in routeStore.allRoutes"
:key="item.path"
:class="{ 'active-item' : item.path == $route.path }"
><a :href="item.path">{{ item.name }}</a></li>
</ul>
</template></ClientOnly>
</nav>
<main :class="{ 'mdui-prose' : clientOnlyStore.isClient }">
<RouterView v-slot="{ Component }">
<component :is="Component" />
</RouterView>
</main>
<footer></footer>
</template>
<style scoped>
.active-item {
background-color: rgb(var(--mdui-color-surface-container-high));
}
.bottom {
position: fixed;
display: flex;
align-items: center;
background-color: rgb(var(--mdui-color-surface-variant));
bottom: 16px;
left: 16px;
display: flex;
}
</style>