来源|https://dev.to/vorillaz/how-to-detect-idle-browser-tabs-with-javascript-88n作者|vorillaz在某些情况下,当用户与我们的最终产品或应用程序进行交互时,我们发现自己在执行许多密集型、CPU密集型任务。启动轮询器、建立WebSocket连接,甚至加载视频或图片等媒体都可能成为性能障碍,尤其是当这些任务在不需要时消耗资源时。在用户未主动与界面交互时,将主线程从不必要的工作负载或网络请求中解放出来是一种非常好的且有意义的做法。换句话说,在大多数托管服务提供商引入基于配额的定价模型的行业中,减少网络请求也可以降低运行应用程序或服务的成本。目录页面可见性APIdocument.visibilityStatevisibilitychange事件轮询示例异步后台加载WebpackRollup使用Javascript预加载微交互可以注册一个事件侦听器来检测可见性何时发生变化。document.visibilityStatedocument.visibilityState可能在页面位于前台、最小化窗口的“选项卡”或隐藏时可见。我们可以通过以下方式直接访问document.visibilityState:console.log(document.visibilityState);//=>它可以是“可见”或“隐藏”的visibilitychange事件我们还可以使用事件侦听器轻松检测visibility属性的变化。constonVisibilityChange=()=>{if(document.visibilityState==='hidden'){console.log('>这个窗口是隐藏的。');}else{console.log('>这个窗口是可见的。');}};document.addEventListener('visibilitychange',onVisibilityChange,false);轮询示例考虑这样一种情况,我们正在轮询API以获取更新,并希望避免对空闲用户的必要调用造成干扰。一个简化的示例如下所示:constpoll=()=>{constinterval=1500;let_poller=null;constrepeat=()=>{console.log(`~Polling:${Date.now()}.`);};return{start:()=>{_poller=setInterval(repeat,interval);},stop:()=>{console.log('~Pollerstopped.');clearInterval(_poller);}};};constpoller=poll();poller.start();constonVisibilityChange=()=>{if(document.visibilityState==='hidden'){poller.stop();}else{poller.start();}};document.addEventListener('visibilitychange',onVisibilityChange,false);是在后台异步加载的,但是有时候我们可以反其道而行之,加快用户的终端体验。我们可以异步加载外部依赖项或资产,而不是取消所有作业和请求。这样,当用户回来时,他们的最终体验将更加“充实”和丰富。通过使用ES2015动态导入建议和适当的Webpack配置清单的Webpack,我们可以轻松地在后台加载其他模块或资产。letloaded=false;constonVisibilityChange=()=>{if(document.visibilityState==='hidden'){//积极预加载externalassetsansscriptsif(loaded){return;}Promise.all([import('./async.js'),import('./another-async.js'),import(/*webpackChunkName:"bar-module"*/'modules/bar'),import(/*webpackPrefetch:0*/'assets/images/foo.jpg')]).then(()=>{loaded=true;});}};document.addEventListener('visibilitychange',onVisibilityChange,false);RollupRollup还支持开箱即用的动态导入。letloaded=false;constonVisibilityChange=()=>{if(document.visibilityState==='hidden'){//积极预加载externalassetsansscriptsif(loaded){return;}Promise.all([import('./modules.js').then(({default:DefaultExport,NamedExport})=>{//dosomethingwithmodules.})]).then(()=>{loaded=true;});}};document.addEventListener('visibilitychange',onVisibilityChange,错误的);使用Javascript预加载除了使用捆绑器,我们还可以仅使用几行JavaScript来预加载静态资源(例如图像)。letloaded=false;constpreloadImgs=(...imgs)=>{constimages=[];imgs.map(url=>newPromise((resolve,reject)=>{images[i]=newImage();images[i].src=url;img.onload=()=>resolve();img.onerror=()=>reject();}));};constonVisibilityChange=()=>{if(document.visibilityState==='hidden'){//积极预加载externalassetsansscriptsif(loaded){return;}Promise.all(preloadImgs('https://example.com/foo.jpg','https://example.com/qux.jpg','https://example.com/bar.jpg')).then(()=>{loaded=true;}).catch(()=>{console.log('>snap.');});}};document.addEventListener('visibilitychange',onVisibilityChange,false);微交互最后,吸引用户注意力的一种巧妙方法是动态更改图标,仅使用几个像素来维持交互。constVisibilityChange=()=>{constfavicon=document.querySelector('[rel="shortcuticon"]');if(document.visibilityState==='hidden'){favicon.href='/come-back.png';}else{favicon.href='/example.png';}};document.addEventListener('visibilitychange',onVisibilityChange,false);代码关注。转载本文请联系前端全栈开发公众号。
