在 vue 中,預設情況下,一個元件範例在被替換掉後會被銷燬。這會導致它丟失其中所有已變化的狀態——當這個元件再一次被顯示時,會建立一個只帶有初始狀態的新範例。但是 vue 提供了 keep-alive 元件,它可以將一個動態元件包裝起來從而實現元件切換時候保留其狀態。本篇文章要介紹的並不是它的基本使用方法(這些官網檔案已經寫的很清楚了),而是它如何結合 VueRouter 來更自由的控制頁面狀態的快取
我們先搭建一個 Vue 專案,裡面有三個頁面a
,b
,c
,並給它們一些相互跳轉的邏輯和狀態
<template>
<div>
<div>A頁面</div>
<input type="text" v-model="dataA" /><br />
<div @click="toB">跳轉B</div>
<div @click="toC">跳轉C</div>
</div>
</template>
<script lang="ts" setup>
import { ref } from "vue";
import { useRouter, useRoute } from "vue-router";
const router = useRouter();
const route = useRoute();
const dataA = ref("");
const toB = () => {
router.push("/bb");
};
const toC = () => {
router.push("/cc");
};
</script>
<template>
<div>
<div>B頁面</div>
<input type="text" v-model="dataB" /><br />
<div @click="toA">跳轉A</div>
</div>
</template>
<script lang="ts" setup>
import { ref } from "vue";
import { useRouter } from "vue-router";
const router = useRouter();
const dataB = ref("");
const toA = () => {
router.push("/aa");
};
</script>
<template>
<div>
<div>C頁面</div>
<input type="text" v-model="dataC" />
<div @click="toA">跳轉A</div>
</div>
</template>
<script lang="ts" setup name="C">
import { ref } from "vue";
import { useRouter } from "vue-router";
const router = useRouter();
const dataC = ref("");
const toA = () => {
router.push("/aa");
};
</script>
然後在 route/index.ts 寫下它們對應的路由設定
import { createRouter, createWebHashHistory, RouteRecordRaw } from "vue-router";
const routes: RouteRecordRaw[] = [
{
path: "/aa",
name: "a",
component: () => import(/* webpackChunkName: "A" */ "../views/a.vue"),
},
{
path: "/bb",
name: "b",
component: () => import(/* webpackChunkName: "B" */ "../views/b.vue"),
},
{
path: "/cc",
name: "c",
component: () => import(/* webpackChunkName: "C" */ "../views/c.vue"),
},
];
const router = createRouter({
history: createWebHashHistory(),
routes,
});
export default router;
在 App.vue 中我們用 keep-alive 將 router-view 進行包裹
<template>
<keep-alive>
<router-view />
</keep-alive>
</template>
啟動專案,測試一下頁面狀態有沒有被快取
此時我們發現狀態並沒有快取,並且控制檯還給了個警告
上面的寫法在 vue2 中是可以的,但是在 vue3 中需要將 keep-alive 寫在 router-view 中才行,我們修改一下寫法
<template>
<router-view v-slot="{ Component }">
<keep-alive>
<component :is="Component" />
</keep-alive>
</router-view>
</template>
這種寫法其實就是 router-view 元件的插槽傳遞了一個帶有當前元件的元件名 Component 的物件,然後用 keep-alive 包裹一個動態元件(迴歸原始寫法)。
我們再試一下頁面的快取效果,這時候發現頁面的狀態被快取了
通常情況下我們並不想將所有頁面狀態都快取,而只想快取部分頁面,這樣的話該怎麼做呢?
其實我們可以在 template 中通過$route 獲取路由的資訊,所以我們可以在需要快取的頁面設定一下 meta 物件,比如 a 頁面我們想快取其狀態,可以將 keepAlive 設定位 true
//route/index.ts
const routes: RouteRecordRaw[] = [
{
path: "/aa",
name: "a",
meta: {
keepAlive: true,
},
component: () => import(/* webpackChunkName: "A" */ "../views/a.vue"),
},
...
];
然後回到 App.vue 中判斷 keepAlive 來決定是否快取
<template>
<router-view v-slot="{ Component }">
<keep-alive>
<component v-if="$route.meta.keepAlive" :is="Component" />
</keep-alive>
<component v-if="!$route.meta.keepAlive" :is="Component" />
</router-view>
</template>
再看下效果
此時我們發現 a 頁面狀態被快取,b 頁面的狀態沒有快取
但是有時候我們想要這樣一個效果
a 跳轉 b 的時候我們需要快取 a 頁面狀態,但是當 a 跳轉 c 的時候我們不需要快取 a 頁面,此時我們該如何做呢?
或許有的同學想到了這樣一個方法,當 a 跳轉 c 的時候將 a 頁面的快取刪除,這樣就實現了上面的效果。可惜我找了半天也沒找到 vue3 中刪除指定頁面快取的方法
我也嘗試過跳轉 c 頁面的時候將 a 的 keepAlive 設定為 false,但是再次回到 a 頁面的時候 keepAlive 會重置,a 頁面狀態依然會被快取。
既然如此為了做到更精細的快取控制只有使用 keep-alive 中的 inclue 屬性了
keep-alive 預設會快取內部的所有元件範例,但我們可以通過 include 來客製化該行為。它的值都可以是一個以英文逗號分隔的字串、一個正規表示式,或是一個陣列。這裡我們使用一個陣列來維護需要快取的元件頁面,注意這個陣列中是元件的名字而不是路由的 name
在 vue3 中給元件命名可以這樣寫
<script lang='ts'>
export default {
name: 'MyComponent',
}
</script>
但是我們通常會使用 setup 語法,這樣的話我們得寫兩個script
標籤,太麻煩。我們可以使用外掛vite-plugin-vue-setup-extend
處理
npm i vite-plugin-vue-setup-extend -D
然後在vite.config.ts
中引入這個外掛就可以使用了
import { defineConfig, Plugin } from "vite";
import vue from "@vitejs/plugin-vue";
import vueSetupExtend from "vite-plugin-vue-setup-extend";
export default defineConfig({
plugins: [vue(), vueSetupExtend()],
});
然後就可以這樣命名了
<script lang="ts" setup name="A"></script>
下面我們修改一下 App.vue
<template>
<router-view v-slot="{ Component }">
<keep-alive :include="['A']">
<component :is="Component" />
</keep-alive>
</router-view>
</template>
這其實就代表元件名為 A 的 頁面才會被快取,接下來我們要做的就是控制這個陣列來決定頁面的快取,但是這個陣列要放在哪裡維護呢? 答案肯定是放到全域性狀態管理器中拉。所以我們引入 Pinia 作為全域性狀態管理器
npm i pinia
在 main.ts 中註冊
import { createPinia } from "pinia";
const Pinia = createPinia();
createApp(App).use(route).use(Pinia).use(RouterViewKeepAlive).mount("#app");
新建 store/index.ts
import { defineStore } from "pinia";
export default defineStore("index", {
state: (): { cacheRouteList: string[] } => {
return {
cacheRouteList: [],
};
},
actions: {
//新增快取元件
addCacheRoute(name: string) {
this.cacheRouteList.push(name);
},
//刪除快取元件
removeCacheRoute(name: string) {
for (let i = this.cacheRouteList.length - 1; i >= 0; i--) {
if (this.cacheRouteList[i] === name) {
this.cacheRouteList.splice(i, 1);
}
}
},
},
});
在 App.vue 中使用 cacheRouteList
<template>
<router-view v-slot="{ Component }">
<keep-alive :include="catchStore.cacheRouteList">
<component :is="Component" />
</keep-alive>
</router-view>
</template>
<script lang="ts" setup>
import cache from "./store";
const catchStore = cache();
</script>
此時就可以根據 cacheRouteList 控制快取頁面了。
此時我們再來實現前面提到的問題a 跳轉 b 的時候我們需要快取 a 頁面狀態,但是當 a 跳轉 c 的時候我們不需要快取 a 頁面
就很簡單了
import cache from "../store";
const catchStore = cache();
const router = useRouter();
const toB = () => {
catchStore.addCacheRoute("A");
router.push("/bb");
};
const toC = () => {
catchStore.removeCacheRoute("A");
router.push("/cc");
};
此時再看下頁面的效果
可以發現 a 到 c 後再回來狀態就重置了,這樣不僅做到了上述效果,還可以讓你隨時隨地的去刪除指定元件的快取。
到這裡我們便完成了使用 inclue 對頁面狀態快取進行更精細化的控制。當然,如果你有更好的方案歡迎在評論區指出,一起討論探索