js判断浏览器窗口的关闭与刷新

今日项目遇到一个问题,有一个功能会在浏览器的主窗口中新开一个窗口,然后业务要求:关闭新窗口的时候往后端发个请求,刷新的时候不发送。知道有个 onbeforeunload 事件是用于捕获关闭浏览器事件(包括刷新)的,但刷新也会走此方法,所以行不通,于是就网上找了找资料,网上回答的最多的大致是这样的

1
2
3
4
5
6
7
8
9
10
11
12
13
14
window.onbeforeunload = function () {
//鼠标相对于用户屏幕的水平位置 - 窗口左上角相对于屏幕左上角的水平位置 = 鼠标在当前窗口上的水平位置
var n = window.event.screenX - window.screenLeft;
//鼠标在当前窗口内时,n<m,b为false;鼠标在当前窗口外时,n>m,b为true。20这个值是指关闭按钮的宽度
var b = n > document.documentElement.scrollWidth - 20;
//鼠标在客户区内时,window.event.clientY>0;鼠标在客户区外时,window.event.clientY<0
if ((b && window.event.clientY < 0) || window.event.altKey || window.event.ctrlKey) {
//关闭浏览器时你想做的事
alert("关闭");
} else if (event.clientY > document.body.clientHeight || event.altKey) {
//刷新浏览器时你想做的事
alert("刷新");
}
};

但是,这个方法是监听浏览器右上角的关闭事件的,我想要的是选项卡的关闭与刷新事件,下面我们先来分析一下关闭窗口相关的几个方法

页面关闭时先执行 onbeforeunload,然后 onunload
页面刷新时先执行 onbeforeunload,然后 onunload,最后 onload。

从上面的分析中,发现关闭与刷新都会走 onbeforeunload 与 onunload,如果我们认为用这两个方法无法区分关闭与刷新事件,那就大错特错了,正因为关闭与刷新事件都会走 onbeforeunload 与 onunload,所以我们利用了一个关键点就能区分出这两种,那就是:时间差

最终解决方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
var beginTime = 0;//执行onbeforeunload的开始时间
var differTime = 0;//时间差
window.onunload = function (){
differTime = new Date().getTime() - beginTime;
if(differTime <= 5) {
console.log("浏览器关闭")
$.ajax({
type: "POST",
url:url,
dataType: "JSON",
cache: false,
success: function(msg){
console.log(msg);
},
error:function(err){
console.log(err)
}
})
}else{
console.log("浏览器刷新")
}

}
window.onbeforeunload = function (){
beginTime = new Date().getTime();
};

分析:虽然刷新与关闭都会走 onbeforeunload 与 onunload,但可能因为刷新在加载新页面前内部机制还需要做一些准备工作,所以刷新事件在执行到 onunload 事件时,用的时间会比关闭事件时间长,在我的测试页面中,页面里面没啥内容,关闭时 onbeforeunload 与 onunload 的时间差一般会在 3 毫秒内,而刷新事件的时间差一般会在 10 毫秒以上,当我真正用到项目中的时候(真正的项目页面中就有很多内容了),刷新事件的时间差竟能达到 100 毫秒左右,而关闭事件时间差还是 3 毫秒左右,这就大大保证了此方法的准确率,所以,判断浏览器窗口或者说是选项卡的关闭与刷新,此方法是比较合适的。
————————————————
版权声明:本文为 CSDN 博主「微行」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/itlsq/article/details/81095323

文章作者: wenmu
文章链接: http://blog.wangpengpeng.site/2020/06/19/js%E5%88%A4%E6%96%AD%E6%B5%8F%E8%A7%88%E5%99%A8%E7%AA%97%E5%8F%A3%E7%9A%84%E5%85%B3%E9%97%AD%E4%B8%8E%E5%88%B7%E6%96%B0/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 温木的博客
微信打赏