效果图展示
当在表格中点击编辑按钮时:点击的行变成文本框且数据回显可以点击确定按钮修改数据或者取消修改回退数据:
具体实现步骤
1. 行数据定义编辑标记
行数据定义编辑标记
当在组件中获取到用于表格展示数据的方法中,针对每一行数据添加一个编辑标记
this.list.forEach(item => {
// item.isEdit = false // 添加一个属性 初始值为false
// 数据响应式的问题 数据变化 视图更新
// 添加的动态属性 不具备响应式特点
// this.$set(目标对象, 属性名称, 初始值) 可以针对目标对象 添加的属性 添加响应式
this.$set(item, 'isEdit', false)
})
注意:为什么不使用item.isEdit = false , 因为动态添加的属性不具备响应式的特点,如果想要具备响应式,可以使用$set
2. 点击行编辑标记状态(isEdit
)
点击编辑时,将当前行的标记isEdit设置为true
<el-table-column align="center" label="操作">
<template>
<!-- 非编辑状态 -->
<el-button size="mini" type="text">分配权限</el-button>
<el-button size="mini" type="text" @click="btnEditRow(row)">编辑</el-button>
<el-button size="mini" type="text">删除</el-button>
</template>
</el-table-column>
点击编辑的方法
// 点击编辑行
btnEditRow(row) {
row.isEdit = true // 改变行的编辑状态
}
3. 插槽根据标记渲染表单
在el-table-column
组件中使用<template v-slot="{ row }">
作用域插槽获取当前行的信息
所以在表格列中可以借助当前行的信息根据当前的isEdit标记-渲染结构
<el-table-column prop="name" align="center" width="200" label="角色">
<template v-slot="{ row }">
<!-- 条件判断 -->
<el-input v-if="row.isEdit" size="mini" />
<span v-else>{{ row.name }}</span>
</template>
</el-table-column>
<el-table-column prop="state" align="center" width="200" label="启用">
<!-- 自定义列结构 -->
<template v-slot="{ row }">
<el-switch v-if="row.isEdit" />
<span v-else> {{ row.state === 1 ? "已启用" : row.state === 0 ? "未启用" : "无" }} </span>
</template>
</el-table-column>
$set的应用
- this.$set(目标对象, 属性名称, 初始值 )
- 等价于 Vue.set(目标对象, 属性名称, 初始值)
- 向响应式对象中添加一个属性,并确保这个新属性同样是响应式的,且触发视图更新。
行内编辑-数据缓存
行内编辑需要做数据缓存
为什么要做数据缓存?
因为编辑时,可以取消会滚到之前的状态,所以编辑时的数据是临时的数据。
假设list
数组中的name
,state
,description
三个属性是用于表格每列展示的数据,当进行行内编辑时需要将这3个属性进行缓存,因为编辑的数据是临时的
如图,editRow的数据是针对当前行的数据做了一份拷贝,针对这个拷贝,我们可以随意修改。
1. 初始化时缓存数据
//缓存数据
this.list.forEach(item => {
this.$set(item, 'isEdit', false)
this.$set(item, 'editRow', {
name: item.name,
state: item.state,
description: item.description
})
})
2. 点击编辑时更新缓存数据
btnEditRow(row) {
row.isEdit = true // 改变行的编辑状态
// 更新缓存数据
row.editRow.name = row.name
row.editRow.state = row.state
row.editRow.description = row.description
}
3. 将编辑时的表单双向绑定缓存数据
<el-table-column prop="name" align="center" width="200" label="角色">
<template v-slot="{ row }">
<!-- 条件判断 -->
<!-- 编辑状态 -->
<el-input v-if="row.isEdit" v-model="row.editRow.name" size="mini" />
<!-- 非编辑状态 -->
<span v-else>{{ row.name }}</span>
</template>
</el-table-column>
行内编辑-确定取消按钮
1. 确定按钮
el-table
中不能使用表单所以在确定按钮的回调中要进行手动判断
然后将编辑好的数据封装到一个对象中发起请求
下方使用Object.assign
方法是因为发送完请求,数据库数据已更新,而前端页面也需要更新
<el-button type="primary" size="mini" @click="btnEditOK(row)">确定</el-button>
// 点击确定时触发
async btnEditOK(row) {
if (row.editRow.name && row.editRow.description) {
// 下一步操作
await updateRole({ ...row.editRow, id: row.id })
// 更新成功
this.$message.success('更新角色成功')
// 更新显示数据 退出编辑状态
// row.name = row.editRow.name // eslint的一校验 误判
// Object.assign(target, source)
Object.assign(row, {
...row.editRow,
isEdit: false // 退出编辑模式
}) // 规避eslint的误判
} else {
this.$message.warning('角色和描述不能为空')
}
}
注意: 这里既然更新成功了,要把缓存数据回显到页面上,并且关闭编辑模式
2. 取消按钮
关闭编辑状态即可
<el-button size="mini" @click="row.isEdit = false">取消</el-button>