如何使用JavaScript檢測空閑的瀏覽器選項(xiàng)卡
在某些情況下,當(dāng)用戶與我們的最終產(chǎn)品或應(yīng)用程序進(jìn)行交互時(shí),我們發(fā)現(xiàn)自己會(huì)執(zhí)行許多密集的,占用大量CPU的任務(wù)。啟動(dòng)輪詢器,建立WebSocket連接,甚至加載視頻或圖片等媒體,都有可能成為性能障礙,尤其是當(dāng)這些任務(wù)在不需要的情況下消耗資源的時(shí)候。
在用戶沒有主動(dòng)與界面交互的同時(shí),從不必要的工作負(fù)載或網(wǎng)絡(luò)請求中釋放主線程是一個(gè)非常好的和有意義的實(shí)踐。換一種方式,在大多數(shù)主機(jī)提供商都在引入基于配額的定價(jià)模式的行業(yè)中,減少網(wǎng)絡(luò)請求也可以降低運(yùn)行應(yīng)用程序或服務(wù)的成本。

頁面可見性(Page Visibility) API
所有現(xiàn)代的網(wǎng)頁瀏覽器都加入了頁面可見性API,它允許我們檢測瀏覽器的標(biāo)簽頁何時(shí)被隱藏,此外,我們還可以注冊一個(gè)事件監(jiān)聽器,以檢測可見性變化時(shí)的信號。
document.visibilityState
當(dāng)頁面處于前臺時(shí),document.visibilityState 可能是 visible ,最小化窗口的“標(biāo)簽”或隱藏。
我們可以通過以下方式直接訪問 document.visibilityState:
console.log(document.visibilityState); // => 它可以是“visible”或“hidden”
visibilitychange Event
我們還可以使用事件偵聽器輕松檢測可見性屬性中的更改。
const onVisibilityChange = () => {
if (document.visibilityState === 'hidden') {
console.log('> 這個(gè)窗口是隱藏的.');
} else {
console.log('> 這個(gè)窗口是可見的.');
}
};
document.addEventListener('visibilitychange', onVisibilityChange, false);
輪詢示例
考慮一種情況,在這種情況下,我們正在輪詢API以獲取更新,并且希望避免對空閑用戶進(jìn)行不必要的調(diào)用。一個(gè)簡化的示例如下所示:
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', onVisibilityChange, false);
在后臺異步加載
但有時(shí)我們可以通過反其道而行之,加速用戶的終端體驗(yàn)。我們可以異步加載外部依賴或資產(chǎn),而不是取消所有的作業(yè)和請求。這樣,當(dāng)用戶回來時(shí),他們的最終體驗(yàn)將更加“充實(shí)”并且豐富。
/ Webpack /
使用ES2015動(dòng)態(tài)導(dǎo)入建議和適當(dāng)?shù)腤ebpack配置清單,我們可以輕松地在后臺加載額外的模塊或資產(chǎn)。
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', onVisibilityChange, false);
/ Rollup /
Rollup還支持開箱即用的動(dòng)態(tài)導(dǎo)入。
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', onVisibilityChange, false);
/ 用Javascript預(yù)加載 /
除了使用捆綁器,我們還可以僅使用幾行JavaScript來預(yù)加載靜態(tài)資源(例如圖像)。
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', onVisibilityChange, false);
微互動(dòng)
最后,一種吸引用戶注意力的巧妙方法是動(dòng)態(tài)更改圖標(biāo),只需使用幾個(gè)像素就可以保持交互。
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', onVisibilityChange, false);
總結(jié)
到此這篇關(guān)于如何使用JavaScript檢測空閑的瀏覽器選項(xiàng)卡的文章就介紹到這了,更多相關(guān)js瀏覽器選項(xiàng)卡內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
npm install報(bào)錯(cuò)無法創(chuàng)建packge.json文件的解決辦法
當(dāng)你在運(yùn)行 npm install 時(shí)遇到錯(cuò)誤,提示無法找到 package.json 文件,也沒有創(chuàng)建一個(gè) package.json 文件,只創(chuàng)建了一個(gè)package-lock.json文件,本文給大家介紹詳細(xì)的解決辦法,需要的朋友可以參考下2024-02-02
js 動(dòng)態(tài)加載事件的幾種方法總結(jié)
本篇文章主要是對js 動(dòng)態(tài)加載事件的幾種方法進(jìn)行了詳細(xì)的總結(jié)介紹,需要的朋友可以過來參考下,希望對大家有所幫助2013-12-12
JS實(shí)現(xiàn)的數(shù)組全排列輸出算法
這篇文章主要介紹了JS實(shí)現(xiàn)的數(shù)組全排列輸出算法,實(shí)例分析了全排列的原理與相關(guān)的javascript實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03
微信小程序?qū)崿F(xiàn)手指拖動(dòng)選項(xiàng)排序
這篇文章主要介紹了微信小程序?qū)崿F(xiàn)手指拖動(dòng)選項(xiàng)排序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04

