element——弹窗
本来以为弹窗挺简单的,结果自己想多了。这几天的使用碰到了好多问题。
在组件中使用弹窗
<div id="app">
<el-button type="primary" @click="open=true">打开弹窗</el-button>
<test :visible.sync="open" :table="tableData"></test>
</div>
<script type="text/javascript">
//定义组件
Vue.component('test',{
template:`
<el-dialog title="收货地址" :visible.sync="visible" @close="clone">
<el-table :data="table">
<el-table-column property="date" label="日期" width="150"></el-table-column>
<el-table-column property="name" label="姓名" width="200"></el-table-column>
<el-table-column property="address" label="地址"></el-table-column>
</el-table>
</el-dialog>
`,
props:{
//visible父组件传给子组件的值,控制是否显示弹窗
visible:{
type:Boolean,
default:false
},
table:{
type:Object,
default:()=>{
return {}
}
}
},
methods:{
clone(){
this.$emit('update:visible',false)
}
}
})
new Vue({
el:'#app',
data:{
//是否打开弹窗
open:false,
tableData:[{
date:'2020-10-01',
name:'张三丰',
address:'武当山'
}]
}
})
注意点:
1、父组件
<test :visible.sync="open" :table="tableData"></test>
:visible.sync="open"
2、子组件
props:{
//visible父组件传给子组件的值,控制是否显示弹窗
visible:{
type:Boolean,
default:false
}
}
3、弹窗的关闭
如果只是关闭弹窗没有其它操作就 this.$emit('update:visible',false)
如果有其它操作就按照子组件给父组件传参的方式
<test :visible.sync="open" :table="tableData" @change="close"</test>
this.$emit('change')
change(){
this.open = false
this.init()
}
是不是固定写法不太清楚,公司很多前辈基本都这样
弹窗嵌套
今天碰见了个弹窗里套弹窗的操作,真让人头大
<template>
<el-button type="text" @click="outerVisible = true">点击打开外层 Dialog</el-button>
<el-dialog title="外层 Dialog" :visible.sync="outerVisible">
<el-dialog
width="30%"
title="内层 Dialog"
:visible.sync="innerVisible"
append-to-body>
</el-dialog>
<div slot="footer" class="dialog-footer">
<el-button @click="outerVisible = false">取 消</el-button>
<el-button type="primary" @click="innerVisible = true">打开内层 Dialog</el-button>
</div>
</el-dialog>
</template>
这个是官方的例子,重点是给外层弹窗加一个append-to-body
属性
新增与编辑操作共用一个弹窗
注意点:
1、弹窗的标题,要根据是新增还是编辑进行修改
2、先点击编辑操作,form表单数据正确。当点击新增时,form中应该没有任何数据,但是却显示之前编辑时的数据
实例
<div id="app">
<el-button type="primary" @click="addOrEdit(true,null)">新增</el-button>
<el-dialog :title="dialogT" :visible.sync="openDialog" @close="closeDialog">
<el-form label-position="right" label-width="80px" :model="form" ref="formRef">
<el-form-item label="名称" prop="name">
<el-input v-model="form.name"></el-input>
</el-form-item>
<el-form-item label="年龄" prop="age">
<el-input v-model="form.age"></el-input>
</el-form-item>
</el-form>
<el-button type="primary" @click="closeDialog">取消</el-button>
</el-dialog>
<el-table :data="tableData" style="width: 60%">
<el-table-column prop="name" label="姓名"></el-table-column>
<el-table-column prop="age" label="年龄"></el-table-column>
<el-table-column label="操作">
<template slot-scope="scope">
<div>
<el-button size="mini" @click="addOrEdit(false,scope.row)">编辑</el-button>
</div>
</template>
</el-table-column>
</el-table>
</div>
<script type="text/javascript">
new Vue({
el:'#app',
data:{
//控制弹窗的打开与关闭
openDialog:false,
dialogT:'新增',
form:{
name:undefined,
age:undefined
},
tableData:[
{name:'张三',age:15},
{name:'李四',age:20}
]
},
methods:{
addOrEdit(type,row){
this.openDialog=true
if(type){
//新增时
this.dialogT='新增'
}else{
this.dialogT='编辑'
this.form.name=row.name
this.form.age=row.age
}
},
closeDialog(){
this.openDialog=false
this.$refs['formRef'].resetFields()
}
}
})
</script>
这样写是错误的,效果图如下
看起来是没问题的,实际上是有问题的。看下面这个图
- 第一个图没问题是先打开的新增,新增时初始状态是空的,所以重置后还是为空。
- 第二个图先打开的编辑,这时初始状态为张三的值,不是没有重置而实重置后还是张三。仔细观察可以看到关闭李四时,值重置成了张三
解决:
this.dialogT='编辑'
this.$nextTick(()=>{
this.form.name=row.name
this.form.age=row.age
})
$nextTick 是在下次 DOM 更新循环结束之后执行延迟回调,在修改数据之后使用 $nextTick,则可以在回调中获取更新后的 DOM
因为form表单初始值都是underfunded,所以在初始化完成后再赋值,初始值是underfunded。之前是因为打开弹窗时还没有初始化完成,这时赋值会替代原来的underfunded
查看详情
<div id="app">
<el-table :data="tableData" style="width: 60%">
<el-table-column prop="name" label="姓名"></el-table-column>
<el-table-column prop="age" label="年龄"></el-table-column>
<el-table-column label="操作">
<template slot-scope="scope">
<div>
<el-button size="mini" @click="showDetail(scope.row)">查看</el-button>
</div>
</template>
</el-table-column>
</el-table>
<detail :visible.sync="openDialog" :details="rowData"></detail>
</div>
<script type="text/javascript">
Vue.component('detail',{
template:`
<el-dialog title="查看详情" :visible.sync="open" @close="closeDialog">
<el-form label-position="right" label-width="80px" :model="form" ref="formRef">
<el-form-item label="名称" prop="name">
<el-input v-model="form.name"></el-input>
</el-form-item>
<el-form-item label="年龄" prop="age">
<el-input v-model="form.age"></el-input>
</el-form-item>
</el-form>
</el-dialog>
`,
props:{
visible:{
type:Boolean,
default:false
},
details:{
type:Object,
default:()=>{
return {}
}
}
},
data(){
return {
form :{},
open:false
}
},
watch:{
details(nval,oval){
this.open=this.visible
this.$nextTick(()=>{
this.form=this.details
})
}
},
methods:{
closeDialog(){
this.$emit('update:visible',false)
}
}
})
new Vue({
el:'#app',
data:{
//控制弹窗的打开与关闭
openDialog:false,
//当前行信息
rowData:{},
tableData:[
{name:'张三',age:15},
{name:'李四',age:20}
]
},
methods:{
showDetail(row){
this.openDialog=true
this.rowData=row
}
}
})
</script>