这是我参与11月更文挑战的第28天,活动详情查看:2021最后一次更文挑战
History 对象
History 对象是浏览器模型中的一个对象,通过 window.history
属性获取,表示的是当前窗口的浏览历史。
在一个浏览器窗口中,所有访问过的页面地址,都会保存在 History 对象中。
URL 的锚点值变了,对应的也会在 History 对象中创建一条浏览记录。
无法通过js读取浏览过的具体地址
处于安全的原因,浏览器不允许js读取具体的地址,但是可以在不同地址之间导航。比如 back、forward、go 等方法。
length
和 state
属性
history.length
:当前窗口访问过的网址数量(包括当前网页)。history.state
:History 堆栈最上层的状态值,即当前状态,通常为 undefined,未设置。
history.back() 回退到上一个网页
history.back()
返回前一个浏览的页面。也就是浏览器中最常见的回退功能。
history.forward() 前进到下一个网址
history.forward()
移动到下一个网址,当前窗口显示下一个页面。前进功能。
history.go(num) 移动到基于当前位置的网址页面
history.go()
的参数为整数,基于的是当前网址的位置,比如 go(1)
相当于 forward()
;go(-1)
相当于 back()
。
go(-2)
移动到当前页面往后的第2个网址。如果指定的参数,超过是实际的范围,则不会有任何效果。
不指定参数,或指定参数 0
,会刷新当前页面。
history.pushState() 在浏览历史中添加一条记录。
window.history.pushState(state, title, url)
:
state
:一个与添加的记录相关联的状态对象,会在popstate
事件中传入回调函数。它是一个单独的对象,可以在重新载入该页面时进行使用。如果不需要,设置为null
即可。title
:新页面的标题,一般不起作用(比如 FireFox浏览器),填空字符串即可。url
:新的网址,必须与当前页面处在同一个域中。也就是,pushState 不允许添加跨域的地址。
history.pushState()
添加记录后,会在浏览器地址栏中显示,但是不会跳转到该页面,只是作为浏览历史中的一个最新记录。
即,pushState()
不会触发页面进入刷新。该方法执行后,可以通过 history.state
读出传入的状态值。
pushState
不允许添加跨域的网址。这样做是因为,防止恶意代码让用户以为他们是在另一个网站上,因为pushState()不会导致页面跳转。
history.replaceState() 修改当前的记录
history.replaceState()
的用法 和 pushState()
一样,只不过 replaceState()
用于修改当前的记录。
比如,当前的网址是 example.com/example.html
。执行如下操作后,对应地址栏中url的显示为:
history.pushState({page: 1}, 'title 1', '?page=1')
// URL 显示为 http://example.com/example.html?page=1
history.pushState({page: 2}, 'title 2', '?page=2');
// URL 显示为 http://example.com/example.html?page=2
history.replaceState({page: 3}, 'title 3', '?page=3');
// URL 显示为 http://example.com/example.html?page=3
history.back()
// URL 显示为 http://example.com/example.html?page=1
history.back()
// URL 显示为 http://example.com/example.html
history.go(2)
// URL 显示为 http://example.com/example.html?page=3
复制代码
popstate 和 hashchange 事件
popstate
事件
当同一个文档的浏览历史发生变化,并无刷新进入改变的url时,就会触发 popstate
事件。
注意,是同一个文档的浏览历史改变,如果产生文档的加载(加载不同的文档或重新加载当前文档),则不会触发该事件。也就是,
浏览历史发生变化,并且页面没有加载时,才会触发 popstate
事件。 并且,页面(重新)加载刷新时(包括第一次加载),浏览器不会触发 popstate
事件。
window.addEventListener('popstate',function(e){
console.log('location: ' + document.location);
console.log('state: ' + JSON.stringify(event.state));
})
复制代码
回调函数中 event 事件对象的 state
属性,指向的是 pushState 和 replaceState 方法中的状态对象(第一个参数)。该 state
对象,也可以通过 history.state
获取。
何时会触发 popstate 事件?
正如上面提到的一点,popstate
事件只在浏览器没有发生页面加载时才会触发。而,通过 history.pushState()
/history.replaceState()
方法、修改 url 的 hash 是,才会出现修改url的同时而又不会发生页面的加载。
也就是,在页面加载完成后,调用了 history.pushState()
/history.replaceState()
方法之后发生页面的前进后退(或调用 history.back()
、history.forward()
、history.go()
)时、或者修改 url 的 hash 时,才会触发 popstate
事件。
要模拟触发 popstate 事件,可以通过此处介绍的方式,修改浏览记录地址栏中的url地址或hash值,测试
popstate
事件的触发执行。
hashchange
事件
hash 即url中 #
后面的内容(包括#
)。hashchange
事件就是对url中hash值改变进行监听的事件。
hash 也可以称作 片段标识符(
fragment identifier
)。
window.addEventListener('hashchange', function() {
console.log('The hash has changed!');
if (location.hash === '#cool-feature') {
console.log("You're visiting a cool feature!");
}
}, false);
复制代码
hashchange 的事件对象包含hash改变前后的URL:oldURL 和 newURL 属性。
window.addEventListener('hashchange', function(e) {
console.log(e.oldURL);
console.log(e.newURL);
}, false);
复制代码
popstate 和 hashchange 的事件顺序
如下,通过修改 hash 值,测试 popstate 和 hashchange 事件发生的顺序。
window.location.href+='#cool-feature'
复制代码
也可以通过浏览器中的前进后退键测试。如下,是这两个事件发生输出内容的顺序:
location: http://127.0.0.1:5500/code/popstateAndhashchange.html#cool-feature
state: null
The hash has changed!
You're visiting a cool feature!
复制代码
修改hash时,会先触发 popstate 事件,然后再触发 hashchange 事件。
popstate 事件的一个优点时,可以使用
pushState
/replaceState
方法传递的状态对象,可以独立保存和传递一些数据。
popstate、hashchange 事件也是处理前端路由的基石。
popstate 仅仅监听 history 的改变,并不能阻止浏览器的前进后退,不过,可以在事件方法中,通过实现跳转、加载其他页面等各种方法,阻止默认的后退操作。
突然想到,应该记录一下 hash 的作用。当进入带hash的url中,或修改hash值时,页面会定位到id和hash值一致的元素的所在位置(或滚动到)。因此也称为锚点。 单纯的改变hash会不引起页面的重新加载,不会发送get请求。在单页应用中很流行,也更有用。
hashchange事件在处理移动端页面效果中的妙用
在移动端,处理页面中的图片时,有一个很常见的功能,点击小图在当前页面上显示大图:
来源于下面参考文章中的截图
对于手机端来说,需要考虑,当显示大图时,如何返回?在大图展示状态下,我们通常会按手机下方的返回按键,退出大图,然后之前的小图状态,继续浏览。
但是,但是,对浏览器,返回表示的是返回上一页,即返回到上一个url页面,而不是退出大图显示。
那么,可以利用 URL 的 hash 改变,页面不会重新加载,但是会被记录到浏览器的历史记录中。这一特性,来处理点击返回键,大图显示退出,而不是返回上一页。
借助 hashchange
事件,在进入大图展示状态时,加上一个特定的hash,点击返回键触发hashchange时,退出大图状态。
以下为主要处理的代码:
// 假设大图展示状态的hash为imgSlider
window.addEventListener('hashchange', function(e) {
var re = /#imgSlider$/;
if ( re.test(e.oldURL) && !re.test(e.newURL) ) {
// 假设imgSlider为大图展示组件对象
imgSlider.hide();
}
}, false);
复制代码
附:关于chrome下点击浏览器后退按钮不会触发 popstate 事件的情况
后面在查找相关资料时,看到一篇 点击浏览器后退按钮 chrome 不会触发popstate事件分析-[七日打卡5] 的文章。
里面介绍,通过 popstate 事件监听机制。实现 当页面load加载时,js执行pushState操作,在浏览器历史记录中添加一条记录;然后,监听 popstate 事件;在用户点击后退按钮触发 popstate 事件时,提示用户“是否确定退出?”。实现“用户挽留”的操作。
比如下面的简要处理,在页面onload时执行下面的函数:
function watchBackBtn() {
window.addEventListener('load', () => {
const state = {title: 'init',url: '#'};
window.history.pushState(state, 'init', '#');
});
window.addEventListener('popstate', () => {
alert('back...');
});
}
复制代码
这个处理流程没有问题。文中提到,在chrome浏览器中,只有手动在页面中进行交互操作(点击元素、滑动页面等)后再点击浏览器的后退按钮才能触发popstate 事件。
文中还引用这篇 Chrome plans to save you from sites that mess with your back button 文章介绍。是为了防止流氓网站禁止用户执行回退操作,把用户困在当前网站,而做的一个改进。【确实遇到过这种流氓界面】
也就是,Chrome中,如果进入页面后没有任何人为交互操作行为的话,是不会触发 popstate 事件的。
但是,目前实际在PC版的Chrome上测试,并没有此问题。
此处仅做个记录。
Location对象
Location
也是浏览器提供的原生对象,可以获取当前页面的URL信息以及进行操作。可以通过window.location
或document.location
属性获取该对象。
location 的属性
Location.href
:当前页面完整的 URL。等同于location.toString()
方法的返回值。Location.protocol
:当前 URL 的协议,包括冒号(:
)。Location.host
:主机。如果端口不是协议默认的80
和433
,则还会包括冒号(:
)和端口。Location.hostname
:主机名,不包括端口。Location.port
:端口号。Location.pathname
:URL 的路径部分,从根路径/
开始。Location.search
:查询字符串部分,从问号?
开始。Location.hash
:片段字符串部分,从#
开始。Location.username
:域名前面的用户名。Location.password
:域名前面的密码。Location.origin
:URL 的协议、主机名和端口。
// 当前网址为
// http://user:[email protected]:4097/path/a.html?x=111#part1
document.location.href
// "http://user:[email protected]:4097/path/a.html?x=111#part1"
document.location.protocol
// "http:"
document.location.host
// "www.example.com:4097"
document.location.hostname
// "www.example.com"
document.location.port
// "4097"
document.location.pathname
// "/path/a.html"
document.location.search
// "?x=111"
document.location.hash
// "#part1"
document.location.username
// "user"
document.location.password
// "passwd"
document.location.origin
// "http://user:[email protected]:4097"
复制代码
除了 origin
属性只读,其他属性都可通过赋值修改,实现对页面的操作。
比如,给 location.href
赋值新的URl地址,就可以实现浏览器的跳转。
// 跳转到新网址
window.location.href = 'http://www.example.com';
// 等同于
window.location = 'http://www.example.com';
复制代码
直接改写
location
和 修改href
属性是一样的。
比如,通过修改 href 或 hash
实现,跳转到指定的hash锚点。
window.location.href = '#top';
// 等同于
window.location.hash = '#top';
复制代码
Location.href属性允许跨域写入
Location.href
是浏览器唯一允许跨域写入的属性,即非同源的窗口可以改写另一个窗口(比如子窗口与父窗口)的Location.href
属性实现跳转。
Location
的其他属性都不允许跨域写入。
方法
location.assign(url) 跳转到指定的url
location.assign(url)
用于跳转到指定的url,参数必须是有效的URL字符串,否则会报错。
// 跳转到新的网址
document.location.assign('http://www.example.com')
复制代码
location.replace(url) 跳转到新的URL,并在 history 中删除当前网址
location.replace(url)
也用于跳转到新的url,不同的是,它会在浏览器的浏览历史 History
中删除当前的网址。
也就是,使用该方法跳转,后退按钮就无法回到当前页面了。即 在浏览历史中,使用新的url替换旧的url。
此方法最大的一个应用是,判断当前设备的类型,并跳转到指定版本页面。比如,判断是否为移动设备,并立刻跳转到移动版网页。
// 跳转到新的网址
document.location.replace('http://www.example.com')
复制代码
location.reload() 刷新当前页面
location.reload()
用于重新加载当前网址,相当于浏览器的刷新。
reload
接受一个布尔值参数,如果参数为true
,浏览器将向服务器重新请求这个网页,重新加载后页面将滚动到顶部(即 scrollTop === 0
)。
如果参数为 false
或为空,浏览器将从本地缓存重新加载该网页,并且重加载后的网页视口,还是之前的位置。
使用参数 true 和 false 的行为,不同的浏览器会有所差异,并不完全按照此行为
// 向服务器重新请求当前网址
window.location.reload(true);
复制代码
刷新当前页面的方法
location.reload() 标准常用刷新方法
location.reload()
是最常用的刷新当前页面的方法。
直接调用 history.go()
history.go()
不指定参数,或指定参数 0
,会刷新当前页面。
使用 document.location.assign(window.location.href)
document.location.assign(window.location.href)
也可以实现刷新当前页面。
附:location.href = window.location.href 并不能实现刷新当前页面
window.location.href=window.location.href
赋值,并不会修改 href 的url地址,因此不会有进入当前页面刷新的效果。