移动端富文本markdown中表格滚动与页面滚动的冲突处理:Touch 事件 + 鼠标滚轮精确控制方案

背景

业务需求:富文本markdown表格区域需要支持双向滚动

  • 水平滚动:查看表格更多列

  • 垂直滚动:滚动整个页面内容

技术挑战:在分享弹窗环境中,表格的垂直滚动被"劫持",无法驱动页面滚动

解决方案:Touch 事件 + 鼠标滚轮精确控制

核心思路

通过 JavaScript 精确控制触摸事件和鼠标滚轮事件,区分滑动/滚动方向并分别处理:

  • 垂直滑动/滚动:阻止表格默认行为,手动控制页面/弹窗滚动

  • 水平滑动/滚动:让表格自然处理,实现列的水平滚动

支持的输入方式

  • 移动端:触摸滑动手势

  • 桌面端:鼠标滚轮滚动

实现步骤

1. 环境检测

首先检测当前是否在指定环境中,因为不同环境的滚动目标不同:

2. 事件监听

监听触摸事件和鼠标滚轮事件:

复制代码
// 触摸开始:记录初始位置
// 触摸移动:方向判断和滚动控制
// 触摸结束:重置状态
// 鼠标滚轮事件:直接处理滚动
3. 滑动方向判断

这是方案的核心,通过计算滑动距离和设置合理阈值来准确判断用户意图:

4. 垂直滚动处理

触摸滑动处理:根据环境选择不同的滚动目标:

鼠标滚轮处理:相对简单,直接根据滚轮方向处理:

5. CSS 配置
复制代码
.test {
  overflow-x: auto; // 允许水平滚动
  overflow-y: hidden; // 禁用垂直滚动,完全由 JS 控制
  overscroll-behavior-x: contain; // 水平滚动限制在表格内
  overscroll-behavior-y: none; // 垂直滚动不产生边界效果
  -webkit-overflow-scrolling: touch; // iOS 滚动优化
}
6. 事件监听器注册

在组件生命周期中正确注册和清理事件监听器:

复制代码
// 组件挂载时注册
​
// 组件卸载时清理

关键技术点

1. 触摸事件的阈值设置

复制代码
const SWIPE_THRESHOLD = 10 // 太小容易误触,太大响应迟钝
const DIRECTION_THRESHOLD = 30 // 确保方向判断的准确性

2. 滚动敏感度调整

复制代码
// 触摸滑动:需要调整敏感度
const scrollSpeed = -deltaY * 0.8 // 0.8 是经过测试的最佳系数

// 鼠标滚轮:直接使用原始值
shareOverview.scrollTop += event.deltaY

3. 连续滚动支持

复制代码
// 触摸滑动:更新起始位置
touchStartY = touch.clientY // 实现流畅的连续滚动

// 鼠标滚轮:天然支持连续滚动,无需特殊处理

4. 事件监听器配置

复制代码
{
  passive: false
} // touchmove 和 wheel 需要能够 preventDefault
{
  passive: true
} // touchstart/touchend 可以被动监听,提升性能

5. 输入方式差异处理

|------|-----------|--------------------|
| 方向判断 | 需要计算阈值 | 直接通过 deltaX/deltaY |
| 滚动距离 | 需要敏感度调整 | 直接使用 deltaY |
| 连续操作 | 需要更新起始位置 | 天然支持 |
| 性能考虑 | 高频触发,需要优化 | 相对低频 |

性能优化

  1. 最小化 DOM 查询:缓存常用的 DOM 元素引用

  2. 合理使用 preventDefault:只在必要时阻止默认行为

  3. 事件委托:在容器级别监听事件,减少监听器数量

  4. 节流处理:对于高频触发的滚动事件可以考虑节流

兼容性考虑

  • Touch Events:现代移动端浏览器支持良好

  • Wheel Events:现代桌面端浏览器支持良好

  • preventDefault :需要 passive: false 才能生效

  • scrollBy/scrollTop:标准 API,兼容性好

  • querySelector:现代浏览器标准支持

实际效果

实现后的效果:

移动端触摸

  • ✅ 分享弹窗中:表格区域上下滑动 → 弹窗页面滚动

  • ✅ 单独使用时:表格区域上下滑动 → 页面滚动

  • ✅ 两种环境:表格区域左右滑动 → 表格水平滚动

桌面端鼠标

  • ✅ 分享弹窗中:表格区域垂直滚轮 → 弹窗页面滚动

  • ✅ 单独使用时:表格区域垂直滚轮 → 页面滚动

  • ✅ 两种环境:表格区域水平滚轮 → 表格水平滚动

整体体验

  • ✅ 滚动体验流畅自然,无卡顿感

  • ✅ 支持多种输入方式,适配不同设备

总结

Touch 事件 + 鼠标滚轮精确控制方案虽然代码相对复杂,但在复杂滚动场景中提供了:

  1. 完全的可控性:可以精确控制每种滚动行为

  2. 全平台支持:同时支持移动端和桌面端

  3. 良好的兼容性:基于标准的 Touch Events 和 Wheel Events API

  4. 灵活的扩展性:可以轻松适配不同的业务场景

在处理复杂滚动冲突问题时,JavaScript 精确控制往往比纯 CSS 方案更可靠,特别是需要同时支持多种输入方式的场景。

相关推荐
浪裡遊1 分钟前
React开发模式解析:JSX语法与生命周期管理
前端·javascript·react.js·前端框架·ecmascript
fruge4 分钟前
Vue 3 完全指南:响应式原理、组合式 API 与实战优化
javascript·vue.js·ecmascript
用户877244753966 分钟前
Lubanno7UniverSheet:开放底层能力,让你的表格需求 “不设限”
前端
张可爱11 分钟前
ES6奶茶铺版通俗笔记 🍵✨
前端
用户8772447539612 分钟前
Lubanno7UniverSheet:选择命令式,为了真正的跨框架通用
前端
Aoda18 分钟前
从痛点到落地:PawHaven 的 Monorepo 架构设计
前端·javascript
幸运小圣19 分钟前
Set数据结构【ES6】
javascript·数据结构·es6
望获linux21 分钟前
【实时Linux实战系列】使用 u-trace 或 a-trace 进行用户态应用剖析
java·linux·前端·网络·数据库·elasticsearch·操作系统
zxg_神说要有光28 分钟前
我好像找到了最适合我的生活状态
前端·javascript
飞哥数智坊32 分钟前
今天,我的个人网站正式上线了!
前端