asb_cloud_front/src/components/Table/EditableTable.jsx

176 lines
5.0 KiB
React
Raw Normal View History

import { Form, Table, Button, Popconfirm } from "antd"
import { EditOutlined, SaveOutlined, PlusOutlined, CloseCircleOutlined, DeleteOutlined } from '@ant-design/icons'
2021-08-11 11:44:20 +05:00
import { useState, useEffect } from "react";
import { EditableCell } from './EditableCell'
2021-07-30 16:14:07 +05:00
const newRowKeyValue = 'newRow'
2021-08-19 11:40:26 +05:00
const tryAddKeys = (items) => {
if(!items?.length)
return []
if(items[0].key)
return items
return items.map((item, index) => ({...item, key: item.key ?? item.id ?? index }))
}
export const EditableTable = ({
columns,
dataSource,
onChange, // Метод вызывается со всем dataSource с измененными элементами после любого действия
onRowAdd, // Метод вызывается с новой добавленной записью. Если метод не поределен, то кнопка добавления строки не показывается
onRowEdit,// Метод вызывается с новой отредактированной записью. Если метод не поределен, то кнопка редактирования строки не показывается
onRowDelete,// Метод вызывается с удаленной записью. Если метод не поределен, то кнопка удаления строки не показывается
...otherTableProps}) => {
2021-07-30 16:14:07 +05:00
const [form] = Form.useForm()
2021-08-19 11:40:26 +05:00
const [data, setData] = useState(tryAddKeys(dataSource))
2021-07-30 16:14:07 +05:00
const [editingKey, setEditingKey] = useState('')
2021-08-11 11:44:20 +05:00
useEffect(()=>{
2021-08-19 11:40:26 +05:00
setData(tryAddKeys(dataSource))
2021-08-11 11:44:20 +05:00
},[dataSource])
2021-07-30 16:14:07 +05:00
const isEditing = (record) => record.key === editingKey
const edit = (record) => {
form.setFieldsValue({...record})
setEditingKey(record.key)
}
const cancel = () => {
2021-08-11 11:44:20 +05:00
if(editingKey === newRowKeyValue)
{
const newData = [...data]
const index = newData.findIndex((item) => newRowKeyValue === item.key)
newData.splice(index, 1)
setData(newData)
}
2021-07-30 16:14:07 +05:00
setEditingKey('')
}
const addNewRow = async () => {
let newRow = {
...form.initialValues,
key:newRowKeyValue
}
2021-08-12 17:47:16 +05:00
const newData = [newRow, ...data]
setData(newData)
edit(newRow)
}
const save = async (record) => {
2021-07-30 16:14:07 +05:00
try {
const row = await form.validateFields()
const newData = [...data]
const index = newData.findIndex((item) => record.key === item.key)
const item = newData[index]
const newItem = { ...item, ...row }
2021-07-30 16:14:07 +05:00
newData.splice(index, 1, newItem)
2021-07-30 16:14:07 +05:00
if(item.key === newRowKeyValue)
item.key = newRowKeyValue + newData.length
2021-07-30 16:14:07 +05:00
setEditingKey('')
setData(newData)
if (editingKey === newRowKeyValue)
onRowAdd(newItem)
else
onRowEdit(newItem)
2021-07-30 16:14:07 +05:00
if(onChange)
onChange(newData)
2021-07-30 16:14:07 +05:00
} catch (errInfo) {
console.log('Validate Failed:', errInfo)
}
}
const deleteRow = (record) =>{
const newData = [...data]
const index = newData.findIndex((item) => record.key === item.key)
newData.splice(index, 1)
setData(newData)
onRowDelete(record)
if(onChange)
onChange(newData)
}
2021-07-30 16:14:07 +05:00
const operationColumn = {
width: 82,
title: (!!onRowAdd) && <Button
onClick={addNewRow}
disabled={editingKey !== ''}
icon={<PlusOutlined/>}
/>,
2021-07-30 16:14:07 +05:00
dataIndex: 'operation',
render: (_, record) => {
const editable = isEditing(record)
return editable
?(<span>
<Button
onClick={() => save(record)}
icon={<SaveOutlined/>}/>
<Button
onClick={cancel}
icon={<CloseCircleOutlined/>}/>
</span>)
:(<span>
{onRowEdit&&<Button
disabled={editingKey !== ''}
onClick={() => edit(record)}
icon={<EditOutlined/>}/>}
{onRowDelete&&
<Popconfirm title="Удалить?" onConfirm={()=>deleteRow(record)}>
<Button icon={<DeleteOutlined/>}/>
</Popconfirm>}
</span>)
2021-07-30 16:14:07 +05:00
},
}
const handleColumn = (col) => {
if (col.children)
col.children = col.children.map(handleColumn)
if (!col.editable)
return col
return {
...col,
onCell: (record) => ({
editing: isEditing(record),
record,
dataIndex: col.dataIndex ?? col.key,
2021-08-19 11:40:26 +05:00
key: col.key ?? col.dataIndex,
2021-07-30 16:14:07 +05:00
input: col.input,
isRequired: col.isRequired,
title: col.title,
dataType: col.dataType,
formItemClass: col.formItemClass,
formItemRules: col.formItemRules,
initialValue: col.initialValue,
2021-07-30 16:14:07 +05:00
}),
}
}
const mergedColumns = [...columns.map(handleColumn), operationColumn]
2021-08-19 11:40:26 +05:00
2021-07-30 16:14:07 +05:00
return (
<Form form={form} component={false}>
<Table
components={{
body: {
cell: EditableCell,
},
}}
columns={mergedColumns}
dataSource={data}
2021-07-30 16:14:07 +05:00
{...otherTableProps}
/>
</Form>
)
}