textare区域或者其他输入位置内容更新页面会发生偏移现象

具体说明:是公司的一个低代码平台,在一个textarea区域里可以配置json文件,但是当编辑完json文件之后不管是自动保存还是command+是(我是mac)或者是说上面的保存按钮都会使页面向上偏移一段距离(页面跳一下然后位置发生变化)

代码:

我采用了比较笨的方法就是控制滚动条的位置

javascript 复制代码
<template>
  <div class="json-editor" @mouseleave="mouseleave">
    <textarea ref="textarea" />
  </div>
</template>
<script>
    scrollCursor() {
      const scrollInfo = this.jsonEditor.getScrollInfo()
      this.jsonEditor.scrollTo(scrollInfo.left, scrollInfo.top)
    },
    mouseleave() {
      const scrollInfo = this.jsonEditor.getScrollInfo()
      const val = this.jsonEditor.getValue()
      this.$emit('input', val)
      this.$nextTick(() => {
        this.jsonEditor.scrollTo(scrollInfo.left, scrollInfo.top)
      })
    }
  }
</script>

写了一个scrollCursor 方法,调取了this.jsonEditor 这个对象的getScrollInfo()方法来获取当前编辑器的滚动信息。

getScrollInfo是一个对象通常包含以下信息:

  • left:编辑器当前的水平滚动位置,即相对于编辑器左边缘的偏移量。
  • top:编辑器当前的垂直滚动位置,即相对于编辑器顶部的偏移量。
  • width:编辑器的宽度。
  • height:编辑器的高度。
  • clientWidth:编辑器可视区域的宽度。
  • clientHeight:编辑器可视区域的高度。

然后设置滚动位置:

kotlin 复制代码
this.jsonEditor.scrollTo(scrollInfo.left, scrollInfo.top);

但是这个方法只有在command+s保存的情况下有用,对用mouseleave这个方法并没有用,经过排查是

kotlin 复制代码
this.$emit('input', val)

这一行代码导致的页面偏移(保存的按钮在父组件里面)可能是组件之间的信息传递导致父组件可能会重新渲染。这可能会导致编辑器在重新渲染后重置滚动位置,从而出现偏移现象。

也有可能是因为父组件在接收到 input 事件后,可能会对编辑器的值进行一些处理或操作,这些处理操作可能间接影响了编辑器的滚动位置。

解决办法:

kotlin 复制代码
 mouseleave() {
      const scrollInfo = this.jsonEditor.getScrollInfo()
      const val = this.jsonEditor.getValue()
      this.$emit('input', val)
      this.$nextTick(() => {
        this.jsonEditor.scrollTo(scrollInfo.left, scrollInfo.top)
      })
    }
    //或者
  mouseleave() {
  const val = this.jsonEditor.getValue();
  this.$emit('input', val);

  setTimeout(() => {
    this.scrollCursor();
  }, 0); // 延迟执行 scrollCursor 方法
}

使用 this.$nextTick 可以防止页面偏移,原因在于 this.$nextTick 的执行时机和 Vue 的响应式更新机制。具体来说,这个方法在更新 DOM 后执行,使得滚动位置调整是在 DOM 更新完成后进行,避免了页面偏移问题

使用 setTimeout(() => { this.scrollCursor(); }, 0); 可以避免页面偏移,主要是因为它将 scrollCursor 方法的执行推迟到 JavaScript 事件队列中的下一个周期。这给 Vue.js 足够的时间来处理 this.$emit('input', val) 引发的响应式更新

那为什么这段代码:

kotlin 复制代码
mouseleave() {
const val = this.jsonEditor.getValue(); 
this.$emit('input', val); 
}

就是会引起页面偏移呢?

没有 this.$nextTicksetTimeout 的代码执行顺序如下:

  1. 获取当前编辑器的值 val
  2. 触发 input 事件,导致父组件或其他地方的响应式数据更新。
  3. 响应式数据更新后,可能触发 DOM 的重新渲染和重新布局。

因为 mouseleave 事件的处理程序在 Vue 完成 DOM 更新之前立即触发并执行,所以在数据更新过程中可能引起滚动位置的变化。

使用 this.$nextTicksetTimeout 的不同

当使用 this.$nextTicksetTimeout 时,实际上是在延迟执行滚动位置的恢复操作,直到 Vue 完成其内部的 DOM 更新。这样可以确保滚动位置的调整发生在 DOM 变化之后,避免了页面偏移的问题。

相关推荐
流殇꧂13 分钟前
Vuex的模块化编程
前端·javascript·vue2
啊猪是的读来过倒38 分钟前
Vue中的axios深度探索:从基础安装到高级功能应用的全面指南
前端·javascript·vue.js·axios
光影少年39 分钟前
import和require的区别
开发语言·前端·javascript
叁分之一2 小时前
“我打包又失败了”
前端·npm
tang游戏王1232 小时前
AJAX进阶-day4
前端·javascript·ajax
无语听梧桐2 小时前
vue3中使用Antv G6渲染树形结构并支持节点增删改
前端·vue.js·antv g6
go2coding2 小时前
开源 复刻GPT-4o - Moshi;自动定位和解决软件开发中的问题;ComfyUI中使用MimicMotion;自动生成React前端代码
前端·react.js·前端框架
freesharer2 小时前
Zabbix 配置WEB监控
前端·数据库·zabbix
web前端神器2 小时前
forever启动后端服务,自带日志如何查看与设置
前端·javascript·vue.js
是Yu欸3 小时前
【前端实现】在父组件中调用公共子组件:注意事项&逻辑示例 + 将后端数组数据格式转换为前端对象数组形式 + 增加和删除行
前端·vue.js·笔记·ui·vue