vue自定義tap指令及tap事件的實(shí)現(xiàn)
1.Vue指令
Vue提供自定義實(shí)現(xiàn)指令的功能, 和組件類(lèi)似,可以是全局指令和局部指令,詳細(xì)可以參見(jiàn)vue官網(wǎng)自定義指令一節(jié)(https://cn.vuejs.org/v2/guide/custom-directive.html).
2.v-tap指令實(shí)現(xiàn)
我個(gè)人的理解,編寫(xiě)指令即是在vue指令對(duì)象提供的鉤子函數(shù)中做相應(yīng)的邏輯處理,tap指令是在bind鉤子函數(shù)中做相應(yīng)的處理, 首先,要明白的是tap是為了處理click事件在iphone上的存在300ms的延時(shí),這樣使得連續(xù)點(diǎn)擊很不流暢,tap通過(guò)移動(dòng)端的touchstart事件和touchend事件判斷移動(dòng)距離為零的話,則觸發(fā)綁定的函數(shù),話不多說(shuō),上代碼:
Vue.directive('tap',{
bind(el, binding, vNode){
let expression = binding.value;
let handler = expression.name;
let args = expression.args
on(el, 'touchstart', (e)=>{
let startX = e.changedTouches[0].clientX;
let startY = e.changedTouches[0].clientY;
once(el, 'touchend',(ev)=>{
let disX = Math.abs(ev.changedTouches[0].clientX-startX);
let disY = Math.abs(ev.changedTouches[0].clientY-startY);
if(disX == 0 && disY ==0){
handler(args);
}
})
})
}
})
使用示例: <div v-tap="{ name : mymethod, args:{arg1:11, args2:22} }"></div>
3.總結(jié)
當(dāng)我們需要復(fù)用一些dom底層操作的時(shí)候,可以考慮使用vue directive的方式復(fù)用代碼.
下面看下vue tap事件的實(shí)現(xiàn)代碼
前兩天做了個(gè)tap.js插件,實(shí)現(xiàn)了移動(dòng)端touch事件模擬click事件,解決點(diǎn)擊延遲的問(wèn)題,但是在vue中并不能用v-tap來(lái)調(diào)用,所以今天做了vue版的vue-tap.js。此前也曾用過(guò)其他的插件來(lái)實(shí)現(xiàn)v-tap,但方式仍有些累贅,于是便用了更簡(jiǎn)潔的方式來(lái)實(shí)現(xiàn),下面附上代碼(只支持vue2.0+)。
vue-tap.js
/*!
* vue-tap.js
* by weijianhua https://github.com/weijhfly/vue-tap
*/
;(function (factory) {
if (typeof define === 'function' && define.amd) {
define(function(){return factory;});
}else if (typeof exports == "object") {
module.exports = factory;
}else{
Vue.use(factory);
}
}({
master:{
bind: function (el, binding) {
var isTouch = "ontouchend" in document;
el.exec = function (e) {
var data = binding.value;
data[0].apply(this, data.slice(1));
};
if (isTouch) {
//touchstart
el.addEventListener('touchstart', function (e) {
binding.modifiers.stop && (e.stopPropagation());
var t = e.touches[0];
el.startX = t.clientX;
el.startY = t.clientY;
el.sTime = + new Date;
});
//touchend
el.addEventListener('touchend', function (e) {
binding.modifiers.stop && (e.stopPropagation());
var t = e.changedTouches[0];
el.endX = t.clientX;
el.endY = t.clientY;
if((+ new Date)-el.sTime<300){
if(Math.abs(el.endX-el.startX)+Math.abs(el.endY-el.startY)<20){
e.preventDefault();
el.exec();
}
}
});
}else {
//click
el.addEventListener('click', function (e) {
binding.modifiers.stop && (e.stopPropagation());
el.exec();
});
}
},
componentUpdated : function(el,binding) {
el.exec = function () {
var data = binding.value;
data[0].apply(this, data.slice(1));
};
},
unbind: function (el) {
el.exec = null;
}
},
install:function(){
Vue.directive('tap', this.master);
}
}))
demo.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1,maximum-scale=1.0, user-scalable=0" />
<title>vue plugin test</title>
<style type="text/css">
strong{
font-size: 15px;
}
pre{
padding: 16px 0;
overflow: auto;
line-height: 1.45;
background-color: #f6f8fa;
border-radius: 3px;
}
</style>
</head>
<body style="padding:30px;">
<div id="app">
<pre>
<strong>vue-tap.js</strong>
<b>簡(jiǎn)潔的調(diào)用方式:</b>
v-tap="[方法,參數(shù)一,參數(shù)二...]"
<b>獲取參數(shù):</b>
methods:{
tap:function(參數(shù)一,參數(shù)二...){
console.log(參數(shù)一,參數(shù)二...);
}
}
<b>阻止冒泡:</b>
v-tap.stop
</pre>
<hr>
<div v-for="(l,i) in list">
<div v-tap="[tap,l,i]">li-{{l}}-{{i}}</div>
</div>
<br>
<hr>
<div v-tap="[test,'parent']">
parent<br><br>
<button v-tap.stop="[test,'son']">stop propagation</button>
</div>
</div>
<script src="https://cdn.bootcss.com/vue/2.5.13/vue.js"></script>
<script src="vue-tap.js"></script>
<script>
new Vue({
el:'#app',
data:{
list:['a','b','c','e','f']
},
methods:{
tap:function(i,k){
console.log(i,k);
},
test:function(i){
console.log(i);
}
}
})
if(window.innerWidth < 768){
document.getElementsByTagName('body')[0].style.padding = 0;
}
</script>
</body>
</html>
github:https://github.com/weijhfly/vue-tap
參考了其他vue-tap插件,但仍有需要完善的地方,后續(xù)更新。此外,在移動(dòng)端解決點(diǎn)擊延遲問(wèn)題,還是比較推薦fastclick,兼容性較好且方便使用,不過(guò)相對(duì)而言模擬tap事件體積較小,也可以拿來(lái)練手了。
總結(jié)
以上所述是小編給大家介紹的vue自定義tap指令及tap事件的實(shí)現(xiàn),希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
Vuejs使用addEventListener的事件如何觸發(fā)執(zhí)行函數(shù)的this
這篇文章主要介紹了Vuejs使用addEventListener的事件觸發(fā)執(zhí)行函數(shù)的this方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-06-06
vue+element-ui實(shí)現(xiàn)主題切換功能
這篇文章主要介紹了vue+element-ui實(shí)現(xiàn)主題切換功能,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06
vue使用自定義指令實(shí)現(xiàn)一鍵復(fù)制功能
在Vue中,通過(guò)自定義指令v-copy和document.execCommand方法,可以實(shí)現(xiàn)點(diǎn)擊按鈕復(fù)制內(nèi)容到剪貼板的功能,適用于處理長(zhǎng)文本或多行文本的復(fù)制需求,而readonly屬性可避免內(nèi)容被修改和移動(dòng)設(shè)備上的虛擬鍵盤(pán)干擾,感興趣的朋友一起看看吧2024-09-09
vue+elementUI實(shí)現(xiàn)點(diǎn)擊左右箭頭切換按鈕功能
這篇文章主要介紹了vue+elementUI實(shí)現(xiàn)點(diǎn)擊左右箭頭切換按鈕功能,樣式可以根據(jù)自己需求改動(dòng),感興趣的朋友可以參考下實(shí)現(xiàn)代碼2024-05-05
詳解vue中$nextTick和$forceUpdate的用法
這篇文章主要介紹了詳解vue中$nextTick和$forceUpdate的用法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12

