移动端触摸(touch)事件

移动端时代已经到来,作为前端开发的我们没有理由也不应该坐井观天,而是勇敢地跳出心里的那口井,去拥抱蔚蓝的天空。该来的总会来,我们要做的就是接受未知的挑战。正如你所看到的,这是一篇关于移动端触摸事件的文章,也就是我们平时在手机中用得最多的动作:touch。现在让我们开始 touch touch touch 吧!

Touch 事件

首先 touch 包含三类事件,它们分别是:touchstart、touchmove、touchend 。望文生义这种本能相信你应该会有,但在这里我还是有必需对这三个词进行一翻不必要的解释。

授课时间

touchstart:手指触摸到一个 DOM 元素时触发。

touchmove:手指在一个 DOM 元素上滑动时触发。

touchend:手指从一个 DOM 元素上移开时触发。

这三个事件又分别对应三个相同的触摸列表:

授课时间

touches:正在触摸屏幕的所有手指的一个列表。

targetTouches:正在触摸当前 DOM 元素上的手指的一个列表。

changedTouches:涉及当前事件的手指的一个列表。

Touch 属性

事件对应的三个列表虽然名字不一样,但是它们里面装的东西都是差不多的,包含了当前事件的一些相关信息,比如:一些坐标信息。

TouchList {0: Touch, length: 1}
length:1
0:Touch
    clientX:65 // 触摸点在浏览器窗口中的横坐标
    clientY:18 // 触摸点在浏览器窗口中的纵坐标
    force:1 // 触摸点压力大小
    identifier:0 // 触摸点唯一标识(ID)
    pageX:65 // 触摸点在页面中的横坐标
    pageY:18 // 触摸点在页面中的纵坐标
    radiusX:11.5 // 触摸点椭圆的水平半径
    radiusY:11.5 // 触摸点椭圆的垂直半径
    rotationAngle:0 // 旋转角度
    screenX:560 // 触摸点在屏幕中的横坐标
    screenY:175 // 触摸点在屏幕中的纵坐标
target:div#touchLog 触摸目标
__proto__:Touch
__proto__:TouchList

上面就是一个 TouchList 列表。它对应的就是前面提到的三种事件(touchstart、touchmove、touchend)中的一种,在触发时生成的一个对象列表。列表里最有用的就是 Touch 对象了,Touch 对象里存放着对应事件的一些相关的信息,我们就是通过这种个事件里这些属性的有机结合来实现各种效果。

通过上面的 radiusX,radiusY,rotationAngle 这三个属性就可以计算出你的手指触摸手机屏幕时的一个接触面,只不过这个接触面是用一个近似的椭圆来表示,也就是说它不是一个真正意义上的接触面,而是一个大概的接触面。相信心细的朋友应该会看到 TouchList 对象里有一个 length 属性,并且它的值为 1  ,这说明当前只有一个手指触发了事件(比如:touchstart 事件),换句话说,此时你只有一个手指放到了手机屏幕上,这个手指对应的一些信息存放在 Touch 对象里。因为只有一个手指放在了屏幕上,所以这个 TouchList 里只有一个 Touch 对象,并且是第一个下标为 0 。TouchList 列表里还有一个 target 属性,这个应该很好理解,就是触摸的目标。

为了让你能更加立体地理解上面的这些属性,我专门从网上找了一段话来作为补充:

来自 mozilla

1.Touch.identifier:此 Touch 对象的唯一标识符。 一次触摸动作(我们指的是手指的触摸)在平面上移动的整个过程中,该标识符不变。 可以根据它来判断跟踪的是否是同一次触摸过程,此值为只读属性。

2.Touch.screenX:触点相对于屏幕左边沿的X坐标。只读属性。

3.Touch.screenY:触点相对于屏幕上边沿的Y坐标。只读属性。

4.Touch.clientX:触点相对于可见视区(visual viewport)左边沿的X坐标。不包括任何滚动偏移。只读属性。

5.Touch.clientY:触点相对于可见视区(visual viewport)上边沿的Y坐标。不包括任何滚动偏移。只读属性。

6.Touch.pageX:触点相对于HTML文档左边沿的X坐标。当存在水平滚动的偏移时,这个值包含了水平滚动的偏移。只读属性。

7.Touch.pageY:触点相对于HTML文档上边沿的Y坐标。当存在水平滚动的偏移时,这个值包含了垂直滚动的偏移。只读属性。

8.Touch.radiusX:能够包围用户和触摸平面的接触面的最小椭圆的水平轴(X轴)半径。这个值的单位和 screenX 相同。只读属性。

9.Touch.radiusY:能够包围用户和触摸平面的接触面的最小椭圆的垂直轴(Y轴)半径。这个值的单位和 screenY 相同。只读属性。

10.Touch.rotationAngle:它是这样一个角度值:由radiusX 和 radiusY描述的正方向的椭圆,需要通过顺时针旋转这个角度值,才能最精确地覆盖住用户和触摸平面的接触面。只读属性。

11.Touch.force:手指挤压触摸平面的压力大小,从0.0(没有压力)到1.0(最大压力)的浮点数。只读属性。

12.Touch.target:当这个触点最开始被跟踪时(在 touchstart 事件中),触点位于的HTML元素。哪怕在触点移动过程中,触点的位置已经离开了这个元素的有效交互区域,或者这个元素已经被从文档中移除。需要注意的是,如果这个元素在触摸过程中被移除,这个事件仍然会指向它,但是不会再冒泡这个事件到 window 或 document 对象。因此,如果有元素在触摸过程中可能被移除,最佳实践是将触摸事件的监听器绑定到这个元素本身,防止元素被移除后,无法再从它的上一级元素上侦测到从该元素冒泡的事件。只读属性。

Touch 初探

单指操作

为了更深入地理解 Touch 事件,我们现在来做一个简单的 DEMO 。

HTML 代码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>移动端触摸(touch)事件</title>
    <meta name="viewport" content="width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no" />
    <style>
        #touchLog {
            padding: 12px;
            width: 100%;
            box-sizing: border-box;
            height: 300px;
        }
    </style>
</head>
<body>
    <div id="touchLog">这里显示 touch 信息</div>
</body>
</html>

JavaScript 代码

<script>
var obj = document.getElementById("touchLog");
obj.addEventListener("touchstart", showMsg);
function showMsg(ev) {
    console.log(ev.touches);
    console.log(ev.targetTouches);
    console.log(ev.changedTouches);
}
</script>

上面绑定的是 touchstart 事件,其它两个事件的用法一样,只是触发的时间点不一样而已。运行上面的代码会输出如下图的结果:

可能你会问:为什么它们的数据都是一样的,其实也很好理解,当你按下屏幕触发 touchstart 此时,然后执行了showMsg() 方法。那改成 touchmove 呢?不出意外的话,当你移动手指时第一组中的这三个 TouchList 也是一样的,为什么说第一组呢,因为 touchmove 是在你移动手指时才会触发的,所以当你不断移动手指时会多次触发自然而然地就会出现多组数据。但对于 touchend 就有点不一样了。你可以运行如下代码查看 touchend 与其它两个的区别:

<script>
var touchLog = document.getElementById("touchLog");
touchLog.addEventListener("touchstart", showMsgTouchstart);
function showMsgTouchstart(ev) {
    console.log("---touchstart---");
    console.log(ev.touches);
    console.log(ev.targetTouches);
    console.log(ev.changedTouches);
}

touchLog.addEventListener("touchmove", showMsgTouchmove);
function showMsgTouchmove(ev) {
    console.log("---touchmove---");
    console.log(ev.touches);
    console.log(ev.targetTouches);
    console.log(ev.changedTouches);
}

touchLog.addEventListener("touchend", showMsgTouchend);
function showMsgTouchend(ev) {
    console.log("---touchend---");
    console.log(ev.touches);
    console.log(ev.targetTouches);
    console.log(ev.changedTouches);
}
</script>

运行后你会发现 touchend 的 touches、targetTouches 里是没有 touch 对象的,length 值都为零,如下图:

这里我们还可以做别一个更有意思的测试,那就是触摸目标元素后,手指再滑出目标元素。上面三个事件对应的 touches、targetTouches 和 changedTouches 又会输出什么呢?还会是一样?

如果不出意外

1.touchstart 事件:touches、targetTouches 和 changedTouches 是一样的。

2.touchmove 事件:touches、targetTouches 和 changedTouches 是一样的(数据依然会有多组,原因上面也已经分析过了)。

3.touchend 事件:当你手指离开屏幕后,也就是 touchend 事件触发时,touches、targetTouches 的 TouchList 的 length 同样为0,也就是说没有 touch 对象。

所以使用 touchend 事件时需要注意只有 changedTouches 会存有触摸对象。但在 TouchList 对象中的 target 属性的值都为 div#touchLog,也就是说不管你触摸屏幕后手指是在目标元素上还是滑出目标元素,这个 target 属性的值还是 div#touchLog 。只要触摸了屏幕 force 的值都是 1 ,所以在这里感觉这个 force 还没有什么用武之地。

多指操作

我们先来看看下面的这一段代码,并且如果条件允许的话请用手机访问:http://yunkus.com/demo/mobile-touch-event/multi-finger-touchstart.html 查看touchstart 事件触发时的效果,代码如下:

<script>
var obj = document.getElementById("touchLog");
obj.addEventListener("touchstart", showMsg);
function showMsg(ev) {
        obj.innerHTML = "";
        var touchesStr = "";
        var targetTouchesStr = "";
        for (var i = 0; i < ev.touches.length; i++) {
            touchesStr += "identifier:" + ev.touches[i].identifier + ",x 轴坐标:" + ev.touches[i].clientX + "<br>";
            targetTouchesStr += "identifier:" + ev.targetTouches[i].identifier + ",x 轴坐标:" + ev.targetTouches[i]
                .clientX + "<br>";
        }
        obj.innerHTML = "下面是 ev.touches 数据:<br>" + touchesStr +
            "<br>下面是 ev.targetTouches 数据:<br>" + targetTouchesStr +
            "<br>下面是 ev.changedTouches 数据:<br>" +
            "identifier:" + ev.changedTouches[0].identifier + ",x 轴坐标:" + ev.changedTouches[0].clientX + "<br>";
}
</script>

访问 http://yunkus.com/demo/mobile-touch-event/multi-finger-touchmove.html 查看 touchmove 事件触发时的效果,touchmove 事件触发后 changedTouches 里的 touch 对象就不只一个了,而是跟其它两个 TouchList 列表一样。

访问 http://yunkus.com/demo/mobile-touch-event/multi-finger-touchend.html 查看 touchend 事件触发时的效果,touchmove 事件触发后就只有  changedTouches 里有 touch 且只有一个 touch 对象了,不管你同时在屏幕上放了多少根手指,这个 touch 对象对应的是你最后一次离开屏幕的那根手指。

上面的三个 demo 都会输出事件触发时的 touches 、targetTouches 和 changedTouches 里的 identifier 值,以及一个clientX 值。clientX 用于让你通过 x 的坐标来判断哪根手指对应哪个 identifier 的。对于 touchstart 事件而言,因为 changedTouches 里总是保存一个 touch 对象,所以没有遍历,而是直接通过下标访问。从上面我们可以得知有 touchstart 事件中 touches 、targetTouches 和 changedTouches 是有区别的:changedTouches 下只有一个 touch 对象,这个对象对应着触发事件最后一根发生改变(比如:最后触摸屏幕)的手指。这也就是为什么 changedTouches 里只有一个 touch 对象的原因,因为某一时刻下总是只有一个手指在变化。

但是也不能以偏盖全,因为 touchmove 事件中的 changedTouches 里就不只一个,而是跟 touches 和 targetTouches 同样有多个 touch 对象。

正如前面所说的 touchend 只有 changedTouches 列表里只有一个 touch 对象,这个对象对应着最后一根手指发生的改变(比如:最后离开屏幕)的手指。

通过研究单指操作跟多指操作,就可以让我们对 touch 事件了解得更加立体,到位。

默认事件

在移动端手指操作时会默认触发一些行为,比如:滚动,缩放。上面的例子是没有阻止触摸事件的默认行为的。所以当你测试上面 multi-finger-touchmove.html 这个例子时,你会发现有时候你会感到很无助,页面很容易发生缩放行为,甚至影响到测试效果。要想阻止触摸事件的默认行为也非常地简单只需要添加如下代码就可以了:

document.addEventListener("touchstart", function (ev) {
    ev.preventDefault();
});

添加触摸事件的阻止默认行为的好处也不仅仅只有这一个。

1.在IOS 10 下设置 meta 禁止用户缩放是没有效果的,使用ev.preventDefault(); 就可以实现禁止用户缩放页面。

2.解决 IOS 10 下溢出隐藏(不起作用)的问题。

3.禁止系统默认的滚动条(如:横向滚动条)、以及橡皮筋效果。

4.禁止长按选中文字、选中图片、系统默认菜单。

5.解决点透问题。

虽然有那么多好处,需要注意的是此时也会带来一些问题,比如:input 不能获取焦点了。不过你可以通过单独的给 input 标签添加 touchstart 事件,并且阻止其冒泡就可以让 input 标签重生了。

var inputObj = document.getElementsByTagName("input")[0];
document.addEventListener("touchstart", function (ev) {
    ev.preventDefault();
});
inputObj.addEventListener("touchstart", function (ev) {
    ev.stopPropagation();
});

这里有一个 Demo,通过 touch 的相关事件实现的一个移动端焦点图切换效果 :http://yunkus.com/demo/mobile-touch-event/

原文地址:https://www.cnblogs.com/cangqinglang/p/8794621.html

时间: 2024-10-11 13:15:28

移动端触摸(touch)事件的相关文章

web移动端触摸滑动事件

web移动端触摸事件的范例~~~ 注意:1.如果不是内联元素,获取style的属性值前需先赋值~不然为Null. 2.亲测andriod 手机 MX4内置浏览器运行妥妥的~~但是微信浏览器并不支持~原因未找到. 1 <!DOCTYPE html> 2 <html> 3 <head> 4 <meta http-equiv="Content-Type" Content="text/html; charset=utf-8;">

android 触摸touch事件的派送和传递

1.touch事件从Activity开始向下逐层派发 一直派发到焦点控件进行处理 2.如果控件的事件处理方法返回true 则,该控件消费且持续关注此事件 否则,该事件向该控件的上一层容器传递 3.控件的touch监听器的onTouch方法 先于 控件本身的onTouchEvent执行 Touch事件派发和处理的方法 dispatchTouchEvent onInterceptTouchEvent(只有ViewGroup有该拦截方法) onTouchListener.onTouch onTouch

手机触摸touch事件

1.Touch事件简介 pc上的web页面鼠 标会产生onmousedown.onmouseup.onmouseout.onmouseover.onmousemove的事件,但是在移动终端如 iphone.ipod Touch.ipad上的web页面触屏时会产生ontouchstart.ontouchmove.ontouchend.ontouchcancel 事件,分别对应了触屏开始.拖拽及完成触屏事件和取消. 当按下手指时,触发ontouchstart: 当移动手指时,触发ontouchmov

移动端触摸相关事件touch、tap、swipe

一.事件定义及分类 1. click事件 单击事件,类似于PC端的click,但在移动端中,连续click的触发有200ms ~ 300ms的延迟 2. touch类事件 触摸事件,有touchstart touchmove touchend touchcancel 四种之分 touchstart:手指触摸到屏幕会触发 touchmove:当手指在屏幕上移动时,会触发 touchend:当手指离开屏幕时,会触发 touchcancel:可由系统进行的触发,比如手指触摸屏幕的时候,突然alert了

移动端html touch事件

诸如智能手机和平板电脑一类的移动设备通常会有一(capacitive touch-sensitivescreen),以捕捉用户的手指所做的交互.随着移动网络的发展,其能够支持越来越复杂的应用,web开发者需要一种方法来处理这些事件.例如,几乎所有的快节奏游戏都需要玩家一次按下多个按钮,这种方式,在触摸屏情况下,意味着多点触摸. Apple在iOS 2.0中引入了触摸事件API,Android正迎头赶上这一事实标准,缩小差距.最近一个W3C工作组正合力制定这一触摸事件规范. 在本文深入研究iOS和

浅谈移动端之touch事件--手指的滑动事件

原理: 当开始一个touchstart事件的时候,获取此刻手指的横坐标startX和纵坐标startY: 当触发touchmove事件时,在获取此时手指的横坐标moveEndX和纵坐标moveEndY;最后,通过这两次获取的坐标差值来判断手指在手机屏幕上的滑动方向. 具体代码如下: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 $("body").on("touchstar

Html5 移动端 触摸滑动事件

以下代码经过测试  没有问题 且可以循环滑动 <!DOCTYPE html><html xmlns="http://www.w3.org/1999/xhtml"><head> <meta name="renderer" content="webkit"> <meta http-equiv="X-UA-Compatible" content="IE=edge,chr

移动端浏览器touch事件的研究总结

$("body").on("touchstart", function(e) {     e.preventDefault();     startX = e.originalEvent.changedTouches[0].pageX,     startY = e.originalEvent.changedTouches[0].pageY; }); $("body").on("touchmove", function(e)

《移动端浏览器Touch事件判断手指滑动方向方法》

1 $("body").on("touchstart", function(e) { 2     e.preventDefault(); 3     startX = e.originalEvent.changedTouches[0].pageX, 4     startY = e.originalEvent.changedTouches[0].pageY; 5 }); 6 $("body").on("touchmove",

js 处理移动端触摸事件

在处理移动端的touch事件时,我们可以选择一些插件来处理,比如jquery ui touch punch.js 提供丰富的触摸效果,可以满足移动端的开发, 但是,有些移动端开发中,并不需要如此复杂的效果,例如我们只需知道滑动的距离,向左还是向右,我们可以自己写一些代码来处理touch事件: 以下代码,只在触摸情况下支持,电脑需要chrome模拟手机: 可以改造成自己需要的效果: <!doctype html> <html> <head> <meta charse