时间:2021-04-25 10:12:25 | 栏目:vue | 点击:次
实现效果:
因为pro手脚架中封装的s-table不支持expand和expandedRowsChange事件,无法实现根据展开节点获取其内部数据的需求,因此直接使用a-table组件
表格外层可以翻页,查询携带页码参数
<a-table size="default" rowKey="dict_id" //根据自己数据内部关键针设定 ref="table" @expandedRowsChange="expandedRowsChange" @expand="expand" // 展开表格节点操作 @change="change" // 外层表格中排序,翻页,修改页面数量等操作 :expandedRowKeys="expandedRowKeys" // 操作展开的节点 :pagination="pagination" // 页码参数 :columns="columns" // 表头 :dataSource="loadData" // 数据 > <a-table size="default" style="margin-bottom:0;" rowKey="key" slot="expandedRowRender" // 以内层方式展现 :columns="innerColumns" :dataSource="data" :pagination="false" :loading="innerloading" @change="innerhandleChange" > <template v-for="(col, i) in ['item_text', 'item_value', 'item_checked', 'item_remark', 'item_sort', 'item_status']" :slot="col" slot-scope="text, record"> <a-input :key="col" v-if="record.editable" style="margin: -5px 0" :value="text" :placeholder="innerColumns[i].title" @change="e => handleChange(e.target.value, record.key, col)" /> <template v-else>{{ text }}</template> </template> // 内部表格可编辑模板 <template slot="operation" slot-scope="text, record"> <template v-if="record.editable"> <span v-if="record.isNew"> <a @click="saveRow(record)">添加</a> <a-divider type="vertical" /> <a-popconfirm title="是否要删除此行?" @confirm="remove(record.key)"> <a>删除</a> </a-popconfirm> </span> <span v-else> <a @click="saveRow(record)">保存</a> <!-- <a-divider type="vertical" /> <a @click="cancel(record.key)">取消</a> --> </span> </template> // 内部表格新增模板 <span v-else> <a @click="toggle(record)">编辑</a> </span> </template> </a-table> <div slot="expandedRowRender" style="margin: 0"> <a-button style="width: 100%; margin-top: 16px; margin-bottom: 8px" type="dashed" icon="plus" @click="newMember">新增属性</a-button> </div> <span slot="action" slot-scope="text, record"> <a @click="handleEdit(record)">编辑</a> </span> </a-table>
主要数据:
expandedRowKeys: [], // 表头 columns: [ { title: '字典编码', dataIndex: 'dict_code' }, { title: '字典名称', dataIndex: 'dict_name' }, { title: '状态', dataIndex: 'dict_status' }, { title: '字典描述', dataIndex: 'dict_remark' }, { title: '操作', width: '150px', dataIndex: 'action', scopedSlots: { customRender: 'action' } } ], loadData: [], innerColumns: [ { title: '字段名', dataIndex: 'item_text', key: 'item_text', width: '15%', scopedSlots: { customRender: 'item_text' } }, { title: '字段值', dataIndex: 'item_value', key: 'item_value', width: '15%', scopedSlots: { customRender: 'item_value' } }, { title: '默认选中(0:否,1:是)', dataIndex: 'item_checked', key: 'item_checked', width: '10%', scopedSlots: { customRender: 'item_checked' } }, { title: '备注', dataIndex: 'item_remark', key: 'item_remark', width: '20%', scopedSlots: { customRender: 'item_remark' } }, { title: '排序号', dataIndex: 'item_sort', key: 'item_sort', width: '10%', sorter: true, scopedSlots: { customRender: 'item_sort' } }, { title: '状态(1:正常,2:异常)', dataIndex: 'item_status', key: 'item_status', width: '10%', scopedSlots: { customRender: 'item_status' } }, { title: '操作', key: 'action', scopedSlots: { customRender: 'operation' } } ], data: [], innerloading: false, parameter: { pageNo: 1, pageSize: 10 }, // 排序参数 sortedInfo: null, pagination: { total: 1, current: 1, showTotal: total => `共 ${total} 条记录 第 ${this.pagination.current} / ${Math.ceil(total / this.pagination.pageSize)} 页`, showQuickJumper: true, showSizeChanger: true, pageSize: 10 }
初始进入页面时,需要获取外层表格
使用初始参数parameter请求第一页数据,从返回数据中对pagination重置翻页组件内部参数,主要有当前页,页码总量,页码大小
getDictList(this.parameter) .then(res => { if (res.code === '200') { console.log(res) this.loadData = res.data this.pagination.total = res.totalCount this.pagination.current = res.pageNo this.pagination.pageSize = res.pageSize } else { this.$message.error(res.message) } })
展开外层数据节点获取内部数据:
expand (expanded, record) { this.expandedRowKeys = [] // 重置展开节点,只展开当前点击的节点(内部数据调用模板,无法做到同时几个内层表格数据直接缓存在页面) if (expanded) { this.expandedRowKeys = [record.dict_id] this.getDictItem() // 获取表格内部数据 } console.log(expanded, record) }, // 展开节点后获取内部表格数据 getDictItem (obj) { let searchparam = { dict_id: this.expandedRowKeys[0] } if (obj) { // 内部表格除了根据其父节点id查找的条件外,还支持排序,因此需要整合排序参数 searchparam = Object.assign(searchparam, obj) } this.innerloading = true getDictItemList(searchparam).then(res => { if (res.code === '200') { this.data = res.data this.innerloading = false } else { this.$message.error(res.message) } }) }, // 外层表格操作 change (pagination, filters, sorter) { // 对页面大小,筛选,排序等条件修改后重新查询数据 this.pagination = pagination this.parameter.pageNo = pagination.current this.parameter.pageSize = pagination.pageSize this.getDict() console.log('pagination', pagination) console.log('filters', filters) console.log('sorter', sorter) }, /* 内层表格操作 */ innerhandleChange (pagination, filters, sorter) { console.log('Various parameters', pagination, filters, sorter) this.sortedInfo = { sortField: sorter.field, sortOrder: sorter.order } this.getDictItem(this.sortedInfo) },
至此,展示功能已经几乎做完啦,现在是内部表格的编辑与新增功能
handleChange (value, key, column) { // 实时更新表格中各个输入框的状态 const newData = [...this.data] const target = newData.filter(item => key === item.key)[0] if (target) { target[column] = value this.data = newData } }, toggle (data) { // 切换输入框与文本状态,实现展示与编辑功能 const target = this.data.filter(item => item.key === data.key)[0] target.editable = !target.editable console.log(this.data) }, newMember () { // 新增内容后的数据字段 this.data.push({ 'item_text': '', 'item_value': '', 'item_checked': '', 'item_remark': '', 'item_sort': '', 'item_status': '', key: this.data.length, editable: true, isNew: true }) }, saveRow (record) { this.innerloading = true if (!record.item_text || !record.item_value) { // 对必填项进行管控 this.innerloading = false this.$message.error('请至少填写填写字段名和字段值。') return } record.item_checked = record.item_checked || 0 // 设置默认值 record.item_sort = record.item_sort || 1 record.item_status = record.item_status || 1 record.dict_id = this.expandedRowKeys[0] if (record.item_id) { // 修改 updateItem(record).then(res => { if (res.code === '200') { this.$message.success(res.message) this.getDictItem() // 修改成功后重新获取当前内部表格数据 } else { this.$message.error(res.message) } }) } else { addItem(record).then(res => { if (res.code === '200') { this.$message.success(res.message) this.getDictItem() } else { this.$message.error(res.message) } }) } }, cancel (key) { const target = this.data.filter(item => item.key === key)[0] target.editable = false }, remove (key) { const newData = this.data.filter(item => item.key !== key) this.data = newData }, /* 内层表格操作结束 */
外层表格与内存表格数据示例:
{ "success": true, "code": "200", "message": "分页查询成功", "data": [{ "dict_id": 1, "dict_code": "common_org_type", "dict_name": "机构类别", "dict_pid": null, "dict_status": 1, "dict_remark": "机构类别" }, { "dict_id": 2, "dict_code": "common_user_type", "dict_name": "人员类别", "dict_pid": null, "dict_status": 1, "dict_remark": "人员类别" }, { "dict_id": 48, "dict_code": "cdsfcsdcf", "dict_name": "修改属性1", "dict_pid": null, "dict_status": 1, "dict_remark": "" }, { "dict_id": 49, "dict_code": "bugbugbug", "dict_name": "有字典id", "dict_pid": null, "dict_status": 1, "dict_remark": "" }, { "dict_id": 50, "dict_code": "1", "dict_name": "名称", "dict_pid": null, "dict_status": 1, "dict_remark": "1" }, { "dict_id": 51, "dict_code": "1", "dict_name": "1", "dict_pid": null, "dict_status": 1, "dict_remark": null }, { "dict_id": 52, "dict_code": "1", "dict_name": "1", "dict_pid": null, "dict_status": 1, "dict_remark": null }, { "dict_id": 53, "dict_code": "1", "dict_name": "1", "dict_pid": null, "dict_status": 1, "dict_remark": null }, { "dict_id": 54, "dict_code": "1", "dict_name": "1", "dict_pid": null, "dict_status": 1, "dict_remark": "" }, { "dict_id": 55, "dict_code": "dbhasuiuh", "dict_name": "测试字典1", "dict_pid": null, "dict_status": 1, "dict_remark": "备注" }], "totalCount": 11, "pageNo": 1, "pageSize": 10, "totalTag": 1 } { "success": true, "code": "200", "message": "查询成功", "data": [{ "item_id": 2, "dict_id": 1, "item_text": "外部", "item_value": "2", "item_status": 1, "item_sort": 2, "item_remark": null, "item_checked": 1, "editable": 0 // 写死就行了 一定要有 用于内部表格编辑功能 }, { "item_id": 1, "dict_id": 1, "item_text": "内部", "item_value": "1", "item_status": 1, "item_sort": 1, "item_remark": "1", "item_checked": 1, "editable": 0 }] }
补充知识:ant design Table可编辑的单元格改进版
antd官方也有给文档,但是超级麻烦,代码量还超级多,改编了一下
如图:
这里table的columns的写法,如下:
const columns2 = [ { title: '尺寸名称', dataIndex: 'name', filterDropdown: true, filterIcon: <Icon type="edit"/>, render: () => <Input />, }, { title: '标准尺寸', dataIndex: 'standard', filterDropdown: true, filterIcon: <Icon type="edit"/>, render:() => <Input />, }, { title: '上偏差', dataIndex: 'upper_deviation', filterDropdown: true, filterIcon: <Icon type="edit"/>, render:() => <Input />, }, { title: '下偏差', dataIndex: 'lower_deviation', filterDropdown: true, filterIcon: <Icon type="edit"/>, render: () => <Input />, }, { title: '工序', dataIndex: 'procedure', filterDropdown: true, filterIcon: <Icon type="edit"/>, render: () => <Select> <Option value='1'>1</Option> <Option value='2'>2</Option> </Select> }, { title: '操作', dataIndex: 'operation', render: (text, record) => ( this.state.size.length >= 1 ? ( <Popconfirm title="确定删除该信息?" onConfirm={() => this.handleDelete(record.key)}> <a href="javascript:;">删除</a> </Popconfirm> ) : null ), } ];
其中
filterDropdown: true,
filterIcon: <Icon type="edit"/>,
这两个是用于给表头添加图标,不需要可以不写
实现可编辑表格最重要的是吧这个表格所有字段都变成可控字段,这样就可以实现对表格数据的提交的基本操作
就拿我这里面的尺寸名称来说,这个字段叫name ,这里的render就要修改成render: (text, record) => <Input value={text} onChange={(e) => this.handleChange({name: e.target.value}, record)}/>,
这里参数text可以理解为 input的初始值,onChange事件是吧这个input变成可控的组件,handleChange这里有两个参数,这里“”name: e.target.value“”必须要这么传,目的是把改组件的值跟name进行绑定。
handleChange 方法
handleChange = (value, record) => { for (var i in value) { record[i] = value[i];//这一句是必须的,不然状态无法更改 this.setState({ size: this.state.size.map((item, key) => item.key == record.key ? {...item, [i]: value[i]} : item) }) } }
这里我把这个可编辑表格的值存在state中的size中,通过key进行匹配(这里key代表我这个表格的rowkey,也就是用来区分行数据的一个标识),然后修改指定行的指定数据,通过改变state中的size更新视图,同时吧更改的数据替换掉原来的 这就实现了对表格数据的实时监听,同时表格的所有数据存在了state中的size中,想要获取表格数据直接用this.state.size即可。
(这里需要注意的时,如果把表格中的某个字段,比如说name设置成表格的rowkey,就会发生name字段只能输入单个字符就会失去焦点的情况)