手写vue-router之history模式【三、具体实现一】

156 阅读2分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 6 月更文挑战」的第23天,点击查看活动详情

前情提要

  • 之前已经讲过前端路由基本功能的实现分析,那么咱们今天开始动手实现history模式的路由。
  • 首先我们通过vue-cli创建一个vue的项目。
  • 然后我们在src目录下创建一个router的文件夹,然后在文件夹下创建一个index.js的文件
  • 最后让我们开始吧...

history模式具体实现

  • class类
因为我们后来需要使用Vue构造函数,所以我们需要将Vue实例记录到全局变量
let _Vue = null;
export default class Router {
    // 该方法是用来向vue实例挂载router实例和router-link以及router-view组件的
    // 但是在项目运行过程中会多次运行该方法,而实例只需要挂载一次即可,因此我们需要做个判断,当该类的静态方法中存在时便直接返回。
    static install(Vue) {
        // 判断是否已经被执行
        if (Router.install.installed) return;
        // 首次执行之后将installed属性置为true
        Router.install.installed = true;
        _Vue = Vue;
        // 此处我们需要将router挂载到vue的原型上。但是如果我们不使用混入则我们在执行this.$options.router时,this指向的是类本身。这显然不是我们想要的,所以我们需要通过混入,以及在beforeCreate生命周期钩子中执行在vue原型上挂载$router操作。
        _Vue.mixin({
            beforeCreate() {
                // 为了保证执行一次就行了
                if (this.$options.router) {
                    _Vue.prototype.$router = this.$options.router;
                    // 此处也可以直接写成:this.createRouteMap();this.initComponents(_Vue);
                    // 用来初始化组件,和routeMap对象。
                    this.$options.router.init();
                }
            }
        })
    }
    // 构造函数,初始化options和routeMap以及data属性
    constructor(options) {
        // 实例化对象时传入的路由配置信息对象
        this.options = options;
        // 初始化路由地址和组件的映射关系
        this.routeMap = {};
        // 由于data代表的是当前路由,所以我们需要通过Vue提供的observable方法来将data构造成响应式属性。
        this.data = _Vue.observable({
            current: '/'
        })
    }
    // 初始化
    init() {
        this.createdRouteMap();
        this.initComponents(_Vue);
    }
    // 根据路由配置信息表得到路由和组件的映射关系
    createRouteMap() {
        this.options.routes.forEach(route => {
            this.routeMap[route.path] = route.component;
        })
    }
    // 生成路由所需的公共组件:router-link
    // 根据公共组件router-link,可以看出其接受一个props:to
    // 同时也运行自定义内容。
    // 因此得到如下的组件,通过slot插槽来实现公共组件的挂载。
    initComponents(Vue) {
        Vue.component('router-link', {
            props: {
                to: String
            },
            template: '<a :href="to"><slot></slot></a>'
        })
    }
}

总结

  • 好了,然后就可以替换项目中引入的vue-router插件了。然后跑起来试一下...
  • 不出意外的话你会得到一个惊喜...