Vue組件之Tooltip的示例代碼
前言
本文主要Alert 組件的大致框架, 提供少量可配置選項(xiàng)。 旨在大致提供思路
tooltip
常用于展示鼠標(biāo) hover 時(shí)的提示信息。
模板結(jié)構(gòu)
<template>
<div style="position:relative;">
<span ref="trigger">
<slot>
</slot>
</span>
<div class="tooltip"
v-bind:class="{
'top': placement === 'top',
'left': placement === 'left',
'right': placement === 'right',
'bottom': placement === 'bottom',
'disable': type === 'disable',
'delete': type === 'delete',
'visible': show === true
}"
ref="popover"
role="tooltip">
<div class="tooltip-arrow"></div>
<div class="tooltip-inner">
<slot name="content" v-html="content"></slot>
</div>
</div>
</div>
</template>
大致結(jié)構(gòu)DOM結(jié)構(gòu) 一個(gè)div 包含 箭頭 及 氣泡內(nèi)容。
v-bind中可選tooltip位置,是否禁用,及顯示隱藏
slot 差值供自定義 默認(rèn)接收content內(nèi)容
script
import EventListener from '../utils/EventListener.js';
export default {
props: {
// 需要監(jiān)聽(tīng)的事件
trigger: {
type: String,
default: 'click'
},
effect: {
type: String,
default: 'fadein'
},
title: {
type: String
},
// toolTip消息提示
content: {
type: String
},
header: {
type: Boolean,
default: true
},
placement: {
type: String
}
},
data() {
return {
// 通過(guò)計(jì)算所得 氣泡位置
position: {
top: 0,
left: 0
},
show: true
};
},
watch: {
show: function(val) {
if (val) {
const popover = this.$refs.popover;
const triger = this.$refs.trigger.children[0];
// 通過(guò)placement計(jì)算出位子
switch (this.placement) {
case 'top' :
this.position.left = triger.offsetLeft - popover.offsetWidth / 2 + triger.offsetWidth / 2;
this.position.top = triger.offsetTop - popover.offsetHeight;
break;
case 'left':
this.position.left = triger.offsetLeft - popover.offsetWidth;
this.position.top = triger.offsetTop + triger.offsetHeight / 2 - popover.offsetHeight / 2;
break;
case 'right':
this.position.left = triger.offsetLeft + triger.offsetWidth;
this.position.top = triger.offsetTop + triger.offsetHeight / 2 - popover.offsetHeight / 2;
break;
case 'bottom':
this.position.left = triger.offsetLeft - popover.offsetWidth / 2 + triger.offsetWidth / 2;
this.position.top = triger.offsetTop + triger.offsetHeight;
break;
default:
console.log('Wrong placement prop');
}
popover.style.top = this.position.top + 'px';
popover.style.left = this.position.left + 'px';
}
}
},
methods: {
toggle() {
this.show = !this.show;
}
},
mounted() {
if (!this.$refs.popover) return console.error("Couldn't find popover ref in your component that uses popoverMixin.");
// 獲取監(jiān)聽(tīng)對(duì)象
const triger = this.$refs.trigger.children[0];
// 根據(jù)trigger監(jiān)聽(tīng)特定事件
if (this.trigger === 'hover') {
this._mouseenterEvent = EventListener.listen(triger, 'mouseenter', () => {
this.show = true;
});
this._mouseleaveEvent = EventListener.listen(triger, 'mouseleave', () => {
this.show = false;
});
} else if (this.trigger === 'focus') {
this._focusEvent = EventListener.listen(triger, 'focus', () => {
this.show = true;
});
this._blurEvent = EventListener.listen(triger, 'blur', () => {
this.show = false;
});
} else {
this._clickEvent = EventListener.listen(triger, 'click', this.toggle);
}
this.show = !this.show;
},
// 在組件銷(xiāo)毀前移除監(jiān)聽(tīng),釋放內(nèi)存
beforeDestroy() {
if (this._blurEvent) {
this._blurEvent.remove();
this._focusEvent.remove();
}
if (this._mouseenterEvent) {
this._mouseenterEvent.remove();
this._mouseleaveEvent.remove();
}
if (this._clickEvent) this._clickEvent.remove();
}
};
// EventListener.js
const EventListener = {
/**
* Listen to DOM events during the bubble phase.
*
* @param {DOMEventTarget} target DOM element to register listener on.
* @param {string} eventType Event type, e.g. 'click' or 'mouseover'.
* @param {function} callback Callback function.
* @return {object} Object with a `remove` method.
*/
listen(target, eventType, callback) {
if (target.addEventListener) {
target.addEventListener(eventType, callback, false);
return {
remove() {
target.removeEventListener(eventType, callback, false);
}
};
} else if (target.attachEvent) {
target.attachEvent('on' + eventType, callback);
return {
remove() {
target.detachEvent('on' + eventType, callback);
}
};
}
}
};
export default EventListener;
封裝的事件監(jiān)聽(tīng)
使用
使用content屬性來(lái)決定hover時(shí)的提示信息。由placement屬性決定展示效果:placement屬性值為:方向-對(duì)齊位置;四個(gè)方向:top、left、right、bottom。trigger屬性用于設(shè)置觸發(fā)tooltip的方式,默認(rèn)為hover。
<d-tooltip content="我是tooltip"> <d-button type="text">鼠標(biāo)移動(dòng)到我上面試試</d-button> </d-tooltip> <d-tooltip content="我是tooltip" trigger="click"> <d-button type="text">點(diǎn)我試試</d-button> </d-tooltip>
content內(nèi)容分發(fā)
設(shè)置一個(gè)名為content的slot。
<d-tooltip> <d-button type="text">鼠標(biāo)移動(dòng)到我上面試試</d-button> <p slot="content" class="tooltip-content">我是內(nèi)容分發(fā)的conent。</p> </d-tooltip>
Attributes
| 參數(shù) | 說(shuō)明 | 類(lèi)型 | 可選值 | 默認(rèn)值 |
|---|---|---|---|---|
| content | 顯示的內(nèi)容,也可以通過(guò) slot#content 傳入 DOM | String | — | — |
| placement | Tooltip 的出現(xiàn)位置 | String | top/right/bottom/left | top |
| trigger | tooltip觸發(fā)方式 | String | — | hover |
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
axios簡(jiǎn)單實(shí)現(xiàn)小程序延時(shí)loading指示
這篇文章主要介紹了axios簡(jiǎn)單實(shí)現(xiàn)小程序延時(shí)loading指示,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07
Vue?transition組件簡(jiǎn)單實(shí)現(xiàn)數(shù)字滾動(dòng)
這篇文章主要為大家介紹了Vue?transition組件簡(jiǎn)單實(shí)現(xiàn)數(shù)字滾動(dòng)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09
解決vscode?Better?Comments插件在vue文件中不顯示相對(duì)應(yīng)的顏色的問(wèn)題
最近使用了Better?Comments這款插件,發(fā)現(xiàn)在ts文件中可以顯示對(duì)應(yīng)的顏色,但在vue文件中并不顯示對(duì)應(yīng)顏色?,下面小編給大家分享解決方法,感興趣的朋友跟隨小編一起看看吧2022-09-09
Vue2.0實(shí)現(xiàn)自適應(yīng)分辨率
這篇文章主要為大家詳細(xì)介紹了Vue2.0實(shí)現(xiàn)自適應(yīng)分辨率,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11
在移動(dòng)端使用vue-router和keep-alive的方法示例
這篇文章主要介紹了在移動(dòng)端使用vue-router和keep-alive的方法示例,vue-router與keep-alive提供的路由體驗(yàn)與移動(dòng)端是有一定差別的,感興趣的小伙伴們可以參考一下2018-12-12
vue?首頁(yè)加載,速度優(yōu)化及解決首頁(yè)白屏的問(wèn)題
這篇文章主要介紹了vue?首頁(yè)加載,速度優(yōu)化及解決首頁(yè)白屏的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。2022-10-10
axios+Vue實(shí)現(xiàn)上傳文件顯示進(jìn)度功能
這篇文章主要介紹了axios+Vue上傳文件顯示進(jìn)度效果,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-04-04

