「前端发动机」Touchmove 禁止默认滚动的几种方案

虚树苔藓
• 阅读 6923

前言

分享一些实际开发过程中遇到的问题和解决方案,文中如有不对之处,欢迎大家指出,共勉。!

个人博客地址 🍹🍰 fe-code

背景

源于最近的一个移动端走马灯需求,使用 touchmove 事件,来触发走马灯的动画。但是在实际运行时发现,滑动走马灯的时候很容易触发页面自身垂直方向的滚动,如下图

注:这里用 overflow: auto 模拟走马灯,只做 touchmove 的测试。

「前端发动机」Touchmove 禁止默认滚动的几种方案

可以看出,在滑动过程中,滑动方向一旦偏向垂直方向,就会触发页面的垂直滚动。

方案

Passive event listeners

因为是 touchmove 事件触发的垂直滚动,所以很容易就想到了通过 e.preventDefault() 来禁用事件的默认行为,又很容易就改了代码。

function Touch() {
    const startTouchRef = useRef({x: 0, y: 0});
    // 保存初始位置
    function onTouchStart(e) {
        startTouchRef.current = { x: e.touches[0].pageX, y: e.touches[0].pageY };
    }
    // 限制垂直方向上的滚动
    function onTouchMove(e) {
        const y = Math.abs(e.touches[0].pageY - startTouchRef.current.y);
        const x = Math.abs(e.touches[0].pageX - startTouchRef.current.x);
        // 简单判断滑动方向是倾向于 y 还是 x
        // 禁止 x 方向的默认滚动,因为 x 方向的滚动会通过 Touchmove 或者 css 动画 实现
        if (y < x) {
            e.preventDefault();
        }
    }
    return (
        <div onTouchStart={onTouchStart}
             onTouchMove={onTouchMove}>
            // ...
        </div>
    )
}

最后很容易得到了一个报错。

「前端发动机」Touchmove 禁止默认滚动的几种方案

一个人性化的报错,让我们去查看 https://www.chromestatus.com/features/5093566007214080 这个 url。

「前端发动机」Touchmove 禁止默认滚动的几种方案

大意是说:addEventListener 有一个参数 passive 默认是 false,但是在 Chrome 56 的时候 把 touchstart 和 touchmove 改成了默认 passive: true。这样,touchmove 事件就不会阻塞页面的滚动。因为在 passive: false 的状态下,不管是否需要调用 e.preventDefault() 来阻止页面滚动,都需要等到 touchmove 函数执行完毕,页面才会做出反应。

做一个简单的测试。

// 没有阻止页面滚动,仅仅是增加了事件处理的时间
function Touch() {
    const ref = useRef(null);
    function onTouchMove(e) {
        console.time();
        let index = 0;
        for (let i = 0; i< 1000000000; i++) {
            index++;
        }
        console.timeEnd();
    }
    useEffect(() => {
        ref.current.addEventListener('touchmove', onTouchMove, { passive: false });
        return () => {
            ref.current.removeEventListener('touchmove', onTouchMove, { passive: false });
        };
    }, []);
    return (
        <div >
            // ...
        </div>
    )
}

「前端发动机」Touchmove 禁止默认滚动的几种方案

每次滑动后页面的响应明显卡顿,因为浏览器需要等 touchmove 执行完才知道是否需要禁止默认滚动。而将 passive 设为 true 后,浏览器将不考虑禁用默认行为的可能性,会立即触发页面行为。

当然,如果确实要阻止默认行为,就像我之前的那个需求一样,就需要手动设置 passive 是 false,然后正常使用 preventDefault 就好。不过,不管是哪种方式,我们都需要优化自己的执行代码,尽量减少时间代码运行时间。否则,还会看到以下警告:

「前端发动机」Touchmove 禁止默认滚动的几种方案

关于被动事件监听,更多的优化是在移动端,pc 端貌似较少处理。我这里只测试了 mousewheel,在 pc 的 Chrome 74 下,尽管设置成了 passive: true,也没有优先触发页面的滚动行为。但是,在移动端模式下,是可以的。大家有兴趣的也可以自己测试一下。

因为 Chrome 56以上才支持 passive,所以在使用时可能需要做一下兼容性测试。代码来自 MDN

// 如果触发对 options 取值 passive 的情况,说明支持 passive 属性
var passiveSupported = false;

try {
  var options = Object.defineProperty({}, "passive", {
    get: function() {
      passiveSupported = true;
    }
  });

  window.addEventListener("test", null, options);
} catch(err) {}

someElement.addEventListener("mouseup", handleMouseUp, passiveSupported
                               ? { passive: true } : false);

touch-action

用于设置触摸屏用户如何操纵元素的区域(例如,浏览器内置的缩放功能)。 — MDN

这是一个 css 属性,简单来说,就是可以通过 css 指定允许用户使用的手势操作。

  • pan-x 启用单指水平平移手势
  • pan-y 启用单指垂直平移手势
  • none 禁止操作

其他属性,大家可以去 MDN 自行查阅。结合我们的需求,使用 pan-y 只开启垂直方向的操作,也能做到类似的效果。需要注意的是,设置 touch-action,和我们设置 passive: false 再调用 preventDefault 效果是一样的,不会再对允许操作方向上的滑动效果进行优化。

「前端发动机」Touchmove 禁止默认滚动的几种方案

另外,这个属性也有兼容性问题,在 Safari 上的支持效果并不好,具体查看 can i use

overflow

对于元素的禁止滚动,其实我们给他的父元素添加 overflow: hidden 也能达到想要的效果。对于整个页面来说,就需要给 html 标签添加 overflow: hidden。但是,基于当前这个需求场景,因为只是希望在水平滑动时不触发垂直方向的滚动,所以需要判断什么时候设置属性,什么时候移除属性。

这里我没有具体去做这个测试,只是提供一种思路。

交流群

qq前端交流群:960807765,欢迎各种技术交流,期待你的加入;

微信群:有需要的同学可以加我好友(q1324210213),我拉你入群。

后记

如果你看到了这里,且本文对你有一点帮助的话,希望你可以动动小手支持一下作者,感谢🍻。文中如有不对之处,也欢迎大家指出,共勉。好了,又耽误大家的时间了,感谢阅读,下次再见!

感兴趣的同学可以关注下我的公众号 前端发动机,好玩又有料。

「前端发动机」Touchmove 禁止默认滚动的几种方案

点赞
收藏
评论区
推荐文章
待兔 待兔
11个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
测试环境治理之MYSQL索引优化篇
测试环境这个话题对于开发和测试同学一定不陌生,大家几乎每天都会接触。但是说到对测试环境的印象,却鲜有好评,下面就主要给大家分享一次实际工作中的Mysql性能优化实践,与大家共勉~
Alex65 Alex65
4年前
一文带你认知MySQL异常
摘要:在本文中,总结了开发过程中最为常见的几种MySQL抛出的异常以及如何解决,包括高版本驱动的问题、时区配置问题、SSL连接问题等,是一篇经验总结贴。前言在本文中,总结了开发过程中最为常见的几种MySQL抛出的异常以及如何解决,包括高版本驱动的问题、时区配置问题、SSL连接问题等,是一篇经验总结贴,于我个人而言,这一篇足以解决目前
springboot升级过程中踩坑定位分析记录 | 京东云技术团队
此文是笔者按照分析流程进行简单验证,分析验证过程中难免有遗漏之处,如有错误遗漏还烦请各位指出共同进步。
Stella981 Stella981
3年前
JavaScript 代码是如何被 Babel 编译的
!(https://oscimg.oschina.net/oscnet/d47ca52f7cbdb3b7b5f9664f4e956e1a6cb.png)前言本文写于 2019年,如有不对之处欢迎指出。Babel对于前端开发者来说应该是很熟悉了,日常开发中基本上是离不开它的。已经9102 年了,我们已经能够熟练地使用
Stella981 Stella981
3年前
25个经典的Spring面试问答
本人收集了一些在大家在面试时被经常问及的关于Spring的主要问题,这些问题有可能在你下次面试时就会被问到。对于本文中未提及的Spring其他模块,我会单独分享面试的问题和答案。欢迎大家向我推荐你在面试过程中遇到关于Spring的问题。我会把大家推荐的问题添加到下面的Spring常用面试题清单中供大家参考。问题清单:1.什么是Spr
Wesley13 Wesley13
3年前
JavaWeb编程小技巧集合
在编程时通常会遇到一些问题,有时候是不知所措,有时候是解决方案不够优雅,本篇旨在记录编程过程中一些个人想不到的,稍微优雅一点的解决方案,方案来源均来自互联网。List使用subList实现分页获取intsubSize1000;intsubCountlist.size();
京东门详一码多端探索与实践 | 京东云技术团队
本文主要讲述京东门详业务在支撑过程中遇到的困境,面对问题我们在效率提升、质量保障等方向的探索和实践,在此将实践过程中问题解决的思路和方案与大家一起分享,也希望能给大家带来一些新的启发
京东云开发者 京东云开发者
5个月前
供应链计划性能优化解决方案-Clickhouse本地Join
作者:京东零售姜波前言本文主要针对供应链计划业务发展过程中系统产生的瓶颈问题的解决方案进行阐述,并且分享一些问题解决过程中用到的一些工具方法,希望对其他业务同类问题提供启发,原理细节不着重介绍,如有兴趣欢迎一起探讨。业务背景供应链计划业务目前数据库主要使用
美凌格栋栋酱 美凌格栋栋酱
5个月前
Oracle 分组与拼接字符串同时使用
SELECTT.,ROWNUMIDFROM(SELECTT.EMPLID,T.NAME,T.BU,T.REALDEPART,T.FORMATDATE,SUM(T.S0)S0,MAX(UPDATETIME)CREATETIME,LISTAGG(TOCHAR(
京东云开发者 京东云开发者
1个月前
供应链计划性能优化解决方案-Clickhouse本地Join
作者:京东零售姜波前言本文主要针对供应链计划业务发展过程中系统产生的瓶颈问题的解决方案进行阐述,并且分享一些问题解决过程中用到的一些工具方法,希望对其他业务同类问题提供启发,原理细节不着重介绍,如有兴趣欢迎一起探讨。业务背景供应链计划业务目前数据库主要使用