經過 自建vue組件 air-ui (5) -- 建立第一個組件 Button 咱們已經知道怎麼建立一個標籤類型的組件了。本節咱們就繼續講怎麼建立服務類型的組件。javascript
以 notification
這個組件爲例,他就是一個典型的內置服務組件。其實就是綁定到Vue的原型上,當作全局方法來使用。css
設置 vue 的全局方法其實很常見,好比發 ajax
請求時喜歡用axios
掛載到vue
原型上,以下:html
// 1 引入vue和axios
import Vue from 'vue'
import axios from 'axios'
// 2 對axios的一些封裝
// code ...
// 3 而後掛載到原型上
Vue.prototype.$axios = axios
複製代碼
用的時候就直接上 this.$axios
vue
// 用axios.get()方法能夠這樣用
this.$axios.get()
複製代碼
這樣確實方便,不用每一個用到 axios
的組件都去引入。 這種方法其實很簡單,而內置的服務也是差很少的原理,只不過會涉及到一些 dom
的操做, 以 notification
爲例,這邊能夠看效果: element-ui 的 Notification 通知java
這邊主要分爲三個步驟,具體的目錄結構以下:ios
components/
| |--- notification/
| | |--- src/
| | | |--- main.js
| | | |--- main.vue
| | |--- index.js
複製代碼
由於涉及到 dom 操做,因此第一步就是先建立一個 notification 對應的 vue 組件, 就是 notification/src/main.vue
:ajax
<template>
<transition name="air-notification-fade">
<div
:class="['air-notification', customClass, horizontalClass]"
v-show="visible"
:style="positionStyle"
@mouseenter="clearTimer()"
@mouseleave="startTimer()"
@click="click"
role="alert"
>
<i
class="air-notification__icon"
:class="[ typeClass, iconClass ]"
v-if="type || iconClass">
</i>
<div class="air-notification__group" :class="{ 'is-with-icon': typeClass || iconClass }">
<h2 class="air-notification__title" v-text="title"></h2>
<div class="air-notification__content" v-show="message">
<slot>
<p v-if="!dangerouslyUseHTMLString">{{ message }}</p>
<p v-else v-html="message"></p>
</slot>
</div>
<div
class="air-notification__closeBtn air-icon-close"
v-if="showClose"
@click.stop="close"></div>
</div>
</div>
</transition>
</template>
<script type="text/babel">
let typeMap = {
success: 'success',
info: 'info',
warning: 'warning',
error: 'error'
};
export default {
data() {
return {
visible: false,
title: '',
message: '',
duration: 4500,
type: '',
showClose: true,
customClass: '',
iconClass: '',
onClose: null,
onClick: null,
closed: false,
verticalOffset: 0,
timer: null,
dangerouslyUseHTMLString: false,
position: 'top-right'
};
},
computed: {
typeClass() {
return this.type && typeMap[this.type] ? `air-icon-${ typeMap[this.type] }` : '';
},
horizontalClass() {
return this.position.indexOf('right') > -1 ? 'right' : 'left';
},
verticalProperty() {
return /^top-/.test(this.position) ? 'top' : 'bottom';
},
positionStyle() {
return {
[this.verticalProperty]: `${ this.verticalOffset }px`
};
}
},
watch: {
closed(newVal) {
if (newVal) {
this.visible = false;
this.$el.addEventListener('transitionend', this.destroyElement);
}
}
},
methods: {
destroyElement() {
this.$el.removeEventListener('transitionend', this.destroyElement);
this.$destroy(true);
this.$el.parentNode.removeChild(this.$el);
},
click() {
if (typeof this.onClick === 'function') {
this.onClick();
}
},
close() {
this.closed = true;
if (typeof this.onClose === 'function') {
this.onClose();
}
},
clearTimer() {
clearTimeout(this.timer);
},
startTimer() {
if (this.duration > 0) {
this.timer = setTimeout(() => {
if (!this.closed) {
this.close();
}
}, this.duration);
}
},
keydown(e) {
if (e.keyCode === 46 || e.keyCode === 8) {
this.clearTimer(); // detele 取消倒計時
} else if (e.keyCode === 27) { // esc關閉消息
if (!this.closed) {
this.close();
}
} else {
this.startTimer(); // 恢復倒計時
}
}
},
mounted() {
if (this.duration > 0) {
this.timer = setTimeout(() => {
if (!this.closed) {
this.close();
}
}, this.duration);
}
document.addEventListener('keydown', this.keydown);
},
beforeDestroy() {
document.removeEventListener('keydown', this.keydown);
}
};
</script>
複製代碼
具體邏輯不細說了,他就是普通的一個 vue 組件,無非就是一個 dom 結構,還有就是各類事件和參數的觸發和監聽,不難理解,能夠理解爲 notification 這個組件的 dom 渲染。element-ui
既然 dom 已經準備好了,接下來只要構建 Notification
的結構體就好了,而後讓這個結構體的各類方法去操做 dom 就好了。 因此 notification/src/main.js
:axios
import Vue from 'vue';
import Main from './main.vue';
import merge from '../../../../src/utils/merge';
import { PopupManager } from '../../../../src/utils/popup';
import { isVNode } from '../../../../src/utils/vdom';
const NotificationConstructor = Vue.extend(Main);
let instance;
let instances = [];
let seed = 1;
const Notification = function(options) {
if (Vue.prototype.$isServer) return;
options = merge({}, options);
const userOnClose = options.onClose;
const id = 'notification_' + seed++;
const position = options.position || 'top-right';
options.onClose = function() {
Notification.close(id, userOnClose);
};
instance = new NotificationConstructor({
data: options
});
if (isVNode(options.message)) {
instance.$slots.default = [options.message];
options.message = 'REPLACED_BY_VNODE';
}
instance.id = id;
instance.$mount();
document.body.appendChild(instance.$el);
instance.visible = true;
instance.dom = instance.$el;
instance.dom.style.zIndex = PopupManager.nextZIndex();
let verticalOffset = options.offset || 0;
instances.filter(item => item.position === position).forEach(item => {
verticalOffset += item.$el.offsetHeight + 16;
});
verticalOffset += 16;
instance.verticalOffset = verticalOffset;
instances.push(instance);
return instance;
};
['success', 'warning', 'info', 'error'].forEach(type => {
Notification[type] = options => {
if (typeof options === 'string' || isVNode(options)) {
options = {
message: options
};
}
options.type = type;
return Notification(options);
};
});
Notification.close = function(id, userOnClose) {
let index = -1;
const len = instances.length;
const instance = instances.filter((instance, i) => {
if (instance.id === id) {
index = i;
return true;
}
return false;
})[0];
if (!instance) return;
if (typeof userOnClose === 'function') {
userOnClose(instance);
}
instances.splice(index, 1);
if (len <= 1) return;
const position = instance.position;
const removedHeight = instance.dom.offsetHeight;
for (let i = index; i < len - 1; i++) {
if (instances[i].position === position) {
instances[i].dom.style[instance.verticalProperty] =
parseInt(instances[i].dom.style[instance.verticalProperty], 10) - removedHeight - 16 + 'px';
}
}
};
Notification.closeAll = function() {
for (let i = instances.length - 1; i >= 0; i--) {
instances[i].close();
}
};
export default Notification;
複製代碼
這個邏輯也不難理解,主要分一下幾個步驟:bash
Vue.extend
生成一個 子類
const NotificationConstructor = Vue.extend(Main);
複製代碼
這個能夠理解爲 notification
的構造函數,只要 new
初始化這個函數,就能夠生成一個 notification
的實例化對象:
instance = new NotificationConstructor({
data: options
});
複製代碼
因此 Notification
這個對象其實就是工廠函數,只要被調用,那麼就會實例化一個擁有 notification vue 組件的全新的對象。
instance.$mount(); //沒有 el 參數,就會掛載到一個未掛載的實例,模板將被渲染爲文檔以外的的元素
document.body.appendChild(instance.$el); // 這時候必須使用原生 DOM API 把它插入文檔中
複製代碼
Notification
這個對象的其餘方法,好比 close
之類的,同時還有一個處於閉包狀態的全局對象 instances
來存放這些實例化的對象,並經過 id 對這些實例進行操做總的來講,邏輯其實並不複雜。
最後的 index.js 其實就是導出這個結構體對象,notification/index.js
:
import Notification from './src/main.js';
export default Notification;
複製代碼
接下來就是掛載了,同樣寫在 components/index.js
的 install 方法中,不過這一次不用 component 語法
而是直接掛載到 vue 原型上:
import Button from './button'
import ButtonGroup from './button-group'
import Notification from './notification'
const components = {
Button,
ButtonGroup
}
const install = function (Vue) {
Object.keys(components).forEach(key => {
Vue.component(components[key].name, components[key])
})
Vue.prototype.$notify = Notification
}
export default {
install
}
複製代碼
固然也不能忽略了他的 scss 文件,因此 src/styles/index.scss
也要加上:
@import "./base.scss";
@import "./button.scss";
@import "./button-group.scss";
@import "./notification.scss";
複製代碼
接下來就是寫例子了,在 home.vue
補上例子:
<air-button plain @click="open1">可自動關閉</air-button>
<air-button plain @click="open2">不會自動關閉</air-button>
<air-button plain @click="open3">成功</air-button>
<air-button plain @click="open4">警告</air-button>
<air-button plain @click="open5">消息</air-button>
<air-button plain @click="open6">錯誤</air-button>
複製代碼
script 補上對應的觸發方法:
<script>
export default {
data () {
return {
msg: `AIR-UI - 基於vue2.x,可複用UI組件`
}
},
methods: {
open1() {
const h = this.$createElement;
this.$notify({
title: '標題名稱',
message: h('i', { style: 'color: teal'}, '這是一條會消失的提示文案')
});
},
open2() {
this.$notify({
title: '提示',
message: '這是一條不會自動關閉的消息',
duration: 0
});
},
open3() {
this.$notify({
title: '成功',
message: '這是一條成功的提示消息',
type: 'success'
});
},
open4() {
this.$notify({
title: '警告',
message: '這是一條警告的提示消息',
type: 'warning'
});
},
open5() {
this.$notify.info({
title: '消息',
message: '這是一條消息的提示消息'
});
},
open6() {
this.$notify.error({
title: '錯誤',
message: '這是一條錯誤的提示消息'
});
}
}
}
</script>
複製代碼
注意調用的方式: this.$nofify
,接下來就能夠看效果了:
這樣子一個內置服務的組件,就建立成功了。
這邊要注意一點的是,內置服務的組件,不能使用 Vue.use()
來調用,不然項目運行會默認執行一次,即便沒有使用它們,舉個例子,好比我在 src/main.js
改爲換成對 notification
顯示掛載和 use 的引用:
import Notification from './components/notification'
Vue.prototype.$notify = Notification
Vue.use(Notification)
複製代碼
這樣子,編譯是沒問題的,可是我加載頁面的時候,會默認彈出一個空的彈出框??
這個是由於我調用 use
的時候,就會默認執行一次 Notification
對象,從而實例化一個 notification
的對象,可是由於沒有傳入任何參數,因此是一個空的彈出框。 因此對於這種內置服務組件,好比 Notification
, Message
, MessageBox
, 只要有綁定到 vue 原型對象
就好了,不用 use
來引入。
內置服務組件的建立,大概就這樣。下節咱們講怎麼建立一個指令組件。
系列文章: