欢迎来到3672js教程,我们关注js教程、js框架、js代码特效等。

使用JavaScript实现检测网页是否为空闲状态,

3672Js.Com2024-04-11 02:24 来源:未知 阅读:16798 关注度4

使用JavaScript实现检测网页是否为空闲状态,


目录
  • 1. 背景
  • 2. 如何判断页面是否空闲(用户长时间无操作)
  • 3. 网页空闲检测实现
    • 3.1 简易实现
    • 3.2 处理频繁触发问题
    • 3.3 处理页面被隐藏的情况(完整实现)
  • 4. 扩展

    1. 背景

    最近开发项目时,常碰到“用户在一定时间内无任何操作时,跳转到某个页面”的需求。

    网上冲浪后,也没有找到一个比较好的js封装去解决这个问题,从而决定自己实现。

    2. 如何判断页面是否空闲(用户长时间无操作)

    首先,我们要知道什么是空闲?用户一定时间内,没有对网页进行任何操作,则当前网页为空闲状态。

    用户操作网页,无非就是通过鼠标键盘两个输入设备(暂不考虑手柄等设备)。因而我们可以监听相应的输入事件,来判断网页是否空闲(用户是否操作网页)。

    • 监听鼠标移动事件mousemove
    • 监听键盘按下事件mousedown
    • 在用户进入网页后,设置延时跳转,如果触发以上事件,则移除延时器,并重新开始。

    3. 网页空闲检测实现

    3.1 简易实现

    以下代码,简易实现了一个判断网页空闲的方法。

    const onIdleDetection = (callback, timeout = 15, immediate = false) => {
      let pageTimer;
      
      const onClearTimer = () => {
        pageTimer && clearTimeout(pageTimer);
        pageTimer = undefined;
      };
      const onStartTimer = () => {
        onClearTimer();
        pageTimer = setTimeout(() => {
          callback();
        }, timeout * 1000);
      };
    
      const startDetection = () => {
        onStartTimer();
        document.addEventListener('mousedown', onStartTimer);
        document.addEventListener('mousemove', onStartTimer);
      };
      const stopDetection = () => {
        onClearTimer();
        document.removeEventListener('mousedown', onStartTimer);
        document.removeEventListener('mousemove', onStartTimer);
      };
      const restartDetection = () => {
          onClearTimer();
          onStartTimer();
      };
    
      if (immediate) {
        startDetection();
      }
    
      return {
        startDetection,
        stopDetection,
        restartDetection
      };
    };
    

    也许你注意到了,我并没有针对onStartTimer事件进行防抖,那这是不是会对性能有影响呢?

    是的,肯定有那么点影响,那我为啥不添加防抖呢?

    这是因为添加防抖后,形成了setTimeout嵌套,嵌套setTimeout会有精度问题(参考)。

    或许你还会说,非活动标签页(网页被隐藏)的setTimeout的执行和精度会有问题(参考非活动标签的超时)。

    确实存在以上问题,接下来我们就来一一解决吧!

    3.2 处理频繁触发问题

    我们可以通过添加一个变量记录开始执行时间,当下一次执行与当前的时间间隔小于某个值时直接退出函数,从而解决这个问题(节流思想应用)。

    const onIdleDetection = (callback, timeout = 15, immediate = false) => {
      let pageTimer;
      // 记录开始时间
      let beginTime = 0;
      const onStartTimer = () => {
        // 触发间隔小于100ms时,直接返回
        const currentTime = Date.now();
        if (pageTimer && currentTime - beginTime < 100) {
          return;
        }
    
        onClearTimer();
        // 更新开始时间
        beginTime = currentTime;
        pageTimer = setTimeout(() => {
          callback();
        }, timeout * 1000);
      };
      const onClearTimer = () => {
        pageTimer && clearTimeout(pageTimer);
        pageTimer = undefined;
      };
      
      const startDetection = () => {
        onStartTimer();
        document.addEventListener('mousedown', onStartTimer);
        document.addEventListener('mousemove', onStartTimer);
      };
      const stopDetection = () => {
        onClearTimer();
        document.removeEventListener('mousedown', onStartTimer);
        document.removeEventListener('mousemove', onStartTimer);
      };
      const restartDetection = () => {
          onClearTimer();
          onStartTimer();
      };
      
      if (immediate) {
        startDetection();
      }
    
      return {
        startDetection,
        stopDetection,
        restartDetection
      };
    };
    

    3.3 处理页面被隐藏的情况(完整实现)

    我们可以监听visibilitychange事件,在页面隐藏时移除延时器,然后页面显示时继续计时,从而解决这个问题。

    /**
     * 网页空闲检测
     * @param {() => void} callback 空闲时执行,即一定时长无操作时触发
     * @param {number} [timeout=15] 时长,默认15s,单位:秒
     * @param {boolean} [immediate=false] 是否立即开始,默认 false
     * @returns
     */
    const onIdleDetection = (callback, timeout = 15, immediate = false) => {
      let pageTimer;
      let beginTime = 0;
      const onClearTimer = () => {
        pageTimer && clearTimeout(pageTimer);
        pageTimer = undefined;
      };
      const onStartTimer = () => {
        const currentTime = Date.now();
        if (pageTimer && currentTime - beginTime < 100) {
          return;
        }
    
        onClearTimer();
        beginTime = currentTime;
        pageTimer = setTimeout(() => {
          callback();
        }, timeout * 1000);
      };
    
      const onPageVisibility = () => {
         // 页面显示状态改变时,移除延时器
         onClearTimer();
    
         if (document.visibilityState === 'visible') {
           const currentTime = Date.now();
           // 页面显示时,计算时间,如果超出限制时间则直接执行回调函数
           if (currentTime - beginTime >= timeout * 1000) {
             callback();
             return;
           }
           // 继续计时
           pageTimer = setTimeout(() => {
             callback();
           }, timeout * 1000 - (currentTime - beginTime));
         }
      };
    
      const startDetection = () => {
        onStartTimer();
        document.addEventListener('mousedown', onStartTimer);
        document.addEventListener('mousemove', onStartTimer);
        document.addEventListener('visibilitychange', onPageVisibility);
      };
    
      const stopDetection = () => {
        onClearTimer();
        document.removeEventListener('mousedown', onStartTimer);
        document.removeEventListener('mousemove', onStartTimer);
        document.removeEventListener('visibilitychange', onPageVisibility);
      };
      
      const restartDetection = () => {
          onClearTimer();
          onStartTimer();
      };
    
      if (immediate) {
        startDetection();
      }
    
      return {
        startDetection,
        stopDetection,
        restartDetection
      };
    };
    

    通过以上代码,我们就完整地实现了一个网页空闲状态检测的方法。

    4. 扩展

    chrome浏览器其实提供了一个Idle DetectionAPI,来实现网页空闲状态的检测,但是这个API还是一个实验性特性,并且Firefox与Safari不支持。API参考

    到此这篇关于使用JavaScript实现检测网页是否为空闲状态的文章就介绍到这了,更多相关JavaScript检测网页内容请搜索3672js教程以前的文章或继续浏览下面的相关文章希望大家以后多多支持3672js教程!

    您可能感兴趣的文章:
    • 基于javascript代码检测访问网页的浏览器呈现引擎、平台、Windows操作系统、移动设备和游戏系统
    • js实现网页检测是否安装了 Flash Player 插件
    • Javascript检查图片大小不要让大图片撑破页面
    • JavaScript实现检查页面上的广告是否被AdBlock屏蔽了的方法
    • js检查页面上有无重复id的实现代码
    • javascript检测页面是否缩放的小例子

    本站文章为3672js教程网友分享投稿,版权归原作者,欢迎任何形式的转载,但请务必注明出处。同时文章内容如有侵犯了您的权益,请联系我们处理。
    评论已被关闭