做一个 ElementUI 弹框组件的二次封装
效果如下:
点击取消按钮发现弹出如下报错信息 :
[Vue warn]: Avoid mutating a prop directly since the value will be overwritten whenever the parent component re-renders. Instead, use a data or computed property based on the prop's value. Prop being mutated: "dialogVisible"
错误的写法:
子组件 MyDialog3.vue
<!-- 子组件 -->
<template><!-- elmentui 的 Dialog --><el-dialog :title=title:visible="dialogVisible"><span slot="footer" class="dialog-footer"><el-button @click="handleCancel">取 消</el-button><el-button type="primary" @click="">确 定</el-button></span></el-dialog>
</template><script>
export default {name: "MyDialog3",props: ['title', 'dialogVisible'],methods: {handleCancel() {this.dialogVisible = false; // 这是错误的写法, 不能在子组件中直接修改props的值}}
}
</script><style scoped></style>
父组件 MyDialog3Test.vue
<template><div><el-button @click="btnClick">点我弹框</el-button><MyDialog3:title="headerText":dialog-visible="dialogShow"></MyDialog3></div>
</template><script>
import MyDialog3 from "@/components/dialog3/MyDialog3";export default {name: "MyDialog3Test",components: {MyDialog3},data() {return {headerText: '测试弹框',dialogShow: false,}},methods: {btnClick() {this.dialogShow = true;}}
}
</script><style scoped></style>
出现这个错误的原因是 父子组件在进行通信时, 子组件直接修改了 props的某个属性的值.
在 Vue.js 中,直接修改 prop(属性)的值是被严格禁止的,因为这会导致父子组件之间的数据流变得难以追踪和调试。Vue 设计之初就采用了单向数据流的原则,即父组件通过 prop 向下传递数据到子组件,而子组件应该通过事件(如自定义事件)来通知父组件更新数据,而不是直接修改 prop 的值。
正确的写法如下:
在子组件中取消按钮的点击事件中不修改 dialogVisible 属性的值, 而是通过 $emit 发送一个自定义事件 dialog-close, 在父组件中使用 @dialog-close 去申明处理这个事件. 代码如下:
子组件:
父组件:
总结:
在 Vue.js 中,单向数据流(One-Way Data Flow)是一个重要的设计理念,指的是数据从父组件通过 props 向下传递给子组件,而子组件不能直接修改这些数据。如果子组件需要修改数据,应该通过事件通知父组件,由父组件更新数据。这种机制确保了组件之间的关系清晰,易于维护。