1. 为什么要写插件

一开始我也是以为Vue插件离我的日常开发很遥远的,但直到有一天公司用的组件库换了,换成了MD风格的Vuetify。

这个组件库优点就是好看、且各种插槽props可以充分满足自定义需求;缺点也很明显:在用惯了Element的人看来,这个组件库不但缺少了很多全局函数,如$message,而且自定义的插槽、Prop太多了,需要一段时间熟悉。其中尤其是缺少了函数式组件让我很难受,虽然可以通过顶层写个message组件来解决,但用起来还是太麻烦了。

于是再三思索下我决定自己用渲染函数封装个函数式组件。

2. 组件编写

Vue的组件封装还是很友好的,render能够直接渲染写好的Vue模板并把Prop传进去,就像下面这样:

Vue.extend({
  data() {
    return {
      ...
    };
  },
  render(h): VNode {
    const props = {
      ...
    };
    return h(Component, { props });
  },
});

看上去和单文件组件也没太大区别嘛,其实Element也是这么实现的函数式组件的。

1. Vuetify封装问题

尽管使用的Vuetify,但Vue模板中不能使用Vuetify组件。原因在于Vuetify对组件库中的所有组件的封装方法会生成一个递归调用自身的实例,详情可以在console中打印vuetify组件dom查看,Vuetify组件会带有一个$vuetify属性,其中递归放置了它自身,初步猜测是为了防止被二次封装。

因此Vue模板中需要使用原生实现。具体实现不在此赘述,可以前往npm下载本项目查看。

2. 同时使用typescript的问题

由于ts中无法直接在Vue实例上挂载额外的全局方法,需要编写声明文件实现,在声明文件中,需要添加以下代码将$message方法声明到Vue实例上:

declare module "vue/types/vue" {
  interface Vue {
    $message(options: MessageOptions): void;
  }
}

同时也必需引入此模块,否则仍会报错未找到模块。引入此模块后会提示模块未使用,忽略即可。

3. 插件注册

Vue中使用install函数定义插件,Vue.use会自动注入install方法的内容,install方法中第一个参数为注入的Vue实例,第二个参数为可选项。

具体实现如下:

const MESSAGE: PluginObject<any> = {
  duration: 2000, // 显示时间常量
  animateTime: 600, // 出现/隐藏动画时间常量
  instances: [], // message组件实例数组(避免同时过多message导致重叠、出现位置错误问题)
  install(Vue, globalOptions: GlobalOptions = {}) {
    function msg(options: MessageOptions) {
      // 初始化可选参数
      const currOptions: GlobalOptions = {};
      currOptions.verticalOffset =
        globalOptions.verticalOffset === 0 || globalOptions.verticalOffset
          ? globalOptions.verticalOffset
          : 40;
      currOptions.flat = globalOptions.flat || false;
      currOptions.position = globalOptions.position || "right";
      currOptions.width = globalOptions.width || "30%";

      // 渲染组件
      const VueMessage = Vue.extend({
        data() {
          return {
            show: false,
          };
        },
        render(h): VNode {
          const props = {
            type: options.type,
            message: options.message,
            show: this.show,
            flat: currOptions.flat,
            position: currOptions.position,
            width: currOptions.width,
          };
          return h(Message, { props });
        },
      });

      // 创建实例并绑定
      const newMessage = new VueMessage();
      let vm: typeof newMessage | null = newMessage.$mount();
      const el = vm.$el;
      document.body.appendChild(el);

      // 重新定位实例垂直位置,避免重叠
      let verticalOffset = currOptions.verticalOffset;
      const offsetTopArr: number[] = [];
      if (MESSAGE.instances.length) {
        const itemHeight = MESSAGE.instances[0].$el.offsetHeight + 8;
        MESSAGE.instances.forEach((item: any) => {
          offsetTopArr.push(item.$el.offsetTop);
        });
        offsetTopArr.sort((prev, next) => prev - next);
        if (offsetTopArr[0] !== verticalOffset) {
          offsetTopArr.unshift(verticalOffset - itemHeight);
        }

        verticalOffset =
          offsetTopArr.reduce((prev, next) => {
            if (next - prev > itemHeight) {
              return prev;
            } else {
              return next;
            }
          }) + itemHeight;
      }
      const child: any = vm.$children[0];
      child.verticalOffset = verticalOffset;

      // 创建实例成功,显示实例
      vm.show = true;
      MESSAGE.instances.push(vm);
      // 实例生命周期结束,清除绑定实例(发现清除实例后Vue会默认在原本实例的dom位置上增加一行注		 释<!-- -->,若要完全清除可能要全原生实现组件渲染和自定义生命周期)
      const t1 = setTimeout(() => {
        clearTimeout(t1);
        vm!.show = false;
        const t2 = setTimeout(() => {
          clearTimeout(t2);
          document.body.removeChild(el);
          newMessage.$destroy();
          vm = null;
          options.callback && options.callback();
          MESSAGE.instances.shift();
        }, MESSAGE.animateTime);
      }, MESSAGE.duration);
    }

    // 绑定全局方法
    Vue.prototype.$message = msg;
  },
};

至此整个插件就编写完成了,限于篇幅,一些interface、模板实现没有介绍,感兴趣的可以到npm下载ca-vuetify-message组件查看源码。

由于本人也是第一次编写Vue插件,其中有错误内容还望互相交流斧正。