如何使用JavaScript檢測空閒的瀏覽器選項卡
在某些情況下,當用戶與我們的最終產品或應用程式進行互動時,我們發現自己會執行許多密集的,佔用大量CPU的任務。啟動輪詢器,建立WebSocket連線,甚至載入視訊或圖片等媒體,都有可能成為效能障礙,尤其是當這些任務在不需要的情況下消耗資源的時候。
在使用者沒有主動與介面互動的同時,從不必要的工作負載或網路請求中釋放主執行緒是一個非常好的和有意義的實踐。換一種方式,在大多數主機提供商都在引入基於配額的定價模式的行業中,減少網路請求也可以降低執行應用程式或服務的成本。
頁面可見性(Page Visibility) API
所有現代的網頁瀏覽器都加入了頁面可見性API,它允許我們檢測瀏覽器的標籤頁何時被隱藏,此外,我們還可以註冊一個事件監聽器,以檢測可見性變化時的訊號。
document.visibilityState
當頁面處於前臺時,document.visibilityState 可能是 visible ,最小化視窗的“標籤”或隱藏。
我們可以通過以下方式直接訪問 document.visibilityState:
console.log(document.visibilityState); // => 它可以是“visible”或“hidden”
visibilitychange Event
我們還可以使用事件偵聽器輕鬆檢測可見性屬性中的更改。
const onVisibilityChange = () => { if (document.visibilityState === 'hidden') { console.log('> 這個視窗是隱藏的.'); } else { console.log('> 這個視窗是可見的.'); } }; document.addEventListener('visibilitychange',onVisibilityChange,false);
輪詢示例
考慮一種情況,在這種情況下,我們正在輪詢API以獲取更新,並且希望避免對空閒使用者進行不必要的呼叫。一個簡化的示例如下所示:
const poll = () => { const interval = 1500; let _poller = null; const repeat = () => { console.log(`~ Polling: ${Date.now()}.`); }; return { start: () => { _poller = setInterval(repeat,interval); },stop: () => { console.log('~ Poller stopped.'); clearInterval(_poller); } }; }; const poller = poll(); poller.start(); const onVisibilityChange = () => { if (document.visibilityState === 'hidden') { poller.stop(); } else { poller.start(); } }; document.addEventListener('visibilitychange',false);
在後臺非同步載入
但有時我們可以通過反其道而行之,加速使用者的終端體驗。我們可以非同步載入外部依賴或資產,而不是取消所有的作業和請求。這樣,當用戶回來時,他們的最終體驗將更加“充實”並且豐富。
/ Webpack /
使用ES2015動態匯入建議和適當的Webpack配置清單,我們可以輕鬆地在後臺載入額外的模組或資產。
let loaded = false; const onVisibilityChange = () => { if (document.visibilityState === 'hidden') { // Aggresively preload external assets ans scripts if (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',false);
/ Rollup /
Rollup還支援開箱即用的動態匯入。
let loaded = false; const onVisibilityChange = () => { if (document.visibilityState === 'hidden') { // Aggresively preload external assets ans scripts if (loaded) { return; } Promise.all([ import('./modules.js').then(({default: DefaultExport,NamedExport}) => { // do something with modules. }) ]).then(() => { loaded = true; }); } }; document.addEventListener('visibilitychange',false);
/ 用Javascript預載入 /
除了使用捆綁器,我們還可以僅使用幾行JavaScript來預載入靜態資源(例如影象)。
let loaded = false; const preloadImgs = (...imgs) => { const images = []; imgs.map( url => new Promise((resolve,reject) => { images[i] = new Image(); images[i].src = url; img.onload = () => resolve(); img.onerror = () => reject(); }) ); }; const onVisibilityChange = () => { if (document.visibilityState === 'hidden') { // Aggresively preload external assets ans scripts if (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',false);
微互動
最後,一種吸引使用者注意力的巧妙方法是動態更改圖示,只需使用幾個畫素就可以保持互動。
const onVisibilityChange = () => { const favicon = document.querySelector('[rel="shortcut icon"]'); if (document.visibilityState === 'hidden') { favicon.href = '/come-back.png'; } else { favicon.href = '/example.png'; } }; document.addEventListener('visibilitychange',false);
總結
到此這篇關於如何使用JavaScript檢測空閒的瀏覽器選項卡的文章就介紹到這了,更多相關js瀏覽器選項卡內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!