101 lines
2.1 KiB
TypeScript
101 lines
2.1 KiB
TypeScript
import { Form, FormInstance, Input, InputRef, Typography } from 'antd';
|
|
import React, { useContext, useEffect, useRef, useState } from 'react';
|
|
|
|
const EditableContext = React.createContext<FormInstance<any> | null>(null);
|
|
const { Text } = Typography;
|
|
|
|
interface EditableRowProps {
|
|
index: number;
|
|
}
|
|
|
|
interface Item {
|
|
key: string;
|
|
name: string;
|
|
age: string;
|
|
address: string;
|
|
}
|
|
|
|
export const EditableRow: React.FC<EditableRowProps> = ({
|
|
index,
|
|
...props
|
|
}) => {
|
|
const [form] = Form.useForm();
|
|
return (
|
|
<Form form={form} component={false}>
|
|
<EditableContext.Provider value={form}>
|
|
<tr {...props} />
|
|
</EditableContext.Provider>
|
|
</Form>
|
|
);
|
|
};
|
|
|
|
interface EditableCellProps {
|
|
title: React.ReactNode;
|
|
editable: boolean;
|
|
children: React.ReactNode;
|
|
dataIndex: keyof Item;
|
|
record: Item;
|
|
handleSave: (record: Item) => void;
|
|
}
|
|
|
|
export const EditableCell: React.FC<EditableCellProps> = ({
|
|
title,
|
|
editable,
|
|
children,
|
|
dataIndex,
|
|
record,
|
|
handleSave,
|
|
...restProps
|
|
}) => {
|
|
const [editing, setEditing] = useState(false);
|
|
const inputRef = useRef<InputRef>(null);
|
|
const form = useContext(EditableContext)!;
|
|
|
|
useEffect(() => {
|
|
if (editing) {
|
|
inputRef.current!.focus();
|
|
}
|
|
}, [editing]);
|
|
|
|
const toggleEdit = () => {
|
|
setEditing(!editing);
|
|
form.setFieldsValue({ [dataIndex]: record[dataIndex] });
|
|
};
|
|
|
|
const save = async () => {
|
|
try {
|
|
const values = await form.validateFields();
|
|
|
|
toggleEdit();
|
|
handleSave({ ...record, ...values });
|
|
} catch (errInfo) {
|
|
console.log('Save failed:', errInfo);
|
|
}
|
|
};
|
|
|
|
let childNode = children;
|
|
|
|
if (editable) {
|
|
childNode = editing ? (
|
|
<Form.Item
|
|
style={{ margin: 0, minWidth: 70 }}
|
|
name={dataIndex}
|
|
rules={[
|
|
{
|
|
required: true,
|
|
message: `${title} is required.`,
|
|
},
|
|
]}
|
|
>
|
|
<Input ref={inputRef} onPressEnter={save} onBlur={save} />
|
|
</Form.Item>
|
|
) : (
|
|
<div onClick={toggleEdit} className="editable-cell-value-wrap">
|
|
<Text>{children}</Text>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
return <td {...restProps}>{childNode}</td>;
|
|
};
|