最推荐的方式是监听wheel事件。它提供deltaY、deltaX和deltaMode属性,能精确获取滚动方向与幅度,通过preventDefault()阻止默认行为并结合{passive: false}实现自定义滚动,现代浏览器支持良好,优于旧的mousewheel和DOMMouseScroll事件。

在JavaScript里处理鼠标滚轮事件,最直接也最推荐的方式是监听DOM元素的
wheel
要处理鼠标滚轮事件,你通常会给目标元素(可以是
window
document
wheel
WheelEvent
deltaY
deltaX
deltaMode
deltaX
deltaY
deltaZ
DOM_DELTA_PIXEL
DOM_DELTA_LINE
DOM_DELTA_PAGE
多数情况下,我们主要关注
deltaY
deltaX
deltaMode
DOM_DELTA_PIXEL
一个基本的处理流程是这样的:
const targetElement = document.getElementById('myScrollableDiv') || window;
targetElement.addEventListener('wheel', function(event) {
// 阻止默认的滚动行为,比如浏览器本身的页面滚动
// 如果不阻止,你的自定义滚动会和浏览器原生滚动同时发生
event.preventDefault();
const scrollAmount = event.deltaY; // 获取垂直滚动量
console.log('滚轮滚动了:', scrollAmount, '像素');
if (scrollAmount > 0) {
console.log('向下滚动');
// 可以在这里实现向下滚动的自定义逻辑
// 例如:targetElement.scrollTop += 50;
} else {
console.log('向上滚动');
// 可以在这里实现向上滚动的自定义逻辑
// 例如:targetElement.scrollTop -= 50;
}
// 对于水平滚动,可以检查 event.deltaX
if (event.deltaX !== 0) {
console.log('水平滚动了:', event.deltaX, '像素');
// 实现水平滚动的自定义逻辑
}
}, { passive: false }); // { passive: false } 允许你在事件处理函数中调用 preventDefault()这里值得一提的是
passive
true
preventDefault()
false
false
preventDefault()
当然,了解一下历史总不是坏事,尤其是在需要支持一些非常老的浏览器环境时。不过,对于现代Web开发而言,
wheel
在
wheel
mousewheel
event.wheelDelta
event.wheelDeltaX
event.wheelDeltaY
DOMMouseScroll
event.detail
wheelDelta
这两种老旧事件在处理滚动方向和量化单位上都有所不同,导致跨浏览器兼容性代码写起来比较麻烦。比如,你需要判断是哪个事件被触发,然后根据不同的属性和方向约定来计算实际的滚动。
现在,如果不是有特别的历史项目包袱,我个人倾向于直接使用
wheel
直接使用
event.deltaY
event.deltaX
scrollTop
scrollLeft
核心思路是:当滚轮事件触发时,我们不是立即跳到最终位置,而是计算一个目标位置,然后通过一个定时器(或更推荐的
requestAnimationFrame
一个简单的平滑滚动实现可能涉及以下步骤:
event.deltaY
currentScrollTop
deltaY
currentScrollTop + 50
requestAnimationFrame
requestAnimationFrame
let isScrolling = false;
let currentScrollTop = 0; // 假设从0开始或者获取当前元素的scrollTop
function smoothScroll(targetScrollTop) {
if (isScrolling) return; // 避免重复触发动画
isScrolling = true;
const startTime = performance.now();
const duration = 300; // 动画持续时间,单位毫秒
const startScrollTop = currentScrollTop;
function animateScroll(currentTime) {
const elapsedTime = currentTime - startTime;
const progress = Math.min(elapsedTime / duration, 1); // 动画进度0到1
// 使用缓动函数(例如,ease-out quard)让动画更自然
const easedProgress = progress < 0.5 ? 2 * progress * progress : 1 - Math.pow(-2 * progress + 2, 2) / 2;
currentScrollTop = startScrollTop + (targetScrollTop - startScrollTop) * easedProgress;
// 应用到元素上,这里假设是body或html元素
document.documentElement.scrollTop = currentScrollTop;
document.body.scrollTop = currentScrollTop; // 兼容性考虑
if (progress < 1) {
requestAnimationFrame(animateScroll);
} else {
isScrolling = false;
}
}
requestAnimationFrame(animateScroll);
}
// 假设我们监听的是window的滚轮事件
window.addEventListener('wheel', function(event) {
event.preventDefault(); // 阻止默认滚动
currentScrollTop = document.documentElement.scrollTop || document.body.scrollTop;
const delta = event.deltaY;
const scrollStep = 100; // 每次滚轮事件移动的距离
let target = currentScrollTop + (delta > 0 ? scrollStep : -scrollStep);
// 确保目标位置不会超出文档范围
const maxScrollTop = document.documentElement.scrollHeight - window.innerHeight;
target = Math.max(0, Math.min(target, maxScrollTop));
smoothScroll(target);
}, { passive: false });这段代码只是一个简化示例,实际应用中你可能需要考虑更复杂的边界情况、多元素滚动、以及更丰富的缓动函数。但核心思想就是通过
requestAnimationFrame
在实际项目中处理滚轮事件,除了基本的监听和阻止默认行为,还有一些陷阱需要规避,以及一些优化策略可以提升用户体验和性能。
性能陷阱:频繁触发与事件节流/防抖 鼠标滚轮事件触发非常频繁,尤其是在用户快速滚动时。如果不加以限制,每次事件都执行复杂的计算或DOM操作,很容易导致页面卡顿。
isScrolling
// 简单的节流函数
function throttle(func, limit) {
let inThrottle;
return function() {
const args = arguments;
const context = this;
if (!inThrottle) {
func.apply(context, args);
inThrottle = true;
setTimeout(() => inThrottle = false, limit);
}
};
}
// 将滚轮事件处理函数包装起来
// window.addEventListener('wheel', throttle(myWheelHandler, 100), { passive: false });在现代前端框架或库中,通常会提供更完善的节流/防抖工具函数。
passive
{ passive: false }event.preventDefault()
passive
true
window.addEventListener('wheel', myHandler, { passive: true });passive: true
焦点与可访问性 (Accessibility) 当你接管了默认的滚轮行为,特别是实现自定义滚动条或页面滚动时,很容易破坏原生的键盘导航和屏幕阅读器功能。
div
overflow: auto
多设备与触控板行为差异 不同鼠标、不同操作系统以及触控板的滚动行为可能存在微妙差异。例如,有些触控板的滚动会发送非常小的
deltaY
deltaY
与其他事件或库的冲突 页面上可能有其他脚本或第三方库也在监听滚轮事件。这可能导致行为冲突或意想不到的副作用。
window
document
处理鼠标滚轮事件,表面看很简单,但要做到性能优越、用户体验良好且兼容性好,还需要一些细致的考量和实践。记住,在自定义行为之前,先问问自己:原生行为是否已经足够好?如果不是,再考虑介入并进行优化。
以上就是js 怎样处理鼠标滚轮事件的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号