滚动事件侦听器 javascript

IT技术 javascript
2021-03-04 01:19:02

当用户在可以使用的某个文本框中滚动时,是否有 js 侦听器?除了滚动,有点像 onclick。我看到了用于数字输入滚动的 HTML5 事件侦听器 - 仅适用于 Chrome,但这似乎仅适用于 chrome。我正在寻找跨浏览器的东西。

5个回答

对于那些发现这个问题希望找到不涉及 jQuery 的答案的人,您可以window使用正常的事件侦听挂钩到“滚动”事件。假设我们要添加滚动监听一些 CSS-selector-able 元素:

// what should we do when scrolling occurs
function runOnScroll(element) {
  // not the most exciting thing, but a thing nonetheless
  console.log(element);
};

// grab elements as array, rather than as NodeList
const elements = document.querySelectorAll(`...`);

// and then make each element do something on scroll
elements.forEach(element => {
  window.addEventListener(
    "scroll",
    () => runOnScroll(element),
    { passive: true }
  );
});

或者,将单个滚动侦听器与evt => runOnScroll(evt)as 处理程序绑定,然后找出如何处理函数elements内部的所有内容runOnScroll

请注意,我们使用passive属性告诉浏览器此事件不会干扰滚动本身。如果我们想要平滑的滚动行为,这很重要(也意味着我们不应该在滚动期间执行触发回流的 DOM 更新)。

对于奖励积分,您可以为滚动处理程序提供一个锁定机制,以便在我们已经在滚动时它不会运行:

// global lock, so put this code in a closure of some sort so you're not polluting.
let locked = false;
let lastCall = false;

function runOnScroll(element) {
  if(locked) return;

  if (lastCall) clearTimeout(lastCall);
  lastCall = setTimeout(() => {
    runOnScroll(element);
    // you do this because you want to handle the last
    // scroll event, even if it occurred while another
    // event was being processed.
  }, 200);

  // ...your code goes here...

  locked = false;
};
只是好奇 - 不确定它是否重要 - 但是 useCapture 应该为滚动事件侦听器返回 false 还是 true ?
2021-04-25 01:19:02
没关系 - 设置useCapture是一种确保处理程序在任何处理程序由于冒泡或“我不关心它被调用的顺序”而获得事件之前触发的方法,不设置useCapture. 然而,这并没有太大用处,因为当使用多个侦听器时没有调用顺序保证useCapture=true
2021-04-28 01:19:02
关于锁定的真/假检查:代码不会只触发一次(用户第一次滚动)??
2021-05-04 01:19:02
非常干净,但是这不会影响滚动性能,因为我们在滚动时添加了一个监听器,无论我们选择了多少元素?我认为我们希望集中 onScroll 逻辑,以便我们可以管理执行的内容。你发现了什么?
2021-05-14 01:19:02
非常如此,因此为了提高性能,您可以将其转换为标准的“事件侦听器仅设置一个标志”(其中“标志”只是滚动目标值),并具有您调用的独立调度的“处理滚动数据”功能超时(如果它没有运行,scroll事件启动它,如果它已经在运行,如果它没有看到新的滚动数据要处理,它会取消它的运行)。
2021-05-16 01:19:02

我一直在寻找使用老式 JS(没有 JQuery)解决 sticy menue 的解决方案。所以我建立了一个小测试来玩它。我认为它可以帮助那些在 js 中寻找解决方案的人。它需要改进菜单,使其更光滑。我还找到了一个很好的 JQuery 解决方案,它克隆原始 div 而不是固定位置,它更好,因为修复后不需要替换页面元素的其余部分。有谁知道如何用 JS 做到这一点?请大家评论,更正,改进。

<!DOCTYPE html>
<html>

<head>
<script>

// addEvent function by John Resig:
// http://ejohn.org/projects/flexible-javascript-events/

function addEvent( obj, type, fn ) {

    if ( obj.attachEvent ) {

        obj['e'+type+fn] = fn;
        obj[type+fn] = function(){obj['e'+type+fn]( window.event );};
        obj.attachEvent( 'on'+type, obj[type+fn] );
    } else {
        obj.addEventListener( type, fn, false );
    }
}
function getScrollY() {
    var  scrOfY = 0;
    if( typeof( window.pageYOffset ) == 'number' ) {
        //Netscape compliant
        scrOfY = window.pageYOffset;

    } else if( document.body && document.body.scrollTop )  {
        //DOM compliant
        scrOfY = document.body.scrollTop;
    } 
    return scrOfY;
}
</script>
<style>
#mydiv {
    height:100px;
    width:100%;
}
#fdiv {
    height:100px;
    width:100%;
}
</style>
</head>

<body>

<!-- HTML for example event goes here -->

<div id="fdiv" style="background-color:red;position:fix">
</div>
<div id="mydiv" style="background-color:yellow">
</div>
<div id="fdiv" style="background-color:green">
</div>

<script>

// Script for example event goes here

addEvent(window, 'scroll', function(event) {

    var x = document.getElementById("mydiv");

    var y = getScrollY();      
    if (y >= 100) {
        x.style.position = "fixed"; 
        x.style.top= "0";
    } 
});

</script>
</body>
</html>

下面的基本方法不能满足您的要求吗?

具有 div 的 HTML 代码

<div id="mydiv" onscroll='myMethod();'>


JS将有以下代码

function myMethod(){ alert(1); }
从狭义上讲,这是一个function,而不是一个method:-)
2021-05-10 01:19:02
这取决于是否myMethod是全球性的 - window.myMethod;)。
2021-05-15 01:19:02

当用户在可以使用的某个文本框中滚动时,是否有 js 侦听器?

DOM L3 UI Events规范给出了最初的定义,但被认为已经过时。

要添加单个处理程序,您可以执行以下操作:

  let isTicking;
  const debounce = (callback, evt) => {
    if (isTicking) return;
    requestAnimationFrame(() => {
      callback(evt);
      isTicking = false;
    });
    isTicking = true;
  };
  const handleScroll = evt => console.log(evt, window.scrollX, window.scrollY);
  document.defaultView.onscroll = evt => debounce(handleScroll, evt);

对于多个处理程序,或者如果出于样式原因更可取,您可以使用addEventListener而不是将您的处理程序分配给onscroll如上所示。

如果使用类似_.debouncelodash 的东西,你可能会逃脱:

const handleScroll = evt => console.log(evt, window.scrollX, window.scrollY);
document.defaultView.onscroll = evt => _.debounce(() => handleScroll(evt));

检查浏览器兼容性,并确保在完成之前在某些实际设备上进行测试。

let lastKnownScrollPosition = 0;
let ticking = false;

function doSomething(scrollPos) {
  // Do something with the scroll position
}

document.addEventListener('scroll', function(e) {
  lastKnownScrollPosition = window.scrollY;

  if (!ticking) {
    window.requestAnimationFrame(function() {
      doSomething(lastKnownScrollPosition);
      ticking = false;
    });

    ticking = true;
  }
});