成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

移動(dòng)端點(diǎn)擊事件全攻略,這里的坑你知多少?

Achilles / 2979人閱讀

摘要:所以這種情況下是不符合點(diǎn)擊事件的定義的。,關(guān)于移動(dòng)端的點(diǎn)擊事件總結(jié)完了,可能你都沒想到一個(gè)簡單的點(diǎn)擊事件會(huì)有那么多坑,如果你在工作中可能會(huì)涉及到移動(dòng)端開發(fā)的話,相信這篇文章還是值得你點(diǎn)贊和收藏的,畢竟是踩了那么多坑的經(jīng)驗(yàn)總結(jié)。

看標(biāo)題的時(shí)候你可能會(huì)想,點(diǎn)擊事件有什么好說的,還寫一篇攻略?哈哈,如果你這么想,只能說明你too young to simple.

接觸過移動(dòng)端開發(fā)的同學(xué)可能都會(huì)面臨點(diǎn)擊事件的第一個(gè)問題:click事件的300ms延遲響應(yīng)。不能立即響應(yīng)給體驗(yàn)造成了很大的困擾,因此解決這個(gè)問題就成為了必然。

這個(gè)問題的解決方案就是:zepto.js的tap事件。tap事件可以理解為在移動(dòng)端的click事件,而zepto.js因?yàn)閹缀跬耆珡?fù)制jQuery的api,因此常常被用在h5的開發(fā)上用來取代jquery.

由于模塊化的原因,導(dǎo)致有的同學(xué)下載的zepto.js的模塊并不全,造成了一大堆悲劇,以為zepto.js不支持一些方法,所以,下載的時(shí)候注意一點(diǎn)。

然而事情到這里并沒有結(jié)束,因?yàn)閠ap事件解決了一個(gè)300ms延遲問題,卻帶來了一個(gè)新的重大bug,點(diǎn)擊穿透。

點(diǎn)擊穿透的意思,就是如果一個(gè)絕對定位或者固定定位元素處于頁面最頂層,對這個(gè)元素綁定一個(gè)點(diǎn)擊事件,那么你點(diǎn)擊這個(gè)點(diǎn)對應(yīng)的下面凡是有點(diǎn)擊事件或者a標(biāo)簽都會(huì)被觸發(fā)執(zhí)行。這里就不貼圖了,自行腦補(bǔ)各種彈窗,這種情況還是非常多的。

為了解決這個(gè)問題,有的人試圖用touchend來搞定。touchend會(huì)在手指離開手機(jī)屏幕時(shí)觸發(fā)一次。沒有300ms延遲,沒有點(diǎn)擊穿透,看上去簡直就是完美的解決方案??墒?!

先在這里簡單總結(jié)一個(gè)小知識點(diǎn)。

移動(dòng)端有touchstart, touchmove, touchend, 以及tap
pc端有mousedown, mousemove, mouseup, click。

他們的關(guān)系和作用幾乎可以對應(yīng)起來,分別表示按下,滑動(dòng),松開。pc端可以用前面三個(gè)事件實(shí)現(xiàn)拖拽,移動(dòng)端也可以用前面三個(gè)事件實(shí)現(xiàn)的滑動(dòng)。

所以,在pc端不用mouseup來替換click,就是因?yàn)樗谒砷_鼠標(biāo)的時(shí)候就會(huì)觸發(fā),導(dǎo)致如果我在很遠(yuǎn)的區(qū)域滑動(dòng)到目標(biāo)元素,然后松開,這樣的情況也會(huì)觸發(fā)mouseup與touchend事件。所以這種情況下是不符合點(diǎn)擊事件的定義的。而且如果你在某種情況下,對某一個(gè)事件需要同時(shí)綁定拖拽和點(diǎn)擊,就更加沒辦法解決了。

另外還有一個(gè)很重要的原因?qū)е聇ouchend不能用來替換點(diǎn)擊,是因?yàn)镻C端不支持。老板們常常希望自己的頁面不僅僅能夠在移動(dòng)端展示,因此還得想其他辦法。

我知道有經(jīng)驗(yàn)的同學(xué)讀這篇文章的時(shí)候,早就在想fastclick.js了。是的,目前來看,這是一個(gè)非常好的解決方案。為了解決300ms延遲的問題,zepto.js給出了tap事件替換的方案,而fastclick.js則是在想辦法讓click事件的延遲消除。因此任然是使用click事件,也就不會(huì)有點(diǎn)擊穿透的問題。

首先想辦法引入fastclick.js

如果你使用原生js開發(fā)則進(jìn)行如下聲明即可。

if ("addEventListener" in document) {
    document.addEventListener("DOMContentLoaded", function() {
        FastClick.attach(document.body);
    }, false);
}

如果你想使用jquery

$(function() {
    FastClick.attach(document.body);
});

如果你在使用CommonJS風(fēng)格的框架,比如requirejs

var attachFastClick = require("fastclick");
attachFastClick(document.body);

AMD

var FastClick = require("fastclick");
FastClick.attach(document.body, options);

進(jìn)行對應(yīng)的聲明之后,你就可以在移動(dòng)端頁面中放心大膽的使用click事件了。說到這里,就會(huì)有一個(gè)關(guān)于zepto.js與jquery選擇的問題。說起來又可以寫一大篇文章了,簡單來說就是,實(shí)際開發(fā)中你就會(huì)發(fā)現(xiàn)zepto還是有點(diǎn)不太爽,雖然體積小點(diǎn),既然click延遲的問題已經(jīng)解決了,我還是更偏向于使用jquery.

當(dāng)然,我們要踩的坑并沒有結(jié)束 - -!

最近開發(fā)了一個(gè)小頁面,財(cái)經(jīng)日歷,在日歷部分,我需要同時(shí)給代表每一天的元素事件實(shí)現(xiàn)獲取當(dāng)天資訊的需求,又要給整個(gè)日歷部分實(shí)現(xiàn)能夠左右滑動(dòng)來選取上一月和下一月的功能。所以我需要同時(shí)對日歷部分綁定click事件和實(shí)現(xiàn)滑動(dòng)的touchstart,touchmove,touchend事件。

這個(gè)時(shí)候問題出現(xiàn)了。在安卓手機(jī)上,對同一個(gè)元素,如果我綁定了click事件,然后在綁定touchstart事件,click事件會(huì)處于幾乎失效的狀態(tài)。就算用了fastclick事件也無法避免這個(gè)問題

錯(cuò)誤演示大概如下

$area.on("click", ".weeknumber", function() {
    // 點(diǎn)擊每一天獲取當(dāng)天資訊
})

// 實(shí)現(xiàn)左右滑動(dòng)
$area.on("touchstart", function() {})
.on("touchmove", function() {})
.on("touchend", function() {})

動(dòng)手能力強(qiáng)的同學(xué)可以去試試這個(gè)坑,這種情況下,fastfclick肯定是沒辦法解決的。怎么辦?

我的第一次嘗試,是在當(dāng)滑動(dòng)距離為0的時(shí)候,運(yùn)行點(diǎn)擊事件里面的內(nèi)容。我們知道在實(shí)現(xiàn)滑動(dòng)[不知道如何實(shí)現(xiàn)的同學(xué),是時(shí)候關(guān)注我的公眾號了,搜索isreact找到我]的時(shí)候,會(huì)計(jì)算一個(gè)滑動(dòng)距離。

// 實(shí)現(xiàn)滑動(dòng)的大概代碼

// 滑動(dòng)元素translateX的初始值
var iscroll = device_width,

    // 用來計(jì)算的中間值
    istarX = 0,
    
    // 手指第一次點(diǎn)在屏幕上的x坐標(biāo)
    istart_pageX = 0; 
   
// 綁定事件
$area.on("touchstart", touchstart)
     .on("touchmove", touchmove)
     .on("touchend", touchend);

function touchstart(event) {
    event.preventDefault();
    istartX = iscroll;
    istart_pageX = event.originalEvent.changedTouches[0].pageX;
}

function touchmove(event) {

    // 滑動(dòng)過程中手指位置x坐標(biāo)會(huì)不停變動(dòng),這里會(huì)保存一個(gè)當(dāng)前位置與初始位置的一個(gè)差值
    var distance = event.originalEvent.changedTouches[0].pageX - istart_pageX;
    iscroll = istartX + distance;

    // 這里是我自定義的一個(gè)css方法,用來設(shè)置元素translateX的當(dāng)前值
    Utils.css(area, { translateX: iscroll });
}

function touchend(event) {
    var distance = event.originalEvent.changedTouches[0].pageX - istart_pageX;
    $area.off("touchstart touchmove touchend");

    // 根據(jù)差值的不同,執(zhí)行不同的動(dòng)作
    if (distance < -80) {
        slideNext(function() {
            addEventSlider($area);
        })
    }
    else if (distance > 80) {
        slidePrev(function() {
            addEventSlider($area);
        })
    }
    else if (distance == 0) {        
        /* 當(dāng)差值為0時(shí),我認(rèn)為這是執(zhí)行了一次點(diǎn)擊 */
        
        addEventSlider($area);
    }
    else {
        ani(area).animate(400, "easeout", { x: -device_width }, function() {
            iscroll = -device_width;
            addEventSlider($area);
        })
    }
}

上面就是我的滑動(dòng)功能的實(shí)現(xiàn),中間會(huì)有一些自定義的方法,因此沒辦法你們直接就復(fù)制過去運(yùn)行。但是原理已經(jīng)講得還算明白,可以自己嘗試一些簡單的實(shí)現(xiàn)。關(guān)注我公眾號會(huì)有更加詳細(xì)的講解哦!

我的這一次嘗試,就是在滑動(dòng)差值distance為0的時(shí)候,認(rèn)為這是一次點(diǎn)擊,因此執(zhí)行點(diǎn)擊事件應(yīng)該有的動(dòng)作。本來我以為這就能夠解決了,測試的時(shí)候也通過了。但是 - -!產(chǎn)品同學(xué)非常有執(zhí)念的在日歷上點(diǎn)擊了50多下,結(jié)果發(fā)現(xiàn),多次點(diǎn)擊之后,在點(diǎn)就失去效果了?。。?/p>

當(dāng)發(fā)現(xiàn)這個(gè)bug的時(shí)候,我的內(nèi)心是崩潰的。好吧,只能說,試圖用touchend來代替點(diǎn)擊事件的想法終究還是有一點(diǎn)不成熟。硬著頭皮想辦法繼續(xù)解決上面的問題。再三思考各種解決方案,最終決定自己封裝一個(gè)tap事件。封裝的tap事件的代碼如下。

http://yangbo5207.github.io/s...

這是對jquery事件的一個(gè)拓展,讓jquery也能夠使用tap事件。放在jquery后面引入就能夠立即使用了。除此之外還擴(kuò)展了longTap,swipe兩個(gè)事件。

使用方式和其他一樣。

$area.on("tap", function() {});

$area.tap(function() {})

當(dāng)然我封裝的這個(gè)tap,任然有避免不了的點(diǎn)擊穿透bug,所以,最終的解決方案是:對于需要同時(shí)綁定點(diǎn)擊事件和滑動(dòng)事件的元素,用tap事件,其他情況都用click事件即可。需要結(jié)合我的tap.js與fastclick.js來完美解決這個(gè)問題。心累啊,終于是搞定了。

OK,關(guān)于移動(dòng)端的點(diǎn)擊事件總結(jié)完了,可能你都沒想到一個(gè)簡單的點(diǎn)擊事件會(huì)有那么多坑,如果你在工作中可能會(huì)涉及到移動(dòng)端開發(fā)的話,相信這篇文章還是值得你點(diǎn)贊和收藏的,畢竟是踩了那么多坑的經(jīng)驗(yàn)總結(jié)。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/116380.html

相關(guān)文章

  • 移動(dòng)點(diǎn)擊事件攻略這里的坑你知多少?

    摘要:所以這種情況下是不符合點(diǎn)擊事件的定義的。,關(guān)于移動(dòng)端的點(diǎn)擊事件總結(jié)完了,可能你都沒想到一個(gè)簡單的點(diǎn)擊事件會(huì)有那么多坑,如果你在工作中可能會(huì)涉及到移動(dòng)端開發(fā)的話,相信這篇文章還是值得你點(diǎn)贊和收藏的,畢竟是踩了那么多坑的經(jīng)驗(yàn)總結(jié)。 看標(biāo)題的時(shí)候你可能會(huì)想,點(diǎn)擊事件有什么好說的,還寫一篇攻略?哈哈,如果你這么想,只能說明你too young to simple. 接觸過移動(dòng)端開發(fā)的同學(xué)可能都...

    Nosee 評論0 收藏0
  • 移動(dòng)點(diǎn)擊事件攻略這里的坑你知多少?

    摘要:所以這種情況下是不符合點(diǎn)擊事件的定義的。,關(guān)于移動(dòng)端的點(diǎn)擊事件總結(jié)完了,可能你都沒想到一個(gè)簡單的點(diǎn)擊事件會(huì)有那么多坑,如果你在工作中可能會(huì)涉及到移動(dòng)端開發(fā)的話,相信這篇文章還是值得你點(diǎn)贊和收藏的,畢竟是踩了那么多坑的經(jīng)驗(yàn)總結(jié)。 看標(biāo)題的時(shí)候你可能會(huì)想,點(diǎn)擊事件有什么好說的,還寫一篇攻略?哈哈,如果你這么想,只能說明你too young to simple. 接觸過移動(dòng)端開發(fā)的同學(xué)可能都...

    microelec 評論0 收藏0
  • 網(wǎng)頁單位和rem使用小結(jié)

    摘要:像素像素是一個(gè)相對單位。相對不同屏幕,其實(shí)際像素大小不同。解決方案直接使用實(shí)現(xiàn)的終端適配有興趣的小伙伴可以看下的解決方案使用實(shí)現(xiàn)手淘頁面的終端適配地址移動(dòng)端適配單位的坑你知道多少關(guān)于移動(dòng)端布局的一些總結(jié) 網(wǎng)頁尺寸單位 百分比(%) 英寸(in) 厘米(cm) 毫米(mm) 磅數(shù)(pt) 12 點(diǎn)活字(pc) 字母高度一半(ex) 父級字體(em) 像素(px) 根元素字體(rem) ...

    王巖威 評論0 收藏0
  • _JavaScript

    摘要:為此決定自研一個(gè)富文本編輯器。例如當(dāng)要轉(zhuǎn)化的對象有環(huán)存在時(shí)子節(jié)點(diǎn)屬性賦值了父節(jié)點(diǎn)的引用,為了關(guān)于函數(shù)式編程的思考作者李英杰,美團(tuán)金融前端團(tuán)隊(duì)成員。只有正確使用作用域,才能使用優(yōu)秀的設(shè)計(jì)模式,幫助你規(guī)避副作用。 JavaScript 專題之惰性函數(shù) JavaScript 專題系列第十五篇,講解惰性函數(shù) 需求 我們現(xiàn)在需要寫一個(gè) foo 函數(shù),這個(gè)函數(shù)返回首次調(diào)用時(shí)的 Date 對象,注意...

    Benedict Evans 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<