本解决方案原理是利用Keep-Alive、监听滚动事件 与 watch中监听$route 实现。
使用watch监听$route的方案适用于滚动区域在子组件中的情况。
因为,beforeRouteLeave 路由导航守卫,只能作用在路由组件内,不能被路由组件内的子组件所触发。
vue-router文档截图如下:
本样例使用element-ui 下的 el-table表格组件,原生或其他UI组件思路同理。
router.js
{
path: '/dispatchDetail',
name: 'dispatchDetail',
component: () => import('@/views/dispatchDetail/index.vue'),
meta: { title: '日调度情况查询', keepAlive: true } //需要缓存
},
App.vue
<keep-alive>
// 缓存组件跳转的页面
<router-view v-if="$route.meta.keepAlive"></router-view>
</keep-alive>
// 不需要缓存组件的页面
<router-view v-if="!$route.meta.keepAlive"></router-view>
dispatchDetail.vue
<template>
<el-table ref="listBox"> // 在el-table标签上 添加ref属性 用于获取滚动区域dom
...
</el-table>
</template>
<script>
export default {
data() {
return {
listBox: '',
scrollTop: 0,
}
}
// 进入缓存组件时监听滚动事件
activated() {
this.$refs.listBox.bodyWrapper.addEventListener('scroll', this.scrollToTop)
},
// 离开缓存组件时监听滚动事件
deactivated() {
this.$refs.listBox.bodyWrapper.removeEventListener('scroll', this.scrollToTop)
},
watch: {
$route(to, from) {
// 如果是来自目标页 并且this.scrollTop > 0 则给滚动区域设置离开时缓存的距顶部高度值
if (from.name == 'schedulingDetails') {
if (this.scrollTop > 0) {
setTimeout(() => {
this.$refs.listBox.bodyWrapper.scrollTo(0, this.scrollTop)
}, 0)
}
}
// 如果不是去目标页 则将 this.scrollTop 恢复为 0
if (to.name !== 'schedulingDetails') {
this.scrollTop = 0
}
},
},
methods: {
// 定义获取 距顶部滚动高度 方法
scrollToTop() {
const scrollTop = this.$refs.listBox.bodyWrapper.scrollTop // 实时获取el-table组件当前滚动位置,距顶部的滚动高度
this.scrollTop = scrollTop // 赋值给 this.scrollTop 用于下次进入时调用
},
}
}
</script>
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。