推广

如何吃透 vue-router

iseeyu2年前 (2024-02-21)推广149

image.png

“您使用的是Vue的仅运行时构建,而模板编译器不可用。要么将模板预编译为呈现函数,要么使用包含编译器的构建。”

出现这个题就涉及到了vue的构建版本的差异。

我们的使用cli 运行项目的时候,使用的是vue的运行时版本构建的,它不支持template模板编译,需要打包的时候提前编译。而完整版包含了运行时版本和编译器,在运行的时候会把模板转换成render函数,因此体积比运行时版本大10KB左右。

针对这种情况 Vue 官方已经给出我们解决的方案(参考链接),只需要我们在vue.config.js中,将 runtimeCompiler 设置成 true 即可。

除此之外我们还可以自己去使用 render 渲染模板。

  initComponents (Vue) {
    Vue.component('router-link', {
      props: {
        to: String
      },
      // template: '<a :href="to"><slot></slot></a>'
      render (h) { // h 是一个函数用来创建虚拟dom
        return h('a', {
          attrs: {
            href: this.to // history 模式
          }
        }, [this.$slots.default])
      }
    })
  }

继续创建 route-view 组件
我们route-view 组件的内容已经存储在 routeMap 里了,所以只需要我们将其取出转换成 vDom 即可

  initComponents (Vue) {
    Vue.component('router-link', {
      props: {
        to: String
      },
      // template: '<a :href="to"><slot></slot></a>'
      render (h) { // h 是一个函数用来创建虚拟dom
        return h('a', {
          attrs: {
            href: this.to // history 模式
          }
        }, [this.$slots.default])
      }
    })

    const _this = this // 记录 当前 vuerouter 对象
    Vue.component('router-view', {
      render (h) {
        const component = _this.routeMap[_this.data.current]
        return h(component)
      }
    })
  }

这个时候我们满心以为我们的vue-router 基本功能已经完成的时候,突然想到了我们忽略了一件事儿,因为我们之前的route-link 使用的 a标签 本身的默认事件是跳转并且从服务器请求。因此需要我们给 route-link 做一下改造。

  initComponents (Vue) {
    Vue.component('router-link', {
      props: {
        to: String
      },
      // template: '<a :href="to"><slot></slot></a>'
      render (h) { // h 是一个函数用来创建虚拟dom
        return h('a', {
          attrs: {
            href: this.to // history 模式
          },
          on: {
            click: this.clickHandler
          }
        }, [this.$slots.default])
      },
      methods: {
        clickHandler (e) {
          history.pushState({/* 传递给 popstate 参数 */}, '', this.to) // 改变地址栏
          this.$router.data.current = this.to // 改变current 重新出发渲染
          e.preventDefault()
        }
      }
    })

    const _this = this // 记录 当前 vuerouter 对象
    Vue.component('router-view', {
      render (h) {
        const component = _this.routeMap[_this.data.current]
        return h(component)
      }
    })
  }

截止到现在我们的基本功能已经可以使用了,但是我们发现之前提到过的 initEvent 没有实现,他的作用是注册 popstate 方法,监听浏览器地址变化,我们点击前进后退的时候就会发现 页面没有变化。因此我们还需要实现initEvent。

  initEvent () {
    window.addEventListener('popstate', () => { // 箭头函数不改变this指向
      this.data.current = window.location.pathname
    })
  }

现在我们完善了我们的代码,奉上

let _Vue = null

export default class VueRouter {
  constructor (options) {
    this.options = options
    this.data = _Vue.observable({
      current: '/'
    })
    this.routeMap = {}
  }

  createRouteMap () {
    this.options.routes.forEach(route => {
      this.routeMap[route.path] = route.component
    })
  }

  init () {
    this.createRouteMap()
    this.initComponents(_Vue)
    this.initEvent()
  }

  initComponents (Vue) {
    Vue.component('router-link', {
      props: {
        to: String
      },
      // template: '<a :href="to"><slot></slot></a>'
      render (h) { // h 是一个函数用来创建虚拟dom
        return h('a', {
          attrs: {
            href: this.to // history 模式
          },
          on: {
            click: this.clickHandler
          }
        }, [this.$slots.default])
      },
      methods: {
        clickHandler (e) {
          history.pushState({/* 传递给 popstate 参数 */}, '', this.to) // 改变地址栏
          this.$router.data.current = this.to // 改变current 重新出发渲染
          e.preventDefault()
        }
      }
    })

    const _this = this // 记录 当前 vuerouter 对象
    Vue.component('router-view', {
      render (h) {
        const component = _this.routeMap[_this.data.current]
        return h(component)
      }
    })
  }

  initEvent () {
    window.addEventListener('popstate', () => { // 箭头函数不改变this指向
      this.data.current = window.location.pathname
    })
  }

  static install (Vue) {
    // 1. 判断当前插件是否被安装
    if (VueRouter.install.installed) {
      return
    }
    VueRouter.install.installed = true
    // 2. 把 Vue 构造函数记录到全局变量
    _Vue = Vue
    // 3. 把创建 Vue 实例的时候传入的 router 对象注入到 Vue 实例上
    _Vue.mixin({
      beforeCreate () {
        if (this.$options.router) { // 在 vue 中执行,其他组件 $options.router 不存在,保证以下代码执行一次
          _Vue.prototype.$router = this.$options.router
          this.$options.router.init()
        }
      }
    })
  }
}

运行!可用!正常!
vue-router 的 history 模式完成!

扫描二维码推送至手机访问。

版权声明:本文由西安泽虎代运营发布,如需转载请注明出处。

转载请注明出处https://www.0291.com.cn/post/57399.html

相关文章

物质可以转换成能量,那么能量也可以转化成物质吗?

说起爱因斯坦,我们都知道他是继牛顿和伽利略之后最伟大的科学家。在1905年,他发表了一篇划时代的论文,开启了物理学的新纪元。这篇论文便是人人都熟知的狭义相对论在狭义相对论中,最著名的莫过于质能方程,他给我们阐述了质量与能量之间当量的?关系?,质能?方程?的?表达式??为?E...

因为不同的搜索引擎优化规则,所以SEO策略也要不一样的。

因为不同的搜索引擎优化规则,所以SEO策略也要不一样的。

每个站点在优化过程中需求针对的搜索引擎都不止一次,尽管国内市场百度的占比更大一些,可其他的搜索引擎就没人用了吗?其实不然,假如一个站点想要包括用户群更广一些,那么在进行优化作业的时分必定不要只是考虑单一搜索引擎,面向简直悉数的搜索引擎做优化才是准确的。 前几年,站长在优化网站的时分,为垂...

3步学会搜索霸屏推广方法效果永久可靠

在互联网上,一个人把做好了,他就能顺风顺水。你不需要样样精通,只专注一项技能,其它的全部外包找人去做。下面这节课,我们将学会如何通过写文章来霸屏搜索引擎这一技能。  做好搜索霸屏一共就三个步骤:  一、分别是分析关键词  1.分析搜索量,判断关键词的需求  2.分析对手数,...

霍思燕穿3万元的背心晒画作,曾经说:自己的财富是靠理财赚来的

霍思燕穿3万元的背心晒画作,曾经说:自己的财富是靠理财赚来的

#霍思燕#霍思燕在某社交平台上晒出一段视频,并配文:三十年没画画啦,给大家露一小手,献一小丑。在视频中,霍思燕大方展出自己画的国画,还念了一首诗,并在自己的画作前合照,看得出来,她很开心特别满意自己画的国画。网友说:画画手艺一般,人长的漂亮。从视频中看到,霍思燕画画时非常认...

百度付费推广类型有哪些?

百度付费推广类型有哪些?

现在还是有着非常多的商家看到了百度广告的优势,所以也就想通过百度广告的方式对自己的公司以及产品有着一定的宣传,在宣传之前也需要进行百度广告开户,所以就会问它百度广告开户怎么样,值不值得自己去选择?因为有的时候自己是不方便去开户,或者说对开户方面的并不了解,如果茫茫然的去开户...

分析100个裂变案例后 总结了一个快速裂变的万能公式。

分析100个裂变案例后 总结了一个快速裂变的万能公式。

本文分析了当下近百个营销案例,帮助大家剥茧抽丝,提炼出一个裂变活动的核心公式,大家以后再做活动的时候,可以基于此公式来参考。 本文认为一个成功的裂变活动=初始流量+基础信任+低参与门槛+高诱饵+流程简单+及时反馈。 【一】初始流量助力裂变 请不要去相信任何一篇文章,一个...

现在,非常期待与您的又一次邂逅

我们努力让每一部企业宣传片和抖音短视频成为商业大片