今天刚好有时间,最近也在观察vue3新特性,抽空玩一玩嵌套路由的vue-router,直接上代码
项目目录结构

代码展示
- app.vue
<template>
  <div id="app">
    <div>
      <router-link to="/">Index</router-link> |
      <router-link to="/person">Person</router-link> |
      <router-link to="/person/info">PersonInfo</router-link>
    </div>
    <!-- 一级路由 -->
    <router-view />
  </div>
</template>
<style>
#app{
  display: flex;
  flex-direction: column;
  align-items: center;
}
</style>
复制代码- Index.vue
<template>
  <div class="index">
    <h1>this is index page</h1>
  </div>
</template>
复制代码- Person.vue
<template>
  <div class="person">
    <h1>this is person page</h1>
     <!-- 二级路由 -->
    <router-view />
  </div>
</template>
复制代码- PersonInfo.vue
<template>
  <div class="personInfo">
    <h2>this is personInfo page</h2>
  </div>
</template>
复制代码js文件
- main.js
import Vue from 'vue'
import App from './App.vue'
import router from './router'
new Vue({
  router,
  render: h => h(App)
}).$mount('#app')
复制代码- router目录下文件
- index.js
import Vue from "vue";
import VueRouter from "./vue-router";
import Index from "../views/Index.vue";
import Person from "../views/Person.vue";
import PersonInfo from "../views/PersonInfo.vue";
Vue.use(VueRouter);
const routes = [
  {
    path: "/",
    name: "Index",
    component: Index
  },
  {
    path: "/person",
    name: "Person",
    component: Person,
    children:[
      {
        path: "/person/info",
        name: "PersonInfo",
        component: PersonInfo
      }
    ]
  }
];
const router = new VueRouter({
  routes
});
export default router;
复制代码- vue-router.js
这里先借助Vue的工具Vue.util.defineReactive实现数据响应式,后续再手撕这个库
import routerLink from "./router-link";
import routerView from "./router-view";
let Vue;
class VueRouter {
  constructor(options) {
    this.$options = options
    this.current = window.location.hash.slice(1) || "/"
    // 设置响应式数组数据
    Vue.util.defineReactive(this, "routerArray", [])
    // 监听hash值变化
    window.addEventListener("hashchange", this.hashChange.bind(this))
    this.getRouterArray()
  }
  hashChange() {
    this.current = window.location.hash.slice(1) || "/"
    this.routerArray = []
    this.getRouterArray()
  }
  getRouterArray(routes) {
    routes = routes || this.$options.routes
    for (const route of routes) {
      if (this.current === '/' && route.path === '/') {
        this.routerArray.push(route)
        return
      }
      if (this.current.indexOf(route.path) !== -1 && route.path !== '/') {
        this.routerArray.push(route)
        if (route.children) {
          // 递归子路由
          this.getRouterArray(route.children)
        }
        return
      }
    }
  }
}
VueRouter.install = function(_Vue) {
  Vue = _Vue
  // Vue全局混入,等new Vue中的router实例创建之后挂载到Vue上
  Vue.mixin({
    beforeCreate() {
      if (this.$options.router) {
        Vue.prototype.$router = this.$options.router
      }
    },
  });
  // 注册router-link和router-view全局组件
  Vue.component("router-link", routerLink)
  Vue.component("router-view", routerView)
}
export default VueRouter
复制代码- router-link.js
export default {
  props: {
    to: {
      type: String,
      required: true
    }
  },
  render(h) {
    return h(
      "a",
      {
        attrs: {
          href: "#" + this.to,
        },
      },
      this.$slots.default
    );
  }
};
复制代码- router-view.js
export default {
  render(h) {
    // 设置嵌套路由标识
    this.$vnode.data.rv = true
    // 嵌套路由设置深度
    let depth = 0
    let parent = this.$parent
    while(parent){
      let vnodeData = parent.$vnode && parent.$vnode.data
      // 上级还有嵌套路由标识rv为true的,深度加一
      if (vnodeData && vnodeData.rv) {
          depth++
      }
      parent = parent.$parent
    }
    // 简单处理
    let component = null
    const route = this.$router.routerArray[depth]
    if (route) {
      component = route.component
    }
    return h(component);
  }
};
复制代码- 效果图

好了,今天就玩到这里了,下次再玩别的哈
© 版权声明
文章版权归作者所有,未经允许请勿转载。
THE END
    






















![[桜井宁宁]COS和泉纱雾超可爱写真福利集-一一网](https://www.proyy.com/skycj/data/images/2020-12-13/4d3cf227a85d7e79f5d6b4efb6bde3e8.jpg)

![[桜井宁宁] 爆乳奶牛少女cos写真-一一网](https://www.proyy.com/skycj/data/images/2020-12-13/d40483e126fcf567894e89c65eaca655.jpg)
