文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

使用JavaScript检测空闲的浏览器选项卡,可以做些什么?

2024-12-24 16:50

关注

来源|https://dev.to/vorillaz/how-to-detect-idle-browser-tabs-with-javascript-88n

作者|vorillaz

在某些情况下,当用户与我们的最终产品或应用程序进行交互时,我们发现自己会执行许多密集的,占用大量CPU的任务。启动轮询器,建立WebSocket连接,甚至加载视频或图片等媒体,都有可能成为性能障碍,尤其是当这些任务在不需要的情况下消耗资源的时候。在用户没有主动与界面交互的同时,从不必要的工作负载或网络请求中释放主线程是一个非常好的和有意义的实践。换一种方式,在大多数主机提供商都在引入基于配额的定价模式的行业中,减少网络请求也可以降低运行应用程序或服务的成本。

目录

页面可见性(Page Visibility) API

所有现代的网页浏览器都加入了页面可见性API,它允许我们检测浏览器的标签页何时被隐藏,此外,我们还可以注册一个事件监听器,以检测可见性变化时的信号。

document.visibilityState

当页面处于前台时,document.visibilityState 可能是 visible ,最小化窗口的“标签”或隐藏。

我们可以通过以下方式直接访问 document.visibilityState:

  1. console.log(document.visibilityState); 
  2. // => 它可以是“visible”或“hidden” 

visibilitychange Event

我们还可以使用事件侦听器轻松检测可见性属性中的更改。

  1. const onVisibilityChange = () => { 
  2.   if (document.visibilityState === 'hidden') { 
  3.     console.log('> 这个窗口是隐藏的.'); 
  4.   } else { 
  5.     console.log('> 这个窗口是可见的.'); 
  6.   } 
  7. }; 
  8. document.addEventListener('visibilitychange', onVisibilityChange, false); 

轮询示例考虑一种情况,在这种情况下,我们正在轮询API以获取更新,并且希望避免对空闲用户进行不必要的调用。一个简化的示例如下所示:

  1. const poll = () => { 
  2.   const interval = 1500; 
  3.   let _poller = null
  4.   const repeat = () => { 
  5.     console.log(`~ Polling: ${Date.now()}.`); 
  6.   }; 
  7.  
  8.   return { 
  9.     start: () => { 
  10.       _poller = setInterval(repeat, interval); 
  11.     }, 
  12.     stop: () => { 
  13.       console.log('~ Poller stopped.'); 
  14.       clearInterval(_poller); 
  15.     } 
  16.   }; 
  17. }; 
  18.  
  19. const poller = poll(); 
  20. poller.start(); 
  21.  
  22. const onVisibilityChange = () => { 
  23.   if (document.visibilityState === 'hidden') { 
  24.     poller.stop(); 
  25.   } else { 
  26.     poller.start(); 
  27.   } 
  28. }; 
  29.  
  30. document.addEventListener('visibilitychange', onVisibilityChange, false); 

在后台异步加载

但有时我们可以通过反其道而行之,加速用户的终端体验。我们可以异步加载外部依赖或资产,而不是取消所有的作业和请求。这样,当用户回来时,他们的最终体验将更加“充实”并且丰富。

Webpack

使用ES2015动态导入建议和适当的Webpack配置清单,我们可以轻松地在后台加载额外的模块或资产。

  1. let loaded = false
  2. const onVisibilityChange = () => { 
  3.   if (document.visibilityState === 'hidden') { 
  4.     // Aggresively preload external assets ans scripts 
  5.     if (loaded) { 
  6.       return
  7.     } 
  8.     Promise.all([ 
  9.       import('./async.js'), 
  10.       import('./another-async.js'), 
  11.       import( 'modules/bar'), 
  12.       import( 'assets/images/foo.jpg'
  13.     ]).then(() => { 
  14.       loaded = true
  15.     }); 
  16.   } 
  17. }; 
  18.  
  19. document.addEventListener('visibilitychange', onVisibilityChange, false); 

Rollup

Rollup还支持开箱即用的动态导入。

  1. let loaded = false
  2. const onVisibilityChange = () => { 
  3.   if (document.visibilityState === 'hidden') { 
  4.     // Aggresively preload external assets ans scripts 
  5.     if (loaded) { 
  6.       return
  7.     } 
  8.     Promise.all([ 
  9.       import('./modules.js').then(({default: DefaultExport, NamedExport}) => { 
  10.         // do something with modules. 
  11.       }) 
  12.     ]).then(() => { 
  13.       loaded = true
  14.     }); 
  15.   } 
  16. }; 
  17.  
  18. document.addEventListener('visibilitychange', onVisibilityChange, false); 

用Javascript预加载

除了使用捆绑器,我们还可以仅使用几行JavaScript来预加载静态资源(例如图像)。

  1. let loaded = false
  2.  
  3. const preloadImgs = (...imgs) => { 
  4.   const images = []; 
  5.   imgs.map( 
  6.     url => 
  7.       new Promise((resolve, reject) => { 
  8.         images[i] = new Image(); 
  9.         images[i].src = url; 
  10.         img.onload = () => resolve(); 
  11.         img.onerror = () => reject(); 
  12.       }) 
  13.   ); 
  14. }; 
  15.  
  16. const onVisibilityChange = () => { 
  17.   if (document.visibilityState === 'hidden') { 
  18.     // Aggresively preload external assets ans scripts 
  19.     if (loaded) { 
  20.       return
  21.     } 
  22.     Promise.all
  23.       preloadImgs( 
  24.         'https://example.com/foo.jpg'
  25.         'https://example.com/qux.jpg'
  26.         'https://example.com/bar.jpg' 
  27.       ) 
  28.     ) 
  29.       .then(() => { 
  30.         loaded = true
  31.       }) 
  32.       .catch(() => { 
  33.         console.log('> snap.'); 
  34.       }); 
  35.   } 
  36. }; 
  37.  
  38. document.addEventListener('visibilitychange', onVisibilityChange, false); 

微互动

最后,一种吸引用户注意力的巧妙方法是动态更改图标,只需使用几个像素就可以保持交互。

  1. const onVisibilityChange = () => { 
  2.   const favicon = document.querySelector('[rel="shortcut icon"]'); 
  3.   if (document.visibilityState === 'hidden') { 
  4.     favicon.href = '/come-back.png'
  5.   } else { 
  6.     favicon.href = '/example.png'
  7.   } 
  8. }; 
  9.  
  10. document.addEventListener('visibilitychange', onVisibilityChange, false); 

本文转载自微信公众号「前端全栈开发者」,可以通过以下二维码关注。转载本文请联系前端全栈开发者公众号。

 

来源:前端全栈开发者内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯