前端开发那些事儿

vue 使用keep-alive 三级以上的路由,无法缓存问题!

2020-07-03  本文已影响0人  宝妞儿

使用vue-element-admin脚手架做系统时,一般系统的侧边栏都是有父级,子级,子子级,发现使用keep-alive,当三级以上的路由时,无法缓存页面??

那首先要了解下keep-alive的作用是啥?

相关的文章介绍很多,就不赘述了。。。(感兴趣的多查查其他文章)
那它具体的属性有2种:

<keep-alive>
  <router-view :key="key" />
</keep-alive>

computed: {
 key() {
  return this.$route.path
 }
}

注意这里有个坑点:

1.png

(1)官方vue组件的name的命名方式,参考链接: https://cn.vuejs.org/v2/guide/components-registration.html

这里,由于默认的eslint配置文件.eslintrc.js中,如下:


2.png

即页面组件的name , 如果不是大写字母开头,就会飘红报错。(我有点强迫症)所以页面所有的路由组件具体页面的name,我都以大写字母开始命名

(2)这个具体vue组件页面的name要跟你router文件中配置的name一样
因为在每次打开标签页和关闭标签页处理时,是根据name来匹配,进行添加删除。如果组件内部页面配置的name是Order,而路由文件配置的name是order,那include和exclude也不会有用!

准备工作做好了后,就可以使用keep-alive 啦

include 属性进行缓存页面组件的操作了,实现参考了github这篇: https://github.com/PanJiaChen/vue-element-admin/commit/9ad96525af3a26c2f4e8d4e9016f801c44df1f80

上面这个链接的代码,实现了三级路由以上的缓存处理:(但还会有个问题,刷新或者关闭当前标签页,再次打开时,清除不了当前组件的缓存(比如搜索条件写了个东西,进行搜索,关闭或者刷新后,应该清空重新请求,但还是保留。。我们后面会解决的,先接着往下看~))

上述链接的解决的思路如下:


3.png
Parent
<script>
export default {
  name: 'Menu1',
  data() {
  return {
      ...
   }
 },
 created() {
   const route = this.$route.matched.find(item => item.name === 'Menu1')
   this.$store.dispatch('tagsView/addCachedView', route)
 }
}
</script>

注: 上述组件的name值和find比较的name 值要均保持一致,这里,如果当前组件的name改变了后,代码改动的地方有2处,有没有更取巧的方式呢,如下代码,就可以直接获取当前组件的name, 而不用相同的name写两遍

this.$options.name

所以上述代码,我们有可以改写成这样:

Parent
<script>
export default {
  name: 'Menu1',
  data() {
  return {
      ...
   }
 },
 created() {
   const route = this.$route.matched.find(item => item.name === this.$options.name)
   this.$store.dispatch('tagsView/addCachedView', route)
 }
}
</script>

那又回到刚才说的问题了,刷新或者关闭再打开当前3级路由的标签页时,缓存未清空?

问题总是有解决办法的,参考链接是这个issue: https://github.com/PanJiaChen/vue-element-admin/pull/3116

tagsView.js

addCachedView({ commit }, view) {
   if (view.matched && view.matched.length >= 3) { // 若为三级及其以上路由点击打开标签页时,将三级路由或以上的根目录路由塞入缓存路由name list中
   commit('ADD_CACHED_VIEW', view.matched[1])
  }
   commit('ADD_CACHED_VIEW', view) // 反之,正常的进行标签页的添加进缓存路由name list中
},


delCachedView({ commit, state }, view) {
  return new Promise(resolve => {
    if (view.matched && view.matched.length >= 3) { // 若为三级及其以上路由关闭当前标签页时,将3级路由以上的根目录name 从list中删除
    commit('DEL_CACHED_VIEW', view.matched[1])
  }
   commit('DEL_CACHED_VIEW', view) // 反之,正常的进行标签页的name 从list中删除
   resolve([...state.cachedViews])
  })
},

以上解决问题啦~~<( ̄▽ ̄)/

还要注意的是:

AppMain.vue里这样写:(注意include方法,取的是cachedViews)

<template>
  <section class="app-main">
    <transition name="fade-transform" mode="out-in">
      <keep-alive :include="cachedViews">
        <router-view :key="key" />
      </keep-alive>
    </transition>
  </section>
</template>


<script>
export default {
  name: 'AppMain',
  computed: {
    cachedViews() {
      return this.$store.state.tagsView.cachedViews
    },
    key() {
      return this.$route.path
    }
  }
}
</script>
<style scope>
...
</style>
上一篇下一篇

猜你喜欢

热点阅读