JS中touchstart事件与click事件冲突的解决方法


Posted in Javascript onMarch 12, 2018

前言

移动互联网是未来的发展趋势,现在国内很多互联网大佬都在争取移动这一块大饼,如微信及支付宝是目前比较成功的例子,当然还有各种APP和web运用。

下面这篇文章主要介绍了关于JS中touchstart事件与click事件冲突解决的相关内容,下面话不多说了,来一起看看详细的介绍吧。

一 · 业务场景的描述

在对已完成的PC站点进行移动端适配时,我们想要站点在移动设备上有更快的响应速度,以带给用户更好的体验,此时,我们应该使用移动设备专用的事件系统,例如,使用 touchstart 事件代替 click 事件。

为什么这样效果会更好呢?根据Google开发者文档中的描述:

移动设备上的浏览器将会在 click 事件触发时延迟 300ms ,以确保这是一个“单击”事件而非“双击”事件。
而对于 touchstart 事件而言,则会在用户手指触碰屏幕的一瞬间触发所绑定的事件。所以,使用 touchstart 替换 click 事件的意义在于,帮助用户在每次点击时节省 300ms 的时间。在页面频繁需要点击,或者点击发生在动画中,对动画流畅度有较高要求的情境下,使用这种技术是非常必要的。

但是,让我们回到我们的初始场景,在 PC端站点适配移动端时 我们不能简单的进行 touchstart和 click 事件的替换,因为PC并不能识别 touchstart 事件。

二 · 产生冲突的原因

当然,我们可以给某个元素同时绑定 touchstart 和 click 事件,但这将会导致本篇文章解决的问题 -- 这两个事件在移动设备上会发生冲突。

由于移动设备能够同时识别 touchstart 和 click 事件,因此当用户点击目标元素时,绑定在目标元素上的 touchstart 事件与 click 事件(约300ms后)会依次被触发,也就是说,我们所绑定的回调函数会被执行两次!。这显然不是我们想要的结果。

三 · 解决方案

针对这样的情境,有以下两种解决方案:

(一)使用 preventDefault

第一种解决方案是使用事件对象中的 preventDefault 方法,preventDefault 方法的作用在于:阻止元素默认事件行为的发生,但有意思的是,当我们在目标元素同时绑定 touchstart 和 click 事件时,在 touchstart 事件回调函数中使用该方法,可以阻止后续 click 事件的发生。

这从道理上是讲不通的,毕竟,我们添加的 click 事件并不是元素的“默认事件”,但它确实奏效了,或者说,被浏览器实现了,因此我们可以使用该方法解决移动设备上 touchstart 事件与 click 事件的冲突问题,具体代码如下:

const Button = document.getElementById("targetButton")
Button.addEventListener("touchstart", e => {
 e.preventDefault()
 console.log("touchstart event!")
})
 
Button.addEventListener("click", e => {
 e.preventDefault()
 console.log("click event!")
})

当你在浏览器上模拟移动设备后点击目标元素,只会在控制台看到 touchstart event! 字段,很显然,click 事件被成功阻止了。

总结

使用该方法的优点在于简单粗暴,直接有效,能够很好的实现我们的目标,但缺点在于, preventDefault 方法为阻止 click 事件的方式是浏览器实现上的,而不是 preventDefault 原理上的,这会带来一些不确定性,虽然我暂时尚未发现该方法失效的具体场景。

(二)基于功能检测绑定事件

我们可以通过判断浏览器是否支持 touchstart 事件来封装元素的点击事件,这样客户端会根据当前环境判定元素应该绑定的事件类型,代码如下:

const Button = document.getElementById("targetButton")
const clickEvent = (function() {
 if ('ontouchstart' in document.documentElement === true)
 return 'touchstart';
 else
 return 'click';
})();
 
Button.addEventListener(clickEvent, e => {
 console.log("things happened!")
})

总结

该方法的优点在于,我们通过增加一次判断,为元素减少了一个不必要的事件绑定,从而避免了 touchstart 与 click 事件的冲突问题。这种方法避免了我们书写两次同样的代码,并且相较于第一种方法更加符合逻辑,因此是我所推荐的。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
jQuery(1.3.2) 7行代码搞定跟随屏幕滚动的层
May 21 Javascript
基于jquery的放大镜效果
May 30 Javascript
JS实现模仿微博发布效果实例代码
Dec 16 Javascript
javascript定时器完整实例
Feb 10 Javascript
node.js [superAgent] 请求使用示例
Mar 13 Javascript
javascript实现汉字转拼音代码分享
Apr 20 Javascript
深入理解bootstrap框架之第二章整体架构
Oct 09 Javascript
利用vue.js插入dom节点的方法
Mar 15 Javascript
jQuery插件jsonview展示json数据
May 26 jQuery
解决Vue开发中对话框被遮罩层挡住的问题
Nov 26 Javascript
D3.js(v3)+react 实现带坐标与比例尺的散点图 (V3版本)
May 09 Javascript
easyUI使用分页过滤器对数据进行分页操作实例分析
Jun 01 Javascript
Node.JS循环删除非空文件夹及子目录下的所有文件
Mar 12 #Javascript
Javascript中prototype与__proto__的关系详解
Mar 11 #Javascript
js中document.write和document.writeln的区别
Mar 11 #Javascript
Javascript 编码约定(编码规范)
Mar 11 #Javascript
JavaScript获取移动设备型号的实现代码(JS获取手机型号和系统)
Mar 10 #Javascript
js经验分享 JavaScript反调试技巧
Mar 10 #Javascript
webpack打包node.js后端项目的方法
Mar 10 #Javascript
You might like
第八节--访问方式
2006/11/16 PHP
使用Curl进行抓取远程内容时url中文编码问题示例探讨
2013/10/29 PHP
zf框架的校验器InArray使用示例
2014/03/13 PHP
php中让人头疼的浮点数运算分析
2016/10/10 PHP
Yii2 hasOne(), hasMany() 实现三表关联的方法(两种)
2017/02/15 PHP
laravel框架中控制器的创建和使用方法分析
2019/11/23 PHP
jquery获得页面元素的坐标值实现思路及代码
2013/04/15 Javascript
javascript删除数组元素并且数组长度减小的简单实例
2014/02/14 Javascript
老生常谈js动态添加事件--- 事件委托
2016/07/19 Javascript
总结Javascript中数组各种去重的方法
2016/10/04 Javascript
JS实现两周内自动登录功能
2017/03/23 Javascript
javascript获取图片的top N主色值方法详解
2018/01/26 Javascript
JavaScript 截取字符串代码实例
2019/09/05 Javascript
Vue 电商后台管理项目阶段性总结(推荐)
2020/08/22 Javascript
[09:43]DOTA2每周TOP10 精彩击杀集锦vol.5
2014/06/25 DOTA
python使用ddt过程中遇到的问题及解决方案【推荐】
2018/10/29 Python
Python列表去重复项的N种方法(实例代码)
2020/05/12 Python
解决Keyerror ''acc'' KeyError: ''val_acc''问题
2020/06/18 Python
Python 使用双重循环打印图形菱形操作
2020/08/09 Python
工程师必须了解的LRU缓存淘汰算法以及python实现过程
2020/10/15 Python
使用html2canvas实现将html内容写入到canvas中生成图片
2020/01/03 HTML / CSS
html5 移动端视频video的android兼容(去除播放控件、全屏)
2020/03/26 HTML / CSS
英国最大的体育&时尚零售公司:JD Sports
2017/12/13 全球购物
英国最大的婴儿监视器网上商店:Baby Monitors Direct
2018/04/24 全球购物
西班牙在线光学:Visual-Click
2020/06/22 全球购物
生产车间实习自我鉴定
2013/09/23 职场文书
大学自我鉴定范文
2013/12/26 职场文书
活动总结怎么写啊
2014/05/07 职场文书
我爱我校演讲稿
2014/05/21 职场文书
服装发布会策划方案
2014/05/22 职场文书
2014全年工作总结
2014/11/27 职场文书
不尊敬老师的检讨书
2014/12/21 职场文书
民政工作个人总结
2015/02/28 职场文书
2016年“5.12”护士节致辞
2015/07/31 职场文书
大学生学习十八届五中全会精神心得体会
2016/01/05 职场文书
用Python简陋模拟n阶魔方
2021/04/17 Python