44 el-dialog 的 appendToBody 属性, 导致 vue 响应式失效

前言

我们经常会碰到 一些 模型和视图 不同步的问题

通常意义上 主要的问题为 列表的某响应式数据更新着更新着 后面就变成非响应式对象了, 然后 就造成了 数据一直在更新, 但是 视图的渲染后面就未渲染了, 这是一个由于 模型上的问题 导致的数据的不在响应式更新

又或者 是因为使用了相关 vue 未能够检测到的 api 来更新 对象, 数组 的元素, 然后导致的数据的不同步, 数据更新了 但是视图未更新

然后 我们这里来看一下 另外的一个 由于vue这边视图更新造成的一个数据 不同步的情况, 这里的情况是 模型里面增加了一条数据, 但是 页面上 没有渲染出来

这个问题 还是很有意思, 是和 vue 这边渲染视图元素有一些关系

核心比较关键的元素是 el-dialog, 然后其中有一个 appendToBody 的属性

vue 模型视图不同步同类型题材的文章可以参见

el-tree defaultCheckedKeys配置 和 树上面选中节点不同步问题

特定的操作之后响应式对象不"响应"了(一)

特定的操作之后响应式对象不"响应"了(二)

直接使用 dom api 更新了 #text节点, 之后响应式更新不生效了

测试用例

<template>
  <div class="testParent">

    <div v-for="dayPlan in weekPlan" style="display: flex;" >
      <el-tag type="warning">{{dayPlan.name}}</el-tag>
      <div class="block" v-for="biz in dayPlan.children" style="display: inline; float:left;">
        <el-tag type="warning">{{biz.time}}</el-tag>
        <el-input v-model="biz.biz" :name="biz.biz" ></el-input>
      </div>

      <el-dialog :visible.sync="visible" v-if="dialogVisible" width="480px" :show-close="true" :modal="false"
                 :append-to-body="true" >
        <span>是否退出登录?</span>
      </el-dialog>
    </div>

    <el-button style="position: absolute; top: 100px; left : 1500px; " @click="handleClick" > click </el-button>

  </div>

</template>

<script>

  export default {
    name: "HelloElInputUpdate",
    data() {
      return {
        visible: false,
        dialogVisible: true,
        weekPlan: [
          {
            id: "01", name: "monday",
            children: [
              {
                time: "morning",
                biz: "chinese"
              }, {
                time: "afternoon",
                biz: "english"
              }
            ]
          },
          {
            id: "02", name: "tuesday",
            children: [
              {
                time: "morning",
                biz: "math"
              }, {
                time: "afternoon",
                biz: "english"
              }
            ]
          },
          {
            id: "03", name: "wednesday",
            children: [
              {
                time: "morning",
                biz: "math"
              }, {
                time: "afternoon",
                biz: "english"
              }
            ]
          }
        ],
      }
    },
    computed: {},
    mounted() {
      let _this = this
      setTimeout(function() {
        _this.weekPlan[2].children[0].biz = "updated"
        console.log(" updated ")
      }, 5000)

      setTimeout(function() {
        _this.weekPlan[1].children.push({time: "night", biz: "tv"})
        console.log(" newly created ")
      }, 6000)

    },
    created() {
    },
    methods: {
      handleClick() {
        this.visible = !this.visible
      }
    },
  }
</script>

<style>

</style>

正常情况

如果是 不点击 click 按钮来操作 对话框相关

可以看到 mounted 之后的两个操作是正常的, 最终 tuesday 可以看到增加的晚上的计划, wednesday morning 的事项发生了改变

异常情况

此时 就需要我们 在 mounted 的操作之前, 点击 click 按钮了

然后 我们看一下 情况是怎么样的?

可以看到 wednesday morning 的事项发生了改变, 但是 tuesday 增加的晚上的计划 在页面是没有展示出来的

首先我们通过 click 按钮, 来看一下 整个数据模型的情况

可以看到 在数据模型上面, 是正确的, tuesday 增加的晚上的计划 是已经在添加进去了

我们再看一下 页面元素渲染的地方

这里可以看到 vue 这边根据页面元素渲染 vnode 的时候也是正确的, 可以正确拿到 tuesday 的三个计划

这里的 patchVNode 是一层一层的再比较

然后 这里可以通过上下文 parentElm, oldCh, newCh 可以判断出当前是在比较 tuestday 中的改动前后的元素

截图 newCh 总共有五个元素, 一个是 el-tag, 中间三个为 tuesday 的 三个工作计划, 最后一个为 对话框的元素

然后这里 和 oldCh 对比, oldCh 只有四个元素, 一个是 el-tag, 中间两个为 tuesday 的 三个工作计划, 最后一个为 对话框的元素

然后比较 oldCh, newCh 的差异是需要在 oldCh 的第二个工作计划之后再增加一个工作计划 : {time : "night" , biz : "tv"}

然后 后面会 根据 vnode 渲染时间的 dom 元素, 然后 添加到页面中

这里会创建 工作计划 : {time : "night" , biz : "tv"} 的父级 div 元素, 以及子级的 el-tag 和 el-input, 这个具体的细节 我们先暂时不管, 可以理解为我们这里 vnode.elm 即为创建好的 dom 元素

然后 另外一个比较关键的元素是 refElem, 是一个参照元素, 标记了目标新增元素 应该放在那里

在我们这里的场景新增的 工作计划 : {time : "night" , biz : "tv"} 按道理其之后应该是 对话框节点, 所以所这里计算的 refElm 即为对话框对应的节点

但是 这个节点因为我们前面配置了 appendToBody 的配置, 这个节点被移动出去了, 在dom层级上已经没有这个节点了, 甚至连它的 comment 节点也没有了

vue 更新页面元素的时候, 数据的 dom 结构如下

可以看到在 monday, tuesday, wednesday 的数据节点下面是没有只有 el-tag, 两个工作节点, 没有 vnode模型上面的 dialog 节点

然后 dialog 节点是在 外层的 body 下面

然后这里添加 dom 节点的时候, 发现 ref节点 甚至不在 parent 下面, 然后 直接跳过了 dom 元素的添加

因此 最终页面上 看不到这个本应该渲染的 工作节点

引起这里的问题, 主要的原因是 dom 结构变化了, 但是 vnode 的结构却没有发生变化

进而导致 vue 这边根据 vnode 的结构渲染新的元素的时候, 出现异常

解决的方式, 不要将 dialog 放置于可能新增元素的节点 后面, 另外用例中 将 dialog 放置于循环中 本来也是一个问题

如果 appendToBody 配置为 false 会怎么样 ?

点击了 dialog 之后, 可以看到对应的 wrapper 的 div 依然还是在原来的 dom 层级上面

然后 vue 这时候基于 dialog节点 来添加元素, 可以正常添加到

添加目标工作节点的时候

相关推荐
Myli_ing13 分钟前
HTML的自动定义倒计时,这个配色存一下
前端·javascript·html
I_Am_Me_44 分钟前
【JavaEE进阶】 JavaScript
开发语言·javascript·ecmascript
℘团子এ1 小时前
vue3中如何上传文件到腾讯云的桶(cosbrowser)
前端·javascript·腾讯云
学习前端的小z1 小时前
【前端】深入理解 JavaScript 逻辑运算符的优先级与短路求值机制
开发语言·前端·javascript
星星会笑滴1 小时前
vue+node+Express+xlsx+emements-plus实现导入excel,并且将数据保存到数据库
vue.js·excel·express
前端百草阁1 小时前
【TS简单上手,快速入门教程】————适合零基础
javascript·typescript
彭世瑜1 小时前
ts: TypeScript跳过检查/忽略类型检查
前端·javascript·typescript
Backstroke fish1 小时前
Token刷新机制
前端·javascript·vue.js·typescript·vue
zwjapple1 小时前
typescript里面正则的使用
开发语言·javascript·正则表达式
小五Five1 小时前
TypeScript项目中Axios的封装
开发语言·前端·javascript