破解Vue自定义弹窗销毁的"nextSibling"之谜

破解Vue自定义弹窗销毁的"nextSibling"之谜

问题场景描述

在动态创建的Dialog组件内部再次使用Dialog时,触发Cannot read properties of null (reading 'nextSibling')错误。

最小复现代码

1.弹窗前端组件

html 复制代码
<!-- ParentDialog.vue -->
<template>
  <!-- 父弹窗直接使用el-dialog -->
  <el-dialog v-model="parentVisible" title="父弹窗">
    <button @click="openChild">打开子弹窗</button>
    <!-- 子弹窗组件 -->
    <el-dialog v-model="childVisible" title="子弹窗" width="60%" append-to-body>
      <p>我是嵌套的子弹窗内容</p>
    </el-dialog>
  </el-dialog>
</template>
<script setup>
    import { ref } from 'vue'
    const parentVisible = ref(true) 
    const childVisible = ref(false) 
    const openChild = () => { childVisible.value = true }
</script>

2.调用代码

typescript 复制代码
import ParentDialog from "@component/ParentDialog.vue"
  function showDialog(){
  createDialog(ParentDialog);
}

3.createDialog函数的代码

typescript 复制代码
import ElementUi from 'element-plus';

export const createDialog = (
  Node: Component<any, any, any, ComputedOptions, MethodOptions, {}, any>
) => {
  const div = document.createElement('div');
  div.id = 'test-dialog';
  document.body.appendChild(div);

  const app = createApp(Node, {
    onClose() {
      console.log('弹窗关闭了');
      app.unmount();
      div.remove();
    }
  });

  app.use(ElementUi);

  app.mount('#test-dialog');
};

问题出在了哪呢?

问题主要来源于 Vue 组件生命周期与 DOM 操作的冲突。在嵌套使用 el-dialog 时:

  1. 动态创建和销毁组件 :父弹窗和子弹窗分别以动态方式挂载到 DOM 上,子弹窗由于 append-to-body 被移到 body 下,这使得 Vue 的虚拟 DOM 与真实 DOM 不完全同步。
  2. nextSibling 属性访问Element-Plus 在处理某些 DOM 操作时,依赖父子元素的 DOM 顺序,但动态挂载会导致 DOM 层级被打乱,从而触发错误。

问题如图所示

sequenceDiagram participant App as Vue App participant Parent as Parent Dialog participant Child as Child Dialog participant DOM as Real DOM App->>+Parent: 1. 创建父弹窗组件 Parent->>+DOM: 2. 挂载到 DOM Parent->>+Child: 3. 动态创建子弹窗 Child->>+DOM: 4. 使用 append-to-body 挂载到 body Note over DOM: 子弹窗被移到 body 下 Parent->>Child: 5. 操作子弹窗状态 Child->>DOM: 6. 更新挂载位置 DOM->>DOM: 7. 尝试 nextSibling 获取 DOM-->>-App: 抛出错误<br>"Cannot read properties of null" deactivate DOM deactivate Child deactivate Parent

解决办法详解

  1. 使用 teleport 明确挂载位置
    在弹窗嵌套场景下,通过 teleport 将子弹窗直接挂载到 body 下,这样可以避免 Vue 在嵌套管理 DOM 时的复杂性,保持组件逻辑清晰。
  2. 调整 append-to-body 和挂载点
    在子弹窗的 el-dialog 上启用 append-to-body,但需确保它的生命周期与父组件一致。
  3. 事件冒泡和通信优化
    使用 Vue 的 emit 或者自定义事件来控制子弹窗的打开和关闭状态,而不是直接操作 DOM。这样可以减少对 nextSibling 的依赖。

解决方案如图所示

sequenceDiagram participant App as Vue App participant Parent as Parent Dialog participant Child as Child Dialog participant DOM as Real DOM App->>+Parent: 1. 创建父弹窗组件 Parent->>+DOM: 2. 挂载到 DOM Parent->>+Child: 3. 使用 Teleport 明确挂载位置 Child->>+DOM: 4. 挂载到 body Parent->>Child: 5. 操作子弹窗状态 Child->>DOM: 6. Teleport 管理挂载 DOM-->>-App: 7. 正常完成所有操作 deactivate Child deactivate Parent

修改后的代码

html 复制代码
<template>
  <!-- 父弹窗 -->
  <el-dialog v-model="dialogState.parentVisible" title="父弹窗" @close="emitClose">
    <button @click="openChild">打开子弹窗</button>
    <!-- 子弹窗通过 teleport 挂载到 body -->
    <teleport to="body">
      <el-dialog
        v-model="dialogState.childVisible"
        title="子弹窗"
        width="60%"
        @close="handleChildClose"
      >
        <p>我是嵌套的子弹窗内容</p>
      </el-dialog>
    </teleport>
  </el-dialog>
</template>

<script lang="ts" setup>
import { ref, reactive, defineExpose, defineProps, defineEmits } from 'vue';

// 定义 props 和 emits
const props = defineProps({
  onClose: {
    type: Function,
    default: null,
  },
});
const emit = defineEmits(['close']);

// 控制弹窗状态
const dialogState = reactive({
  parentVisible: true,
  childVisible: false,
});

// 定义 expose 方法供父组件调用
defineExpose({
  close: () => {
    dialogState.parentVisible = false;
  },
});

// 打开子弹窗
const openChild = () => {
  dialogState.childVisible = true;
};

// 子弹窗关闭逻辑
const handleChildClose = () => {
  dialogState.childVisible = false;
};

// 父弹窗关闭时触发回调
const emitClose = () => {
  if (typeof props.onClose === 'function') {
    props.onClose();
  }
  emit('close');
};
</script>

描述 :仅仅改动了 子弹窗的挂载方式,将其通过 teleport 明确指定到 body ,从而避免 DOM 层级错乱的问题,而 createDialog 函数的代码不需要修改哦~

总结

其实只需要在组件中使用 teleport 明确指定子弹窗的挂载位置,同时让父子弹窗的状态管理逻辑清晰即可啦~

相关推荐
爱勇宝6 分钟前
《置身钉内》之后:普通前端的出路在哪里?
前端·后端·程序员
KaMeidebaby10 分钟前
卡梅德生物技术快报|羊驼免疫:分子生物学实战:基于羊驼免疫的重链抗体制备与全流程验证方案
前端·网络·数据库·人工智能·算法·百度
MacroZheng14 分钟前
别再求前端了!这款对标Claude Design的开源工具,让你一秒拥有顶级设计能力!
前端·vue.js·人工智能
namexingyun20 分钟前
GPT-5.6 前端生成能力深度解析:kindle/kepler/Levi三版本UI实测与技术推演
java·前端·人工智能·gpt·机器学习·ui
掘金酱22 分钟前
📱 TRAE SOLO 移动端上线征文——“我的第一次移动端AI办公” 评测 | 获奖名单公示
前端·人工智能·trae
随风行酱28 分钟前
前端工程师的副业之路:周末跑滴滴的真实体验
前端·javascript·ai编程
北城笑笑33 分钟前
Vibe Coding 主流 AI 编程工具:Claude Code 与 Codex 全面解析( Claude and Codex )
前端·ai·ai编程·fpga
Darling噜啦啦35 分钟前
JS 数据结构实战:从栈队列到链表,一文吃透数组底层原理与线性数据结构
前端·javascript·数据结构
Asize1 小时前
重生之我在 Vibe Coding 时代当程序员:第十五课,正则表达式和 HTTP 请求:规则不是背出来的,是拆出来的
前端·javascript·后端
Mintopia1 小时前
从意图到评估:理解用户操作产品的完整行动链路
前端