后台管理系统--动态 tabs 
效果预览 

核心思路 
这里的 tabs 可以理解为 nav 的另一种展现形式而已,其实说白了都是一个个 router-link。 点击侧边栏的时候手动在 vuex 里面维护一个队列 前端再 v-for 渲染出来而已。 ——【花裤衩】
全局维护 tabOptions 
vuex 实例中,创建一个 tabOptions数组,用于保存已打开的标签页。
默认保留【首页】,tabOptions 初始值为[{route: '/', name: '首页'}]
currentIndex则用来匹配当前路由所打开的 tab 标签。
用户退出时需清空数组。
import { ADD_TAB, DELETE_TAB, SET_INDEX, CLEAR_TAB } from "./mutations-type";
export default {
  nameSpace: true,
  state: {
    tabOptions: [{ route: "/", name: "首页" }],
    currentIndex: "/",
    breadcrumbList: [],
  },
  mutations: {
    // 添加标签
    [ADD_TAB](state, data) {
      state.tabOptions.push(data);
    },
    // 删除标签
    [DELETE_TAB](state, route) {
      const index = state.tabOptions.findIndex((op) => op.route === route);
      state.tabOptions.splice(index, 1);
    },
    // 激活标签
    [SET_INDEX](state, index) {
      state.currentIndex = index;
    },
    // 清空标签
    [CLEAR_TAB](state) {
      state.tabOptions = [];
    },
  },
  actions: {},
};实现 commonTabView 组件 
template
el-tab-pane 标签公用一个 route-view 视图,
keep-alive 加include规则并对视图进行动态缓存。
需要缓存的页面,在其组件 name 以Keep结尾即可,如 coverKeep。
<template>
  <div style="height: 100%;">
    <el-tabs
      v-if="tabOptions.length"
      class="content-wrap"
      v-model="currentIndex"
      type="border-card"
      @tab-click="tabClick"
      @tab-remove="tabRemove"
    >
      <!-- 路由为'/'时不可关闭标签 -->
      <el-tab-pane
        v-for="item in tabOptions"
        :closable="item.route !== '/'"
        :key="item.route"
        :label="item.name"
        :name="item.route"
      ></el-tab-pane>
      <!-- 缓存组件 name 以 Keep 结尾的组件 -->
      <transition name="fade" mode="out-in">
        <keep-alive :include="/Keep$/">
          <router-view></router-view>
        </keep-alive>
      </transition>
    </el-tabs>
  </div>
</template>script
对路由进行监听,
并对路由实现全匹配(/detail?code=1和/detail?code=2为不同的页面),
根据路由添加|激活标签。
全匹配(/detail?code=1):路由fullPath ; 不完全匹配(/detail): 路由path
实现 tab 标签点击、移除事件。
<script>
export default {
  // immediate立即监听路由,进入页面即可触发监听事件
  watch: {
    $route: {
      immediate: true,
      handler (to) {
        const flag = this.tabOptions.findIndex(op => op.route === to.fullPath) > -1
        !flag && this.$store.commit('add-tab', { route: to.fullPath, name: to.name })
        this.$store.commit('set-index', to.fullPath)
      }
    }
  },
  computed: {
    tabOptions () {
      return this.$store.state.tab.tabOptions
    },
    currentIndex: {
      get () {
        return this.$store.state.tab.currentIndex
      },
      set (index) {
        this.$store.commit('set-index', index)
      }
    }
  },
  data () {
    return {}
  },
  methods: {
    // 点击tab
    tabClick (tab) {
      this.$router.push({ path: this.currentIndex })
    },
    // 移除tab
    tabRemove (tabName) {
      if (tabName === '/') {
        return
      }
      this.$store.commit('delete-tab', tabName)
      if (this.currentIndex === tabName) {
        if (this.tabOptions && this.tabOptions.length) {
          this.$store.commit('set-index', this.tabOptions[this.tabOptions.length - 1].route)
          this.$router.replace({ path: this.currentIndex })
        } else {
          this.$router.replace({ path: '/' })
        }
      }
    }
  }
}
</script>
<style lang="scss" scoped>
  .content-wrap {
    height: 100%;
    padding: 0;
    overflow: hidden;
    /deep/ .el-tabs__content {
      overflow: scroll;
      height: 100%;
    }
  }
  .fade-enter-active,
  .fade-leave-active {
    transition: all .2s ease;
  }
  .fade-enter,
  .fade-leave-active {
    opacity: 0;
  }
</style>左侧菜单栏 
由于设置了路由监听,左侧菜单栏点击事件只需进行正常的路由跳转即可。
// commonMenu.js
clickMenu (item) {
    this.$router.push({ path: item.path })
}面包屑导航 
面包屑导航实现关键是
this.$route.matched属性,它是一个数组,记录了路由的匹配过程。
vux 维护一个全局数组用来保存当前路由
export default new Vuex.Store({
  state: {
    breadcrumbList: [],
  },
  mutations: {
    [SET_BREADCRUMB](state, data) {
      state.breadcrumbList = [...data];
    },
  },
});将路由与面包屑映射
    getBreadCrumb (route) {
      // 过滤掉空path、name的路由
      let matchList = route.matched.filter(m => m.path && m.name)
      // 添加保持首页在最前面
      if (matchList[0].path !== '/') {
        matchList = [{ path: '/', name: '首页' }].concat(matchList)
      }
      this.$store.commit('set-breadcrumb', matchList)
    }渲染面包屑
<el-breadcrumb
  style="display: inline-block; margin-left: 20px"
  separator-class="el-icon-arrow-right"
>
  <el-breadcrumb-item v-for="tab in breadcrumbList" :key="tab.path" :to="tab">
    {{ tab.name }}
  </el-breadcrumb-item>
</el-breadcrumb>遇坑填坑 
1. 切换 tab 多次请求,次数和 tab 数量一致 
【错误示范】
routeView 若写在 el-tab-pane 标签内,则标签创建时都会声明一个特定的 routeview。
这样,3 个标签时则触发 3 次 route-view 的生命周期方法。
【TODO: 解释的不太好,后面遇到应该更新】
<el-tabs>
  <el-tab-pane
    v-for="item in tabOptions"
    :closable="item.route !== '/'"
    :key="item.route"
    :label="item.name"
    :name="item.route"
  >
    <router-view></router-view>
  </el-tab-pane>
</el-tabs>【正确写法】
让 el-tab-pane 公用一个 tab 内容进行展示。
<el-tabs>
  <el-tab-pane
    v-for="item in tabOptions"
    :closable="item.route !== '/'"
    :key="item.route"
    :label="item.name"
    :name="item.route"
  ></el-tab-pane>
  <router-view></router-view>
</el-tabs>2. 多次点击同一 tab,控制台报错 
这是由于没有处理路由跳转报错的原因(本例 tab 点击采用push方式)。
// 单独处理异常
this.$router.push("/").catch((err) => err);或者全局设置【推荐】
在引入 router 时,统一处理 push | repalce 的错误
import VueRouter from "vue-router";
const originalPush = VueRouter.prototype.push;
VueRouter.prototype.push = function push(location) {
  // 路由出错不打印:例如重复路由跳转
  return originalPush.call(this, location).catch((err) => err);
};3. 监听的路由初始化时没有触发,须在生命周期手动触发 
- handler 方法和 immediate 属性
  watch: {
    $route: {
      immediate: true,
      handler (to) {
        const flag = this.tabOptions.findIndex(op => op.route === to.fullPath) > -1
        !flag && this.$store.commit('add-tab', { route: to.fullPath, name: to.name })
        this.$store.commit('set-index', to.fullPath)
      }
    }
  },初始化tabOptions保留首页
tabOptions: [{ route: "/", name: "首页" }];