feat: And surface switching loading optimization

This commit is contained in:
vben
2024-06-23 19:39:44 +08:00
parent 6afed34437
commit 24aab5b4bb
24 changed files with 596 additions and 1696 deletions

View File

@@ -2,15 +2,18 @@
import type { RouteLocationNormalizedLoaded } from 'vue-router';
import { preferences, usePreferences } from '@vben-core/preferences';
import { Spinner } from '@vben-core/shadcn-ui';
import { storeToRefs, useTabsStore } from '@vben-core/stores';
import { IFrameRouterView } from '../../iframe';
import { useContentSpinner } from './use-content-spinner';
defineOptions({ name: 'LayoutContent' });
const { keepAlive } = usePreferences();
const tabsStore = useTabsStore();
const { onTransitionEnd, spinning } = useContentSpinner();
const { getCacheTabs, getExcludeTabs, renderRouteView } =
storeToRefs(tabsStore);
@@ -29,32 +32,45 @@ function getTransitionName(route: RouteLocationNormalizedLoaded) {
}
// 如果页面已经加载过,则不使用动画
if (route.meta.loaded) {
return;
}
// if (route.meta.loaded) {
// return;
// }
// 已经打开且已经加载过的页面不使用动画
const inTabs = getCacheTabs.value.includes(route.name as string);
return inTabs && route.meta.loaded ? undefined : transitionName;
}
</script>
<template>
<IFrameRouterView />
<RouterView v-slot="{ Component, route }">
<Transition :name="getTransitionName(route)" appear mode="out-in">
<KeepAlive
v-if="keepAlive"
:exclude="getExcludeTabs"
:include="getCacheTabs"
<div class="relative h-full">
<Spinner
v-if="preferences.transition.loading"
:spinning="spinning"
class="h-[var(--vben-content-client-height)]"
/>
<IFrameRouterView />
<RouterView v-slot="{ Component, route }">
<Transition
:name="getTransitionName(route)"
appear
mode="out-in"
@transitionend="onTransitionEnd"
>
<component
:is="Component"
v-if="renderRouteView"
v-show="!route.meta.iframeSrc"
:key="route.fullPath"
/>
</KeepAlive>
<component :is="Component" v-else :key="route.fullPath" />
</Transition>
</RouterView>
<KeepAlive
v-if="keepAlive"
:exclude="getExcludeTabs"
:include="getCacheTabs"
>
<component
:is="Component"
v-if="renderRouteView"
v-show="!route.meta.iframeSrc"
:key="route.fullPath"
/>
</KeepAlive>
<component :is="Component" v-else :key="route.fullPath" />
</Transition>
</RouterView>
</div>
</template>

View File

@@ -0,0 +1,56 @@
import { computed, ref } from 'vue';
import { useRouter } from 'vue-router';
import { preferences } from '@vben-core/preferences';
function useContentSpinner() {
const spinning = ref(false);
const isStartTransition = ref(false);
const startTime = ref(0);
const router = useRouter();
const minShowTime = 500;
const enableLoading = computed(() => preferences.transition.loading);
const onEnd = () => {
if (!enableLoading.value) {
return;
}
const processTime = performance.now() - startTime.value;
if (processTime < minShowTime) {
setTimeout(() => {
spinning.value = false;
}, minShowTime - processTime);
} else {
spinning.value = false;
}
};
router.beforeEach((to) => {
if (to.meta.loaded || !enableLoading.value) {
return true;
}
isStartTransition.value = false;
startTime.value = performance.now();
spinning.value = true;
return true;
});
router.afterEach((to) => {
if (to.meta.loaded || !enableLoading.value) {
return true;
}
// 未进入过渡动画
if (!isStartTransition.value) {
// 关闭加载动画
onEnd();
}
isStartTransition.value = false;
return true;
});
return { onTransitionEnd: onEnd, spinning };
}
export { useContentSpinner };