在前端的移动Web开发中,有一部分事件只在移动端产生,如触摸相关的事件。接下来给你们简单总结一下移动端的事件。javascript
因为移动端默认的布局视口宽度是980像素,因此网页文字很是小,为了快速让网页还原到原来的大小,Safari最新引入了双击缩放功能:用户双击手机页面的时候,浏览器会智能的缩放当前页面到原始大小。css
双击缩放的原理就是,当用户click一次以后,浏览器会通过约300ms以后检测是否再有一次click,若是有的话,就会缩放页面。不然的话就是一个click事件。html
因为双击缩放功能存在,click事件触发就会有大约200~300ms的延迟。前端
因为双击缩放的存在,pc端的dblclick事件也失效了。java
因为移动端设备大都具有触摸功能,因此移动端浏览器都引入了触摸(touch)事件。node
touch相关的事件跟普通的其余dom事件同样使用,能够直接用addEventListener来监听和处理。git
最基本的touch事件包括4个事件:github
touchstart: 当在屏幕上按下手指时触发web
touchmove: 当在屏幕上移动手指时触发api
touchend: 当在屏幕上抬起手指时触发
touchcancel 当一些更高级别的事件发生的时候(如电话接入或者弹出信息)会取消当前的touch操做,即触发touchcancel。通常会在touchcancel时暂停游戏、存档等操做。
在不少状况下,触摸事件和鼠标事件会同时被触发(目的是让没有对触摸设备优化的代码仍然能够在触摸设备上正常工做)。
由于双击缩放检测的存在,在移动设备屏幕上点击操做的事件执行顺序:
touchstart(瞬间触发) → touchend → click(200-300ms延迟)
若是你使用了触摸事件,能够调用 event.preventDefault()来阻止鼠标事件被触发。
当用户手指触摸到的触摸屏的时候触发。事件对象的 target 就是touch 发生位置的那个元素。
<div> 点击我! </div> <script> var box = document.querySelector("div"); box.addEventListener("touchstart", function (e) { console.log('touchstart'); }); </script>
当用户在触摸屏上移动触点(手指)的时候,触发这个事件。必定是先要触发touchstart事件,再有可能触发 touchmove 事件。
touchmove 事件的target 与最早触发的 touchstart 的 target 保持一致。touchmove事件和鼠标的mousemove事件同样都会屡次重复调用,因此,事件处理时不能有太多耗时操做。不一样的设备,移动一样的距离 touchmove 事件的触发频率是不一样的。
注意:
- 即便手指移出了 原来的target 元素,则 touchmove 仍然会被一直触发,并且 target 仍然是原来的 target 元素。
- touchmove事件会屡次重复触发,因为移动端计算资源宝贵,尽可能保证事件节流
<div> <p></p> </div> <script> var i = 1; var box = document.querySelector("div"); var p = document.querySelector("p"); box.addEventListener("touchmove", function (e){ p.innerHTML = e.target.tagName + ", " + i++; }) </script>
当用户的手指抬起的时候,会触发 touchend 事件。如何用户的手指从触屏设备的边缘移出了触屏设备,也会触发 touchend 事件。
touchend 事件的 target 也是与 touchstart 的 target 一致,即便已经移出了元素。
当触点因为某些缘由被中断时触发。有几种可能的缘由以下(具体的缘由根据不一样的设备和浏览器有所不一样):
touchcancel 事件通常用于保存现场数据。好比:正在玩游戏,若是发生了 。touchcancel 事件,则应该把游戏当前状态相关的一些数据保存起来。
TouchEvent
是一类描述手指在触摸平面(触摸屏、触摸板等)的状态变化的事件。这类事件用于描述一个或多个触点,使开发者能够检测触点的移动,触点的增长和减小,等等。
每 个 Touch
对象表明一个触点; 每一个触点都由其位置,大小,形状,压力大小,和目标 element
描述。 TouchList
对象表明多个触点的一个列表.
TouchEvent
的属性继承了 UIEvent
和 Event
。
属性列表:
TouchEvent.changedTouches
: 一个 TouchList
对象,包含了表明全部从上一次触摸事件到这次事件过程当中,状态发生了改变的触点的 Touch
对象。
TouchEvent.targetTouches
: 一个 TouchList
对象,是包含了以下触点的 Touch
对象:触摸起始于当前事件的目标 element
上,而且仍然没有离开触摸平面的触点。
TouchEvent.touches
: 一 个 TouchList
对象,包含了全部当前接触触摸平面的触点的 Touch
对象,不管它们的起始于哪一个 element
上,也不管它们状态是否发生了变化。
<style> .box { width: 100px; height: 100px; border: 1px solid #09c; background-color: #0dc; } </style> <div class="box"></div> <script> window.onload = function() { var box = document.querySelector('.box'); box.addEventListener('touchstart', function(e) { console.dir(e); // 查看TouchEvent对象的属性和方法 }); } </script>
一个TouchList
表明一个触摸屏幕上全部触点的列表。
举例来说, 若是一个用户用三根手指接触屏幕(或者触控板), 与之相关的TouchList
对于每根手指都会生成一个 Touch
对象, 共计 3 个.
只读属性:length
返回这个TouchList
中Touch
对的个数。(就是有几个手指接触到了屏幕)
方法:item(index)
返回TouchList
中指定索引的Touch
对象。
<div> <p style="font-size: 50px; color: #ffffff;"></p> </div> <script> var box = document.querySelector("div"); var p = document.querySelector("p"); box.addEventListener("touchend", function (e){ p.innerHTML = e.changedTouches.length; //返回Touch对象的个数 for(var i = 0; i < e.changedTouches.length; i++){ //遍历出来每一个Touch对象 console.log(e.changedTouches.item(i)); } }) </script>
测试多个手机触摸屏幕:
<div></div> <p></p> <script> var div = document.querySelector("div"); var p = document.querySelector("p"); div.addEventListener("touchstart", function (e){ var msg = "touches.length: " + e.touches.length + "<br> targetTouches.length: " + e.targetTouches.length + "<br> changedTouches.length: " + e.changedTouches.length; p.innerHTML = msg; }) </script>
操做:
放1个手指在div上
div
上div
上 Touch
表示用户和触摸设备之间接触时单独的交互点(a single point of contact
)。 这个交互点一般是一个手指或者触摸笔, 触摸设备一般是触摸屏或者触摸板。
基本属性列表(都是只读):
编号 | 属性名 | 属性说明 |
---|---|---|
1. | identifier |
表示每 1 个 Touch 对象 的独一无二的 identifier 。有了这个 identifier 能够确保你总能追踪到这个 Touch 对象。 |
2. | screenX |
触摸点相对于屏幕左边缘的 x 坐标。 |
3. | scre enY |
触摸点相对于屏幕上边缘的 y 坐标。 |
4. | clientX |
触摸点相对于浏览器的 viewport 左边缘的 x 坐标。不会包括左边的滚动距离。 |
5. | clientY |
触摸点相对于浏览器的 viewport 上边缘的 y 坐标。不会包括上边的滚动距离。 |
6. | pageX |
触摸点相对于 document 的左边缘的 x 坐标。 与 clientX 不一样的是,他包括左边滚动的距离,若是有的话。 |
7. | pageY |
触摸点相对于 document 的左边缘的 y 坐标。 与 clientY 不一样的是,他包括上边滚动的距离,若是有的话。 |
8. | target |
老是表示 手指最开始放在触摸设备上的触发点所在位置的 element 。 即便已经移出了元素甚至移出了document , 他表示的element 仍然不变 |
案例:
var box = document.querySelector("div"); var p = document.querySelector("p"); box.ontouchstart = function (e){ var touchList = e.changedTouches; for (var i = 0; i < touchList.length; i++){ var touch = touchList[i]; var msg = `id : ${touch.identifier} <br> screenX : ${touch.screenX} <br> screenY : ${touch.screenY} <br> clientX : ${touch.clientX} <br> clientY : ${touch.clientY} <br> pageX : ${touch.pageX} <br> pageY : ${touch.pageY} <br> target: ${touch.target.nodeName} <br> `; p.innerHTML = msg; } }
没有左右滚动:
左右滚动:pageX
明显大于 clientX
因为点击事件常用,若是用click会有延迟问题,通常咱们会用touch事件模拟移动端的点击事件, 如下是封装的几个事件,仅供参考。
(function (window){ //传入window,提升变量的查找效率 function myQuery(selector){ //这个函数就是对外提供的接口。 //调用这个函数的原型对象上的_init方法,并返回 return myQuery.prototype._init(selector); } myQuery.prototype = { /*初始化方法,获取当前query对象的方法*/ _init: function (selector){ if (typeof selector == "string"){ //把查找到的元素存入到这个原型对象上。 this.ele = window.document.querySelector(selector); //返回值其实就是原型对象。 return this; } }, /*单击事件: * 为了规避click的300ms的延迟,自定义一个单击事件 * 触发时间: * 当抬起手指的时候触发 * 须要判断手指落下和手指抬起的事件间隔,若是小于500ms表示单击时间。 * 若是是大于等于500ms,算是长按时间 * */ tap: function (handler){ this.ele.addEventListener("touchstart", touchFn); this.ele.addEventListener("touchend", touchFn); var startTime, endTime; function touchFn(e){ e.preventDefault() switch (e.type){ case "touchstart": startTime = new Date().getTime(); break; case "touchend": endTime = new Date().getTime(); if (endTime - startTime < 500){ handler.call(this, e); } break; } } }, /** * 长按 * @param handler */ longTag: function (handler){ this.ele.addEventListener("touchstart", touchFn); this.ele.addEventListener("touchmove", touchFn); this.ele.addEventListener("touchend", touchFn); var timerId; function touchFn(e){ switch (e.type){ case "touchstart" : //500ms以后执行 timerId = setTimeout(function (){ handler.call(this, e); }, 500) break; case "touchmove" : //若是中间有移动也清除定时器 clearTimeout(timerId) break; case "touchend" : //若是在500ms以内抬起了手指,则须要定时器 clearTimeout(timerId); break; } } }, /** * 左侧滑动。 * 记录手指按下的左边,在离开的时候计算 deltaX是否知足左滑的条件 */ slideLeft: function (handler){ this.ele.addEventListener("touchstart", touchFn); this.ele.addEventListener("touchend", touchFn); var startX, startY, endX, endY; function touchFn(e){ e.preventDefault(); var firstTouch = e.changedTouches[0]; switch (e.type){ case "touchstart": startX = firstTouch.pageX; startY = firstTouch.pageY; break; case "touchend": endX = firstTouch.pageX; endY = firstTouch.pageY; //x方向移动大于y方向的移动,而且x方向的移动大于25个像素,表示在向左侧滑动 if (Math.abs(endX - startX) >= Math.abs(endY - startY) && startX - endX >= 25){ handler.call(this, e); } break; } } }, /* 右侧滑动 */ rightLeft: function (e){ //TODO: } } window.$ = window.myQuery = myQuery; })(window); // ======================== // 使用: $("div").tap(function (e){ console.log("单击事件") }) $("div").longTag(function (){ console.log("长按事件"); }) $("div").slideLeft(function (e){ console.log(this); this.innerHTML = "左侧滑动了....." })
手势相关的事件通常就是tap类(触屏)和滑动(swipe)事件两类。都是基于原生的touchstart、touchmove、touchend事件,封装成不一样的手势类型自定义事件。
触碰事件,我目前还不知道它和touch的区别,通常用于代替click事件,有tap longTap singleTap doubleTap四种之分。
滑动事件,有swipe swipeLeft swipeRight swipeUp swipeDown 五种之分。
Zepto.js 是一个轻量级的针对现代高级浏览器的JavaScript库, 它适配了jQuery的大部分api,也就是jQuery怎么用,Zepto.js就怎么用。它很是小,很是适合移动端。
Zepto.js的touch模块中封装了手势相关的代码。封装了再触摸设备上触发tap– 和 swipe– 相关事件,也适用于全部的touch
(iOS, Android)和pointer
事件(Windows Phone)。
<style>.delete { display: none; }</style> <ul id=items> <li>List item 1 <span class=delete>DELETE</span></li> <li>List item 2 <span class=delete>DELETE</span></li> </ul> <script> $('#items li').swipe(function(){ $('.delete').hide() $('.delete', this).show() }) $('.delete').tap(function(){ $(this).parent('li').remove() }) </script>
hammer.js
hammer提供了不只仅tap、swipe等事件,还提供了:pan(平移)、pinch类(捏拿缩放)、 press类(按住)、 rotate类(旋转)类手势支持, hammer.js详解教程
若是某个返回按钮的位置,刚好在要返回的这个页面的带有href属性的a标签的范围内,在点击返回按钮后,页面快速切换到有a标签的页面,300ms后触发了click事件,从而触发了a标签的意外跳转,这个就是典型的点击穿透问题。罪魁祸首其实就是a标签跳转默认是click事件触发,而移动端的touch事件触发以后,依然会在300ms后触发click事件。
解决办法:
1.就是阻止触发touch事件完成后的click事件。
2.不要混用touch和click事件。显然不可能都绑定click事件,由于要解决300ms延迟问题(除了fastclick),那么只能都绑定touch事件,这样click事件永远不会被触发。
注意:zepto并无阻止click事件,因此使用zepto的tap事件依然会致使点击穿透问题,你须要手动添加 e.preventDefault() 来阻止click事件。
参考文章: