美文网首页二次封装组件vue
Vue原理解析(十一):搞懂extend和$mount原理并实现

Vue原理解析(十一):搞懂extend和$mount原理并实现

作者: 飞跃疯人院_a | 来源:发表于2019-12-16 22:02 被阅读0次

    上一篇 Vue原理解析(十):搞懂事件API原理及在组件库中的妙用
    在学习老黄的Vue2.0开发企业级移动端音乐Web App课程时,里面有一个精美的确认弹窗组件,如下:

    image
    不过使用起来并不是很方便,如每个使用的地方需要引入该组件,需要注册,需要给组件加ref引用,需要调用事件来控制状态。其实这个组件相对来说是比较独立的,我们在使用组件库的时候,相信都有调用过命令式弹窗组件的经历,今天我们就来搞懂这种命令式组件的实现原理,以及将这个精美的弹窗组件改为命令式的,也就是这样调用:
    this.$Confirm({...})
      .then(confirm => {
        ...
      })
      .catch(cancel => {
        ...
      })
    

    原理解析之extend和$mount

    这两个都是vue提供的API,不过在平时的业务开发中使用并不多。在vue的内部也有使用过这一对API。遇到嵌套组件时,首先将子组件转为组件形式的VNode时,会将引入的组件对象使用extend转为子组件的构造函数,作为VNode的一个属性Ctor;然后在将VNode转为真实的Dom的时候实例化这个构造函数;最后实例化完成后手动调用$mount进行挂载,将真实Dom插入到父节点内完成渲染。

    所以这个弹窗组件可以这样实现,我们自己对组件对象使用extend转为构造函数,然后手动调用$mount转为真实Dom,由我们来指定一个父节点让它插入到指定的位置。

    在动手前,我们再多花点时间深入理解下流程细节:

    extend

    接受的是一个组件对象,再执行extend时将继承基类构造器上的一些属性、原型方法、静态方法等,最后返回Sub这么一个构造好的子组件构造函数。拥有和vue基类一样的能力,并在实例化时会执行继承来的_init方法完成子组件的初始化。

    Vue.extend = function (extendOptions = {}) {
      const Super = this  // Vue基类构造函数
      const name = extendOptions.name || Super.options.name
      
      const Sub = function (options) {  // 定义构造函数
        this._init(options)  // _init继承而来
      }
      
      Sub.prototype = Object.create(Super.prototype)  // 继承基类Vue初始化定义的原型方法
      Sub.prototype.constructor = Sub  // 构造函数指向子类
      Sub.options = mergeOptions( // 子类合并options
        Super.options,  // components, directives, filters, _base
        extendOptions  // 传入的组件对象
      )
      Sub['super'] = Super // Vue基类
    
      // 将基类的静态方法赋值给子类
      Sub.extend = Super.extend
      Sub.mixin = Super.mixin
      Sub.use = Super.use
    
      ASSET_TYPES.forEach(function (type) { // ['component', 'directive', 'filter']
        Sub[type] = Super[type]
      })
      
      if (name) {  让组件可以递归调用自己,所以一定要定义name属性
        Sub.options.components[name] = Sub  // 将子类挂载到自己的components属性下
      }
    
      Sub.superOptions = Super.options
      Sub.extendOptions = extendOptions
    
      return Sub  // 返回子组件的构造函数
    }
    

    实例化Sub

    执行_init组件初始化的一系列操作,初始化事件、生命周期、状态等等。将dataprops内定义的变量挂载到当前this实例下,最后返回一个实例化后的对象。

    Vue.prototype._init = function(options) {  // 初始化
      ...
      initLifecycle(vm)
      initEvents(vm)
      initRender(vm)
      callHook(vm, 'beforeCreate')
      initInjections(vm)
      initState(vm)
      initProvide(vm)
      callHook(vm, 'created')  // 初始化阶段完成
      ...
      
      if (vm.$options.el) {  // 开始挂载阶段
        vm.$mount(vm.$options.el)  // 执行挂载
      }
    }
    

    $mount

    在得到初始化后的对象后,开始组件的挂载。首先将当前render函数转为VNode,然后将VNode转为真实Dom插入到页面完成渲染。再完成挂载之后,会在当前组件实例this下挂载$el属性,它就是完成挂载后对应的真实Dom,我们就需要使用这个属性。

    组件改造

    1. 写出组件 (完整代码在最后)

    因为是Promise的方式调用的,所以显示后返回Promise对象,这里只放出主要的JavaScript部分:

    export default {
      data() {
        return {
          showFlag: false,
          title: "确认清空所有历史纪录吗?",  // 可以使用props
          ConfirmBtnText: "确定",  // 为什么不用props接受参数
          cancelBtnText: "取消"  // 之后会明白
        };
      },
      methods: {
        show(cb) {  // 加入一个在执行Promise前的回调
          this.showFlag = true;
          typeof cb === "function" && cb.call(this, this);
          return new Promise((resolve, reject) => { // 返回Promise
            this.reject = reject;  // 给取消按钮使用
            this.resolve = resolve;  // 给确认按钮使用
          });
        },
        cancel() {
          this.reject("cancel");  // 抛个字符串
          this.hide();
        },
        confirm() {
          this.resolve("confirm");
          this.hide();
        },
        hide() {
          this.showFlag = false;
          document.body.removeChild(this.$el);  // 结束移除Dom
          this.$destroy();  // 执行组件销毁
        }
      }
    };
    

    2. 转换调用方式

    组件对象已经有了,接下来就是将它转为命令式可调用的:

    confirm/index.js
    
    import Vue from 'vue';
    import Confirm from './confirm';  // 引入组件
    
    let newInstance;
    const ConfirmInstance = Vue.extend(Confirm);  // 创建构造函数
    
    const initInstance = () => { // 执行方法后完成挂载
      newInstance = new ConfirmInstance();  // 实例化
      document.body.appendChild(newInstance.$mount().$el);
      // 实例化后手动挂载,得到$el真实Dom,将其添加到body最后
    }
    
    export default options => { 导出一个方法,接受配置参数
      if (!newInstance) {
        initInstance(); // 挂载
      }
      Object.assign(newInstance, options);
      // 实例化后newInstance就是一个对象了,所以data内的数据会
      // 挂载到this下,传入一个对象与之合并
      
      return newInstance.show(vm => {  // 显示弹窗
        newInstance = null;  // 将实例对象清空
      })
    }
    

    这里其实可以使用install做成一个插件,还没介绍它就略过了。首先使用extend将组件对象转换为组件构造函数,执行initInstance方法后就会将真实Dom挂载到body的最后。为什么之前不使用props而是用的data,因为它们初始化后都会挂载到this下,不过data代码量少。导出一个方法给到外部使用,接受配置参数,调用后返回一个Promise对象。

    3. 挂载到全局

    main.js内将导出的方法挂载到Vue的原型上,让其成为一个全局方法:

    import Confirm from './base/confirm/index';
    
    Vue.prototype.$Confirm = Confirm;
    
    试试这样调用吧~
    this.$Confirm({
      title: 'vue大法好!'
    }).then(confirm => {
      console.log(confirm)  
    }).catch(cancel => {
      console.log(cancel)
    })
    

    组件完整代码如下:

    confirm/confirm.vue
    
    <template>
      <transition name="confirm-fade">
        <div class="confirm" v-show="showFlag">
          <div class="confirm-wrapper">
            <div class="confirm-content">
              <p class="text">{{title}}</p>
              <div class="operate" @click.stop>
                <div class="operate-btn left" @click="cancel">{{cancelBtnText}}</div>
                <div class="operate-btn" @click="confirm">{{ConfirmBtnText}}</div>
              </div>
            </div>
          </div>
        </div>
      </transition>
    </template>
    
    <script>
    export default {
      data() {
        return {
          showFlag: false,
          title: "确认清空所有历史纪录吗?", 
          ConfirmBtnText: "确定",
          cancelBtnText: "取消"
        };
      },
      methods: {
        show(cb) {
          this.showFlag = true;
          typeof cb === "function" && cb.call(this, this);
          return new Promise((resolve, reject) => {
            this.reject = reject;
            this.resolve = resolve;
          });
        },
        cancel() {
          this.reject("cancel");
          this.hide();
        },
        confirm() {
          this.resolve("confirm");
          this.hide();
        },
        hide() {
          this.showFlag = false;
          document.body.removeChild(this.$el);
          this.$destroy();
        }
      }
    };
    </script>
    
    <style scoped lang="stylus">
    .confirm {
      position: fixed;
      left: 0;
      right: 0;
      top: 0;
      bottom: 0;
      z-index: 998;
      background-color: rgba(0, 0, 0, 0.3);
      &.confirm-fade-enter-active {
        animation: confirm-fadein 0.3s;
        .confirm-content {
          animation: confirm-zoom 0.3s;
        }
      }
      .confirm-wrapper {
        position: absolute;
        top: 50%;
        left: 50%;
        transform: translate(-50%, -50%);
        z-index: 999;
        .confirm-content {
          width: 270px;
          border-radius: 13px;
          background: #333;
          .text {
            padding: 19px 15px;
            line-height: 22px;
            text-align: center;
            font-size: 18px;
            color: rgba(255, 255, 255, 0.5);
          }
          .operate {
            display: flex;
            align-items: center;
            text-align: center;
            font-size: 18px;
            .operate-btn {
              flex: 1;
              line-height: 22px;
              padding: 10px 0;
              border-top: 1px solid rgba(0, 0, 0, 0.3);
              color: rgba(255, 255, 255, 0.3);
              &.left {
                border-right: 1px solid rgba(0, 0, 0, 0.3);
              }
            }
          }
        }
      }
    }
    @keyframes confirm-fadein {
      0% {opacity: 0;}
      100% {opacity: 1;}
    }
    @keyframes confirm-zoom {
      0% {transform: scale(0);}
      50% {transform: scale(1.1);}
      100% {transform: scale(1);}
    }
    </style>
    

    试着实现一个全局的提醒组件吧,原理差不多的~

    最后按照惯例我们还是以一道vue可能会被问到的面试题作为本章的结束~

    面试官微笑而又不失礼貌的问道:

    • 请说明下组件库中命令式弹窗组件的原理?

    怼回去:

    • 使用extend将组件转为构造函数,在实例化这个这个构造函数后,就会得到$el属性,也就是组件的真实Dom,这个时候我们就可以操作得到的真实的Dom去任意挂载,使用命令式也可以调用。

    下一篇: 埋头书写中...

    顺手点个赞或关注呗,找起来也方便~

    分享一个组件库,你可能会用的上哦 ~ ↓

    你可能会用的上的一个vue功能组件库,持续完善中...

    相关文章

      网友评论

        本文标题:Vue原理解析(十一):搞懂extend和$mount原理并实现

        本文链接:https://www.haomeiwen.com/subject/knkjnctx.html