React使用antd实现可编辑单元格

简介: React结合Ant Design实现可编辑单元格的表格组件,通过EditableRow和EditableCell封装实现单元格编辑功能,并提供saveFun回调保存编辑内容。
import React, {
    useContext, useState, useEffect, useRef } from 'react';
import {
    Input, Form, } from 'antd';
import Table from '@com/Table';
import 'antd/dist/antd.css';
import '../styles/search.less';
// 
import '../styles/edittables.less'

/**
 * 只适用于 单个单元格修改
 * 必传字段 :
 *       columns:[],
 *       dataSource:[]
 * 必传方法
 *      saveFun  用于改变input输入值之后触发的回调函数
 */

const EditableContext = React.createContext(null);

const EditableRow = ({
    index, ...props }) => {
   
    const [form] = Form.useForm();
    return (
        <Form form={
   form} component={
   false}>
            <EditableContext.Provider value={
   form}>
                <tr {
   ...props} />
            </EditableContext.Provider>
        </Form>
    );
};

const EditableCell = ({
   
    title,
    editable,
    children,
    dataIndex,
    record,
    saveFun,
    ...restProps
}) => {
   
    const [editing, setEditing] = useState(false);
    const inputRef = useRef(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();
            // 返回当前行信息  包括更改后的信息
            saveFun({
    ...record, ...values })
        } catch (errInfo) {
   
            console.log('Save failed:', errInfo);
        }
    };

    let childNode = children;
    //判断是否可编辑
    if (editable) {
   
        childNode = editing ? (
            <Form.Item
                style={
   {
   
                    margin: 0,
                    lineHeight: "27px"
                }}
                name={
   dataIndex}
                rules={
   [
                    {
   
                        required: true,
                        message: `${
     title} is required.`,
                    },
                ]}
            >
                <Input ref={
   inputRef} style={
   {
    height: "22px", borderRadius: "4px" }} onPressEnter={
   save} onBlur={
   save} />
            </Form.Item>
        ) : (
                <div
                    className="editable-cell-value-wrap"
                    style={
   {
   
                        paddingRight: 24,
                    }}
                    onClick={
   toggleEdit}
                >
                    {
   children}
                </div>
            );
    }

    return <td {
   ...restProps}>{
   childNode}</td>;
};

class EditableTable extends React.Component {
   
    static defaultProps = {
   
        columns: [],
        dataSource: []
    }
    render() {
   
        const {
    dataSource,columns,saveFun, ...rest } = this.props;
        // 每一行 每一个单元格 
        const components = {
   
            body: {
   
                row: EditableRow,
                cell: EditableCell,
            },
        };
        const newColumns = columns.map((col) => {
   
            if (!col.editable) {
   
                return col;
            }
            return {
   
                ...col,
                onCell: (record) => ({
   
                    record,
                    editable: col.editable,
                    dataIndex: col.dataIndex,
                    title: col.title,
                    saveFun: saveFun
                }),
            };
        });
        return (
            <div>
                <Table
                    {
   ...rest}
                    components={
   components}
                    rowClassName={
   () => 'editable-row'}
                    dataSource={
   dataSource}
                    columns={
   newColumns}
                />
            </div>
        );
    }
}

export default EditableTable

使用:

<EditTable
     bordered={
   false}
     size={
   'default'}
     columns={
   columns} //表格的列
     dataSource={
   dataTable} //表格渲染的数据
     title={
   tableName} //表格的名字
     pagination={
   false}
     saveFun={
   this.saveFun}
/>

columns:

 columns: [
                {
   
                    title: () => '序号(可编辑)',
                    dataIndex: 'priority',
                    width: '10%',
                    key: 'priority',
                    editable: true,
                },
                {
   
                    title: () => 'xx',
                    dataIndex: 'generate',
                    key: 'generate'
                },
                {
   
                    title: () => 'xx',
                    dataIndex: 'ttl',
                    key: 'ttl'
                },
                {
   
                    title: () => 'xx',
                    dataIndex: 'type',
                    key: 'type'
                },
                {
   
                    title: () => 'xx',
                    dataIndex: 'value',
                    key: 'value'
                },
                ]

editable属性为true的时候表示当前单元格可编辑;saveFun为该单元格离开焦点或者回车之后出发的回调函数。

saveFun:

saveFun=(item)=>{
   
      // item为当前行 的所有属性
    }

参考antd官方可编辑单元格:https://ant.design/components/table-cn/#components-table-demo-edit-cell

其实上边98%的代码都是固定写法,ctrl C V 即可实现…

目录
相关文章
|
6月前
|
前端开发 JavaScript 测试技术
从零开始搭建react+typescript+antd+redux+less+vw自适应项目
从零开始搭建react+typescript+antd+redux+less+vw自适应项目
178 0
|
6月前
|
前端开发 JavaScript 数据安全/隐私保护
详解React antd中setFieldsValu的简便使用
详解React antd中setFieldsValu的简便使用
238 0
|
2月前
|
前端开发 JavaScript 网络架构
react对antd中Select组件二次封装
本文介绍了如何在React中对Ant Design(antd)的Select组件进行二次封装,包括创建MSelect组件、定义默认属性、渲染Select组件,并展示了如何使用Less进行样式定义和如何在项目中使用封装后的Select组件。
89 2
react对antd中Select组件二次封装
|
2月前
|
前端开发
React给antd中TreeSelect组件左侧加自定义图标icon
本文介绍了如何在React中为Ant Design的TreeSelect组件的每个树节点添加自定义图标,并解决了因缺少key属性而导致的警告问题,展示了如何通过递归函数处理treeData数据并为每个节点添加图标。
113 2
React给antd中TreeSelect组件左侧加自定义图标icon
|
2月前
|
前端开发
react使用antd中的Checkbox实现多选
在React项目中,通过Ant Design的Checkbox组件实现多选。引入Checkbox,使用Checkbox.Group来管理Checkbox,设置`value`属性绑定选中项数组,通过`onChange`更新数组。维护一个全选状态,根据选中项数量与总数决定全选按钮状态。全选按钮的`onChange`事件用于控制所有Checkbox的选中状态。
82 1
react使用antd中的Checkbox实现多选
|
2月前
|
前端开发
React添加路径别名alias、接受props默认值、并二次封装antd中Modal组件与使用
本文介绍了在React项目中如何添加路径别名alias以简化模块引入路径,设置组件props的默认值,以及如何二次封装Ant Design的Modal组件。文章还提供了具体的代码示例,包括配置Webpack的alias、设置defaultProps以及封装Modal组件的步骤和方法。
67 1
React添加路径别名alias、接受props默认值、并二次封装antd中Modal组件与使用
|
2月前
|
前端开发 JavaScript 区块链
react18函数组件+antd使用指南-使用代码集合以及报错记录汇总
本文介绍了多个React开发中常见的问题及其解决方案,包括但不限于:1)`useForm`实例未连接到任何`Form`元素的警告及解决方法;2)监听页面滚动事件的实现方式;3)React 18与antd 5.8.6中定制主题的方法;4)React结合antd 4.x版本自定义主题色的步骤;5)解决`ResizeObserver loop`相关报错的技巧;6)处理React设计表单时遇到的CDN资源加载失败问题;7)解决onClick事件传参问题;8)修复类型错误等。每部分均提供详细分析与实用代码示例,帮助开发者快速定位并解决问题。
45 2
|
2月前
|
前端开发 数据安全/隐私保护
react antd 实现修改密码(原密码,新密码,再次输入新密码,新密码增加正则复杂度校验)
文章介绍了如何在React项目中使用Ant Design实现一个修改密码的组件,包括原密码、新密码和再次输入新密码的表单项,并为新密码增加了正则表达式复杂度校验。
70 0
react antd 实现修改密码(原密码,新密码,再次输入新密码,新密码增加正则复杂度校验)
|
2月前
|
前端开发
React按需加载antd步骤以及出现的问题
在使用`babel-plugin-import`插件时,可以在项目的根目录创建`.babelrc`文件或在`package.json`中添加babel配置。这两个文件中不应该存在重复的配置。如果出现"Multiple configuration files found"错误,需要选择其中一个文件进行配置,并删除另一个文件中的babel配置。使用该插件后,可以直接从`antd`引入组件,无需手动引入样式文件。
63 1
|
2月前
|
前端开发
react antd点击table行时加选中背景色
react antd点击table行时加选中背景色
84 5