问题复现:移动端必现,pc 端和浏览器版本有关(公司电脑必现,家里的没有复现)。
通过 setTimeout
或 setInterval
实现的倒计时,会在页面隐藏后,实测会延缓计时或经过15s 左右会停止计时,导致计时不准确!直到页面再次激活。
可以通过判断页面可见性,计算出经过的隐藏时间来重置倒计时的时间点。
无论使用 pc 还是移动端,都会有当前页面被隐藏的情况:
document.addEventListener("visibilitychange", () => {
// 页面可见
if (document.visibilityState === "visible") {
console.log("visible");
} else {
console.log("hidden");
}
});
举例:计算隐藏时间
<body>
<div id="box">100</div>
<script>
const box = document.getElementById("box");
// 初始时间
let count = 100;
const inerval = setInterval(() => {
if (count <= 0) {
box.innerHTML = "倒计时结束";
clearInterval(inerval);
document.removeEventListener("visibilitychange", visibilitychange);
return;
}
box.innerHTML = count--;
}, 1000);
let startTime2Hidden = 0; // 页面隐藏瞬间的时间
let count2Hidden = 0; // 记录页面隐藏瞬间的 count 值
document.addEventListener("visibilitychange", visibilitychange);
function visibilitychange() {
if (document.visibilityState === "visible") {
const minus = parseInt((new Date().getTime() - startTime2Hidden) / 1000);
count = count2Hidden - minus; // 正确经过的时间
} else {
startTime2Hidden = new Date().getTime();
count2Hidden = count;
}
}
</script>
visibilitychange 的问题:在 safari 浏览器下,这个事件不总是触发,比较怪异。
谷歌实验室开源项目,兼容性很好。
使用举例:
<script src="./lifecycle.es5.js"></script>
<script>
lifecycle.addEventListener("statechange", function (event) {
console.log(event.oldState, event.newState);
if (event.oldState == "passive" && event.newState == "hidden") {
console.log("hidden");
} else if (event.oldState == "hidden" && event.newState == "passive") {
console.log("visibile");
}
});
</script>
无论使用哪种解决方案,倒计时都不是准确的,因为用户可能会修改本地时间,况且 js 计时本身就不精准。
要实现精准计时,还得靠后端接口返回正确的时间(后端也会做校验)。
以上面这个问题来说,另一种解决方案:在页面激活时再次请求一次倒计时相关的接口,前端重置倒计时时间点。
以上。