bug 记录 - 使用 el-dialog 的 before-close 的坑

需求说明

  • 弹窗中内嵌一个 form 表单
  • 原始代码
html 复制代码
<script setup lang="ts">
import { reactive, ref } from "vue"
import type { FormRules } from 'element-plus'
const ruleFormRef = ref()
interface RuleForm {
    name: string
    region: number | null
}
const ruleForm = reactive<RuleForm>({
    name: '',
    region: null,
})
const rules = reactive<FormRules<RuleForm>>({
    name: [
        { required: true, message: '请填写姓名', trigger: 'blur' },
    ],
    region: [
        { required: true, message: '请选择区域编码', trigger: 'change' },
    ]
})
const dialogVisible = ref(false);
function handelDialogPop(type: string) {
    if (type == "open") {
        dialogVisible.value = true; // 开启弹窗
    } else if (type == "submit") {
        ruleFormRef.value.validate((valid: any, fields: any) => {
            if (valid) {
                console.log('提交成功')
                // 关闭弹窗
                ruleFormRef.value.resetFields();
                dialogVisible.value = false;
            } else {
                console.log('error submit!', fields)
            }
        })
    } else if (type == "cancel") {
        // 关闭弹窗
        ruleFormRef.value.resetFields();
        dialogVisible.value = false
    }
}
</script>
<template>
    <div class="dataBox">
        <el-button type="primary" @click="handelDialogPop('open')">开启弹窗</el-button>
        <el-dialog v-model="dialogVisible" title="弹窗" width="500">
            <el-form ref="ruleFormRef" :model="ruleForm" :rules="rules" label-width="auto">
                <el-form-item label="姓名" prop="name">
                    <el-input v-model="ruleForm.name" />
                </el-form-item>
                <el-form-item label="区域" prop="region">
                    <el-select v-model="ruleForm.region" placeholder="区域编码">
                        <el-option label="第一区" :value="1" />
                        <el-option label="第二区" :value="2" />
                    </el-select>
                </el-form-item>
            </el-form>
            <template #footer>
                <div class="dialog-footer">
                    <el-button type="primary" plain @click="handelDialogPop('cancel')">取消</el-button>
                    <el-button type="primary" @click="handelDialogPop('submit')">提交</el-button>
                </div>
            </template>
        </el-dialog>
    </div>
</template>
  • 点击取消按钮,或者提交通过校验时,先执行了 form 表单的 resetFields 事件。此代码会将表单的值重置为初始值,并且移除所有校验。这样可以保证下次点开弹窗时,不会保留上次操作 form 的痕迹。
  • 为了省事儿,我把所有对 dialog 的操作都写进了 handelDialogPop 函数里,传入 type 来区分是什么事件(这里就是出问题的关键

发现问题

  • 手动点击关闭按钮,或者点击 model 空白区域时,弹窗被关闭。未执行 form 表单的 resetFields 方法,所以再次打开时,弹窗保留了上次操作的痕迹(form 表单未重置)

  • 因此需要监听 dialog 的关闭事件,官方提供了一个方法,before-close

html 复制代码
<el-dialog v-model="dialogVisible" title="弹窗" width="500" :before-close="handelDialogPop('cancel')">
...
  • 问题出现了,一进入页面时,直接抛错。报错信息可看出,before-close 被触发了

问题解决

  • 文档中表明,before-close 传参,done 说明会执行。
  • 重新为 before-close 封装一个函数

完整代码

html 复制代码
<script setup lang="ts">
import { reactive, ref } from "vue"
import type { FormRules } from 'element-plus'
const ruleFormRef = ref()
interface RuleForm {
    name: string
    region: number | null
}
const ruleForm = reactive<RuleForm>({
    name: '',
    region: null,
})
const rules = reactive<FormRules<RuleForm>>({
    name: [
        { required: true, message: '请填写姓名', trigger: 'blur' },
    ],
    region: [
        { required: true, message: '请选择区域编码', trigger: 'change' },
    ]
})
const dialogVisible = ref(false);
function beforeCloseDialog() {
    console.log("执行了 beforeCloseDialog");

    ruleFormRef.value.resetFields();
    dialogVisible.value = false
}
function handelDialogPop(type: string) {
    if (type == "open") {
        dialogVisible.value = true
    } else if (type == "submit") {
        ruleFormRef.value.validate((valid: any, fields: any) => {
            if (valid) {
                console.log('提交成功')
                beforeCloseDialog();
            } else {
                console.log('error submit!', fields)
            }
        })
    } else if (type == "cancel") {
        // 关闭弹窗
        beforeCloseDialog()
    }
}
</script>
<template>
    <div class="dataBox">
        <el-button type="primary" @click="handelDialogPop('open')">开启弹窗</el-button>
        <el-dialog v-model="dialogVisible" title="弹窗" width="500" :before-close="beforeCloseDialog">
            <el-form ref="ruleFormRef" :model="ruleForm" :rules="rules" label-width="auto">
                <el-form-item label="姓名" prop="name">
                    <el-input v-model="ruleForm.name" />
                </el-form-item>
                <el-form-item label="区域" prop="region">
                    <el-select v-model="ruleForm.region" placeholder="区域编码">
                        <el-option label="第一区" :value="1" />
                        <el-option label="第二区" :value="2" />
                    </el-select>
                </el-form-item>
            </el-form>
            <template #footer>
                <div class="dialog-footer">
                    <el-button type="primary" plain @click="handelDialogPop('cancel')">取消</el-button>
                    <el-button type="primary" @click="handelDialogPop('submit')">提交</el-button>
                </div>
            </template>
        </el-dialog>
    </div>
</template>
相关推荐
是一碗螺丝粉几秒前
React Native 运行时深度解析
前端·react native·react.js
Jing_Rainbow1 分钟前
【前端三剑客-9 /Lesson17(2025-11-01)】CSS 盒子模型详解:从标准盒模型到怪异(IE)盒模型📦
前端·css·前端框架
爱泡脚的鸡腿5 分钟前
uni-app D6 实战(小兔鲜)
前端·vue.js
青年优品前端团队7 分钟前
🚀 不仅是工具库,更是国内前端开发的“瑞士军刀” —— @qnvip/core
前端
骑自行车的码农10 分钟前
🍂 React DOM树的构建原理和算法
javascript·算法·react.js
北极糊的狐15 分钟前
Vue3 中父子组件传参是组件通信的核心场景,需遵循「父传子靠 Props,子传父靠自定义事件」的原则,以下是资料总结
前端·javascript·vue.js
看到我请叫我铁锤1 小时前
vue3中THINGJS初始化步骤
前端·javascript·vue.js·3d
q***25211 小时前
SpringMVC 请求参数接收
前端·javascript·算法
q***33371 小时前
Spring Boot项目接收前端参数的11种方式
前端·spring boot·后端
烛阴1 小时前
从`new()`到`.DoSomething()`:一篇讲透C#方法与构造函数的终极指南
前端·c#