popstate 监听app、浏览器的返回

本文介绍了一种在Web开发中监听app或浏览器返回事件的实用方法,通过使用history.pushState和监听popstate事件,可以有效捕捉页面的返回操作,适用于各种应用场景。

在实际开发中,遇到需要在当前web页面监听app、浏览器返回的情况,进过一顿搜索后,终于找到一种比较靠谱的方式!

下面话不多说,直接上代码:

<!DOCTYPE html>
<html>
<head>
	<meta charset="utf-8">
	<meta name="viewport" content="initial-scale=1.0,maximum-scale=1.0,minimum-scale=1.0,user-scalable=no" />
	<title>popstate 监听返回</title>
	<style type="text/css">
	</style>
</head>
<body>
    
	<div class="cat">应用场景一:监听当前页面返回</div>
    <div class="cat">应用场景二:监听第三方返回</div>
	<a href="http://baidu.com">跳转到百度</a>
	<script>
	    window.onload = function(){
	        pushHistory();
            setTimeout(function(){ //处理:部分机型加载页面时即会执行popstate事件的问题
              /*popstate触发时机:当前历史条目为pushState创建时,事件被触发*/
              window.addEventListener('popstate',function(event){
                    let state = event;
                    console.log(state);
                    alert(state + "-------------");
              },false);
            },100);
	    }
	    /*pushState:往history添加历史条目
	      state:状态对象,popstate事件触发时,该对象会传入回调函数  
          title:新页面标题(目前浏览器忽略该参数)
	      URL:新的历史URL记录*/
	    function pushHistory(){
	        var state = {title: "呵呵哒",url: "#"};
	        window.history.pushState(state,"新页面标题#","new_url");
	    }
	</script>
</body>
</html>

 

<think>我们面临的核心问题是:在Vue应用中,当用户点击浏览器左上角的返回按钮时,我们希望能够拦截这个行为,并执行自定义的导航逻辑(例如弹出确认对话框,或者跳转到指定页面)。 根据引用[2][3][4]的内容,我们可以总结出以下关键点: 1. 使用`window.history.pushState`方法向浏览器的历史记录中添加一条记录,这样当用户点击返回按钮时,会先回到我们添加的这个记录(当前页面),从而触发`popstate`事件。 2. 通过监听`popstate`事件,我们可以在这个事件的处理函数中执行自定义操作(例如弹出确认框,或者跳转到其他页面)。 3. 注意在组件销毁前移除事件监听,避免内存泄漏。 具体实现步骤: 1. 在组件的`mounted`(或`onMounted`,如果使用Composition API)钩子中: - 使用`history.pushState`添加一条历史记录(这样当前页面会被记录两次,但URL不变)。 - 添加`popstate`事件监听器,绑定自定义处理函数。 2. 在自定义处理函数中: - 执行需要的操作(例如提示用户是否确认离开)。 - 如果用户确认离开,则移除事件监听,并执行返回操作(或者使用`router.go(-1)`等)。 - 如果用户取消离开,则再次使用`history.pushState`将当前页面加入历史记录(这样用户再次点击返回时又会触发我们的监听)。 3. 在组件的`beforeDestroy`(或`onBeforeUnmount`)钩子中移除事件监听。 但是注意:这种方法会改变历史记录栈,可能会让用户需要点击多次返回才能离开当前页面(因为我们在历史记录中插入了多条当前页面的记录)。因此,我们需要在确认离开时,移除事件监听,并让浏览器正常返回(即让历史记录回退一步)。 然而,如果我们希望直接跳转到另一个页面(比如回到原Vue网站),则可以使用路由跳转。 下面是一个在Vue组件中实现的示例(使用Options API): 注意:这种方法适用于监听浏览器返回按钮(包括左上角的返回按钮),但不适用于Vue Router的编程式导航(如this.$router.go(-1)),因为编程式导航不会触发popstate事件(但会触发Vue Router的导航守卫)。因此,我们需要区分处理。 但是,根据引用[4],我们也可以结合Vue Router的导航守卫来处理,但这里我们主要针对浏览器左上角的返回按钮(它触发的是浏览器popstate事件)。 示例代码:</think>### Vue 监听浏览器返回按钮并自定义导航行为 在 Vue 中监听浏览器左上角的返回按钮事件并自定义导航行为,核心是通过 **`popstate` 事件 + `history.pushState`** 实现。以下是完整解决方案: #### 核心实现代码(Vue 3 Composition API) ```vue <script setup> import { onMounted, onBeforeUnmount } from 'vue'; import { useRouter } from 'vue-router'; const router = useRouter(); // 自定义返回处理逻辑 const handleBack = () => { if (confirm('确定要离开当前页面吗?未保存的数据将丢失')) { // 确认返回:跳转到指定页面 router.push('/home'); } else { // 取消返回:重新添加历史记录 history.pushState(null, null, location.href); } }; // 添加历史记录并监听事件 const setupHistoryListener = () => { if (window.history && window.history.pushState) { history.pushState(null, null, location.href); window.addEventListener('popstate', handleBack); } }; onMounted(setupHistoryListener); onBeforeUnmount(() => { window.removeEventListener('popstate', handleBack); }); </script> ``` #### 关键实现原理 1. **`history.pushState`** 在历史栈中添加当前页面的记录,使浏览器返回按钮需要两次点击才能离开当前页: ```js history.pushState(null, null, location.href); ``` - 第一个参数:状态对象(可存储页面数据) - 第二个参数:标题(现代浏览器忽略) - 第三个参数:URL(保持当前URL不变) 2. **`popstate` 事件监听** 监听浏览器返回操作: ```js window.addEventListener('popstate', handleBack); ``` 3. **自定义导航逻辑** 在事件处理函数中实现业务逻辑: ```js const handleBack = () => { if (confirm('确定离开?')) { router.push('/target-page'); // Vue Router跳转 } else { history.pushState(null, null, location.href); // 阻止返回 } }; ``` #### 完整实现方案(Vue 2 Options API) ```vue <script> export default { mounted() { this.setupHistoryListener(); }, beforeDestroy() { window.removeEventListener('popstate', this.handleBack); }, methods: { setupHistoryListener() { if (window.history && window.history.pushState) { history.pushState(null, null, location.href); window.addEventListener('popstate', this.handleBack); } }, handleBack() { if (this.$route.query.fromExternal) { // 从外部网站返回的特殊处理 this.$router.replace('/dashboard'); } else if (this.hasUnsavedChanges) { // 有未保存数据的确认 if (confirm('数据未保存,确定离开?')) { this.$router.go(-1); } else { history.pushState(null, null, location.href); } } else { // 默认返回行为 this.$router.go(-1); } } } } </script> ``` #### 实际应用场景 1. **阻止未保存离开** ```js handleBack() { if (this.unsavedChanges && !confirm('数据未保存,确定离开?')) { history.pushState(null, null, location.href); } } ``` 2. **跳转到指定页面** ```js handleBack() { this.$router.replace('/home'); // 替代默认返回 } ``` 3. **多级外部网站返回控制** ```js // 跳转外部网站时添加标记 gotoExternal() { this.$router.push({ path: '/redirect', query: { url: encodeURIComponent('https://external.com'), from: 'app' } }) } // 返回时检测标记 handleBack() { if (this.$route.query.from === 'external') { this.$router.replace('/recovery-page'); } } ``` #### 注意事项 1. **内存泄漏预防** 务必在 `beforeUnmount` 中移除事件监听: ```js onBeforeUnmount(() => { window.removeEventListener('popstate', handleBack); }); ``` 2. **与 Vue Router 的兼容** - 使用 `router.push/go` 代替 `location.href` 确保路由状态一致 - 在路由守卫中处理程序化导航: ```js router.beforeEach((to, from, next) => { if (from.meta.requiresBackConfirm) { // 处理程序化导航的确认 } next(); }); ``` 3. **移动端特殊处理** 对于 Hybrid App 中的 WebView,需结合平台 API: ```js // 监听 Android 物理返回键 if (window.plus) { plus.key.addEventListener('backbutton', handleBack); } ``` #### 方案对比 | 方法 | 优点 | 缺点 | |--------------------|-------------------------|-------------------------| | `popstate` 监听 | 纯前端实现,兼容性好 | 需处理历史记录栈 | | Vue Router 守卫 | 与路由系统集成度高 | 无法捕获浏览器返回按钮 | | 平台 API | 支持移动端物理按键 | 仅限特定环境(如 5+ Runtime)| > **最佳实践**:对于纯 Web 应用,优先使用 `popstate + pushState` 方案;对于混合应用,结合平台 API 实现[^2][^4]。
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值