VUE 2.x中全局组件的封装(三)

Vue.extend 属于 Vue 的全局 API,在实际业务开发中我们很少使用,因为相比常用的 Vue.component 写法使用 extend 步骤要更加繁琐一些。但是在一些独立组件开发场景中,Vue.extend + $mount 这对组合是我们需要去关注的。
用法:
使用基础 Vue 构造器,创建一个“子类”。参数是一个包含组件选项的对象
data 选项是特例,需要注意 - 在 Vue.extend() 中它必须是函数

为什么使用 extend

在 vue 项目中,我们有了初始化的根实例后,所有页面基本上都是通过 router 来管理,组件也是通过 import 来进行局部注册,所以组件的创建我们不需要去关注,相比 extend 要更省心一点点。但是这样做会有几个缺点:

  1. 组件模板都是事先定义好的,如果我要从接口动态渲染组件怎么办?
  2. 所有内容都是在 #app 下渲染,注册组件都是在当前位置渲染。如果我要实现一个类似于 window.alert() 提示组件要求像调用 JS 函数一样调用它,该怎么办?

这时候,Vue.extend + vm.$mount 组合就派上用场了。

一个简单示例-hello world

index.vue

<template>
  <div>
    <p>text:{{ text }}</p>
    <p>content:{{ content }}</p>
    <p>isShow:{{ isShow }}</p>
  </div>
</template>

<script>
export default {
  props: {
    text: {
      type: String,
      default: "textDefault"
    }
  },
  data() {
    return {
      isShow: false,
      content: "contentDefault"
    };
  }
};
</script>

跟他同级目录的 index.js

import Vue from "vue";
import helloWorld from "./index.vue";
const HelloWorldConstructor = Vue.extend(helloWorld);

const Hello = option => {
  const HelloWorldInstance = new HelloWorldConstructor({
    data: {
      content: option.content
    },
    // 传props 值必须用 propsData
    propsData: {
      text: option.text
    }
  }); // 实例化一个带有content内容的Notice
  // 设置 data 中的值也可以这样直接写
  HelloWorldInstance.isShow = option.isShow;
  HelloWorldInstance.$mount(); // 挂载但是并未插入dom,是一个完整的Vue实例
  let helloDom = HelloWorldInstance.$el;
  document.body.appendChild(helloDom); // 将dom插入body
  // 一
  // return HelloWorldInstance;
  // 二
  return () => {
    HelloWorldInstance.$el.parentNode.removeChild(HelloWorldInstance.$el);
  };
};

export default {
  install: Vue => {
    Vue.prototype.$hello = Hello;
  }
};

main.js中注入

import Hello from "@/components/global/Hello"; //这个是 index.js
Vue.use(Hello); // 使用全局组件 Hello

使用--在test.vue中

<template>
  <div>
    测试 hello 组件
  </div>
</template>
<script>
export default {
  name: "test",
  created() {
    let removeHello = this.$hello({
      content: "hello--content",
      text: "hello--text",
      isShow: "isShow-true"
    });
    setTimeout(() => {
      // 一
      // removeHello.$el.parentNode.removeChild(removeHello.$el);
      // 二
      removeHello();
    }, 3000);
  }
};
</script>

Vue.extend 自定义插件形式全局弹窗提示组件

来个升级版,写个类似alert的组件
notice.vue

<template>
  <transition name="message-fade">
    <div :class="['message', 'notice-' + type]" v-show="visible">
      <i :class="'el-icon-' + type + ' content-' + type"></i>
      <div :class="['content', 'content-' + type]">{{ content }}</div>
    </div>
  </transition>
</template>

<script>
export default {
  data() {
    return {
      content: "",
      type: "info", //'success','warning','error'
      duration: 2500, // 弹窗展示时长
      visible: false,
      hasClose: false,
      noticeTimer: null
    };
  },
  mounted() {
    this.close();
    if (window.noticeEl && window.noticeEl.length > 0) {
      let top = parseFloat(window.noticeEl[window.noticeEl.length - 1].style.top);
      this.$el.style.top = top + 80 + "px";
      window.noticeEl.push(this.$el);
    } else {
      window.noticeElTop = this.$el.offsetTop || document.body.clientHeight * 0.04;
      this.$el.style.top = window.noticeElTop + "px";
      window.noticeEl = [this.$el];
    }
  },
  methods: {
    close() {
      setTimeout(() => {
        this.visible = false;
        window.noticeEl = window.noticeEl.filter(val => val !== this.$el);
        let top = window.noticeElTop;
        window.noticeEl.map((val, index) => {
          val.style.top = top + index * 80 + "px";
          return val;
        });
        setTimeout(() => {
          this.$destroy(true);
          this.$el.parentNode.removeChild(this.$el);
        }, 800); // 销毁自身组件,主要是要大于动画执行时间
      }, this.duration);
    }
  }
};
</script>

<style scoped lang="scss">
.message {
  position: absolute;
  top: 8vh;
  left: 50%;
  padding: 15px 20px;
  border-radius: 8px;
  background-color: #fff;
  transform: translateX(-50%);
  transition: opacity 0.2s, transform 0.2s, top 0.4s;
  font-size: 30px;
}
.content {
  display: inline-block;
  margin-left: 12px;
  min-width: 380px;
}
/* .#2ed573 */
.notice-success {
  background-color: #f0f9eb;
  border-color: #e1f3d8;
}
.el-icon-success,
.content-success {
  color: #67c23a;
}
.notice-warning {
  background-color: #fdf6ec;
  border-color: #faecd8;
}
.notice-warning .content-warning {
  color: #e6a23c;
}
.notice-error {
  background-color: #fef0f0;
  border-color: #fde2e2;
}
.notice-error .content-error {
  color: #f56c6c;
}

// donghua
.message-fade-enter,
.message-fade-leave-to {
  opacity: 0;
  transform: translateX(-50%) translateY(-8vh);
}
.message-fade-enter-to,
.message-fade-leave {
  opacity: 1;
  transform: translateX(-50%) translateY(0px);
}

.message-fade-enter-active {
  transition: all 0.4s ease;
}
.message-fade-leave-active {
  transition: all 0.4s cubic-bezier(1, 0.2, 0.8, 1);
}
</style>

跟notice.vue同级目录的 index.js

import Vue from "vue";
const NoticeConstructor = Vue.extend(require("./notice.vue").default);

let nId = 1;
// 为了同时兼容单独引入此组件的情况,加 export ,可以单独导出
export const Notice = (option = { message: "提交成功!", type: "success" }) => {
  const NoticeInstance = new NoticeConstructor({
    data: {
      content: option.message,
      type: option.type
    }
  }); // 实例化一个带有content内容的Notice
  NoticeInstance.visible = true;
  NoticeInstance.id = "notice-" + nId++;
  NoticeInstance.$mount(); // 挂载但是并未插入dom,是一个完整的Vue实例
  NoticeInstance.$el.style.zIndex = nId + 10000;
  document.body.appendChild(NoticeInstance.$el); // 将dom插入body
  return NoticeInstance;
};
// 全局组件挂载所用,对应 Vue.use()
export default {
  install: Vue => {
    Vue.prototype.$notice = Notice;
  }
};

main.js中注入

import Notice from "@/components/global/Notice"; //这个是 index.js
Vue.use(Notice); // 使用全局组件 notice

使用--在test.vue中

<template>
  <div>
    <el-button type="danger" @click="testClick">
      Notice全局组件使用
    </el-button>
    <el-button type="danger" @click="NoticeClick">
      Notice单独引入使用
    </el-button>
  </div>
</template>

<script>
import { Notice } from "@/components/global/Notice"; //这个是 index.js
export default {
  name: "test",
  methods: {
    NoticeClick() {
      let num = Math.floor(Math.random() * (1 - 10) + 10);
      let type = "";
      if (num <= 3) {
        type = "success";
      } else if (num <= 6) {
        type = "warning";
      } else {
        type = "error";
      }
      Notice({
        message: "提交成功!",
        type
      });
    },
    testClick() {
      let num = Math.floor(Math.random() * (1 - 10) + 10);
      let type = "";
      if (num <= 3) {
        type = "success";
      } else if (num <= 6) {
        type = "warning";
      } else {
        type = "error";
      }
      this.$notice({
        message: "提交成功!",
        type
      });
      //   this.$notify({
      //     title: "成功",
      //     message: "这是一条成功的提示消息",
      //     type
      //   });
    }
  }
};
</script>

上面引入定义的组件用了两种方式

import helloWorld from "./index.vue";
const HelloWorldConstructor = Vue.extend(helloWorld);

等同于

const NoticeConstructor = Vue.extend(require("./notice.vue").default);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容