国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > JavaScript > 正文

javascript移動(dòng)設(shè)備Web開(kāi)發(fā)中對(duì)touch事件的封裝實(shí)例

2019-11-20 14:56:28
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

在觸屏設(shè)備上,一些比較基礎(chǔ)的手勢(shì)都需要通過(guò)對(duì) touch 事件進(jìn)行二次封裝才能實(shí)現(xiàn)。
zepto 是移動(dòng)端上使用率比較高的一個(gè)類庫(kù),但是其 touch 模塊模擬出來(lái)的一些事件存在一些兼容性問(wèn)題,如 tap 事件在某些安卓設(shè)備上存在事件穿透的 bug,其他類型的事件也或多或少的存在一些兼容性問(wèn)題。

于是乎,干脆自己動(dòng)手對(duì)這些常用的手勢(shì)事件進(jìn)行了封裝,由于沒(méi)有太多真實(shí)的設(shè)備來(lái)進(jìn)行測(cè)試,可能存在一些兼容性問(wèn)題,下面的代碼也只是在 iOS 7、Andorid 4 上的一些比較常見(jiàn)的瀏覽器中測(cè)試通過(guò)。

tap事件

tap 事件相當(dāng)于 pc 瀏覽器中的 click 效果,雖然在觸屏設(shè)備上 click 事件仍然可用,但是在很多設(shè)備上,click 會(huì)存在一些延遲,如果想要快速響應(yīng)的 “click” 事件,需要借助 touch 事件來(lái)實(shí)現(xiàn)。

復(fù)制代碼 代碼如下:

var startTx, startTy;

element.addEventListener( 'touchstart', function( e ){
  var touches = e.touches[0];

  startTx = touches.clientX;
  startTy = touches.clientY;
}, false );

element.addEventListener( 'touchend', function( e ){
  var touches = e.changedTouches[0],
    endTx = touches.clientX,
    endTy = touches.clientY;

  // 在部分設(shè)備上 touch 事件比較靈敏,導(dǎo)致按下和松開(kāi)手指時(shí)的事件坐標(biāo)會(huì)出現(xiàn)一點(diǎn)點(diǎn)變化
  if( Math.abs(startTx - endTx) < 6 && Math.abs(startTy - endTy) < 6 ){
    console.log( 'fire tap event' );
  }
}, false );

doubleTap事件

doubleTap 事件是當(dāng)手指在相同位置范圍內(nèi)和極短的時(shí)間內(nèi)兩次敲擊屏幕時(shí)觸發(fā)的事件。在部分瀏覽器下,doubleTap 事件會(huì)選中文本,如果不希望選中文本,可以給元素添加 user-select:none 的 css 屬性。

復(fù)制代碼 代碼如下:

var isTouchEnd = false,
  lastTime = 0,
  lastTx = null,
  lastTy = null,
  firstTouchEnd = true,
  body = document.body,
  dTapTimer, startTx, startTy, startTime;

element.addEventListener( 'touchstart', function( e ){
  if( dTapTimer ){
    clearTimeout( dTapTimer );
    dTapTimer = null;
  }

  var touches = e.touches[0];

  startTx = touches.clientX;
  startTy = touches.clientY;  
}, false );

element.addEventListener( 'touchend', function( e ){
  var touches = e.changedTouches[0],
    endTx = touches.clientX,
    endTy = touches.clientY,
    now = Date.now(),
    duration = now - lastTime;

  // 首先要確保能觸發(fā)單次的 tap 事件
  if( Math.abs(startTx - endTx) < 6 && Math.abs(startTx - endTx) < 6 ){
    // 兩次 tap 的間隔確保在 500 毫秒以內(nèi)
    if( duration < 301 ){
      // 本次的 tap 位置和上一次的 tap 的位置允許一定范圍內(nèi)的誤差
      if( lastTx !== null &&
        Math.abs(lastTx - endTx) < 45 &&
        Math.abs(lastTy - endTy) < 45 ){

        firstTouchEnd = true;
        lastTx = lastTy = null;
        console.log( 'fire double tap event' );
      }
    }
    else{
      lastTx = endTx;
      lastTy = endTy;
    }
  }
  else{
    firstTouchEnd = true;
    lastTx = lastTy = null;
  }

  lastTime = now;
}, false );

// 在 iOS 的 safari 上手指敲擊屏幕的速度過(guò)快,
// 有一定的幾率會(huì)導(dǎo)致第二次不會(huì)響應(yīng) touchstart 和 touchend 事件
// 同時(shí)手指長(zhǎng)時(shí)間的touch不會(huì)觸發(fā)click

if( ~navigator.userAgent.toLowerCase().indexOf('iphone os') ){

  body.addEventListener( 'touchstart', function( e ){
      startTime = Date.now();
  }, true );

  body.addEventListener( 'touchend', function( e ){
      var noLongTap = Date.now() - startTime < 501;

      if( firstTouchEnd ){
          firstTouchEnd = false;
          if( noLongTap && e.target === element ){
              dTapTimer = setTimeout(function(){
                  firstTouchEnd = true;
                  lastTx = lastTy = null;
                  console.log( 'fire double tap event' );
              }, 400 );
          }
      }
      else{
          firstTouchEnd = true;
      }
  }, true );

// iOS 上手指多次敲擊屏幕時(shí)的速度過(guò)快不會(huì)觸發(fā) click 事件
element.addEventListener( 'click', function( e ){
  if( dTapTimer ){
    clearTimeout( dTapTimer );
    dTapTimer = null;
    firstTouchEnd = true;
  }
}, false );

}

longTap事件

longTap 事件是當(dāng)手指長(zhǎng)時(shí)間按住屏幕保持不動(dòng)時(shí)觸發(fā)的事件。

復(fù)制代碼 代碼如下:

var startTx, startTy, lTapTimer;

element.addEventListener( 'touchstart', function( e ){
  if( lTapTimer ){
    clearTimeout( lTapTimer );
    lTapTimer = null;
  }

  var touches = e.touches[0];

  startTx = touches.clientX;
  startTy = touches.clientY;

  lTapTimer = setTimeout(function(){
    console.log( 'fire long tap event' );
  }, 1000 );

  e.preventDefault();
}, false );

element.addEventListener( 'touchmove', function( e ){
  var touches = e.touches[0],
    endTx = touches.clientX,
    endTy = touches.clientY;

  if( lTapTimer && (Math.abs(endTx - startTx) > 5 || Math.abs(endTy - startTy) > 5) ){
    clearTimeout( lTapTimer );
    lTapTimer = null;
  }
}, false );

element.addEventListener( 'touchend', function( e ){
  if( lTapTimer ){
    clearTimeout( lTapTimer );
    lTapTimer = null;
  }
}, false );

swipe事件

swipe 事件是當(dāng)手指在屏幕上滑動(dòng)后觸發(fā)的事件,根據(jù)手指滑動(dòng)的方向又分為 swipeLeft (向左)、swipeRight (向右)、swipeUp (向上)、swipeDown (向下)。

復(fù)制代碼 代碼如下:

var isTouchMove, startTx, startTy;

element.addEventListener( 'touchstart', function( e ){
  var touches = e.touches[0];

  startTx = touches.clientX;
  startTy = touches.clientY;
  isTouchMove = false;
}, false );

element.addEventListener( 'touchmove', function( e ){
  isTouchMove = true;
  e.preventDefault();
}, false );

element.addEventListener( 'touchend', function( e ){
  if( !isTouchMove ){
    return;
  }

  var touches = e.changedTouches[0],
    endTx = touches.clientX,
    endTy = touches.clientY,
    distanceX = startTx - endTx
    distanceY = startTy - endTy,
    isSwipe = false;

  if( Math.abs(distanceX) >= Math.abs(distanceY) ){
    if( distanceX > 20 ){
      console.log( 'fire swipe left event' );
      isSwipe = true;
    }
    else if( distanceX < -20 ){
      console.log( 'fire swipe right event' );   
      isSwipe = true;
    }
  }
  else{
    if( distanceY > 20 ){
      console.log( 'fire swipe up event' );       
      isSwipe = true;
    }
    else if( distanceY < -20 ){
      console.log( 'fire swipe down event' );        
      isSwipe = true;
    }
  }

  if( isSwipe ){
    console.log( 'fire swipe event' );
  }
}, false );

上面模擬的事件都封裝在 MonoEvent 中了。完整代碼地址:https://github.com/chenmnkken/monoevent,需要的朋友看看吧~

 PS:這里再為大家推薦一款關(guān)于JS事件的在線查詢工具,歸納總結(jié)了JS常用的事件類型與函數(shù)功能:

javascript事件與功能說(shuō)明大全:

http://tools.VeVB.COm/table/javascript_event

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 南投县| 天镇县| 定州市| 射洪县| 桐庐县| 宁都县| 罗城| 册亨县| 玉山县| 塘沽区| 吴旗县| 灵石县| 松阳县| 宜丰县| 贵港市| 乳山市| 浪卡子县| 西吉县| 芜湖市| 合作市| 武功县| 宿松县| 沙雅县| 措勤县| 重庆市| 青州市| 桐梓县| 长子县| 漠河县| 监利县| 陆川县| 博兴县| 兴义市| 仙桃市| 小金县| 武平县| 永清县| 隆安县| 福海县| 昌图县| 洛宁县|