react实现动态表单

 更新时间:2022年08月24日 08:36:00   作者:追逐蓦然  
这篇文章主要为大家详细介绍了react实现动态表单,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了react实现动态表单的具体代码,供大家参考,具体内容如下

1.小要求

在工作中,我们也会碰到这样子的需求:在填写信息的时候,可以填写多个人名、多个需求、以及动态生产一个分组。
今天我们就以: 可以动态的添加/删除人名、路径以及可以添加/删除一个分组的需求来开始今天的学习之旅。需求如下图所示:

2.技术点分析

1.数据结构
2.react+antd 动态编辑表格数据提及的知识点
3.js操作数据的方法: 添加数据、根据下标删除数据

3.代码分析

3.1 数据结构分析

/**
*    1. 最外面是一个数组
*    2. 中间是一个字典,每个字典就是一个分组
*    3. name表示人名,也是一个数组,这样子我们就可以动态的添加/删除人名(path路径,跟name同理)
**/
const [data, setData] = useState([
        {
            'name': [''],
            'path': ['']
        }
 ])

3.2添加人名分析

添加路径和添加人名的代码类似,就不重复编写了,大家可以去看完整的代码。这里也“添加人名”举例子:

<Button type="dashed" width={200} onClick={() => {
         // 采用了[...xxx]性质,在对应分组中名字数据中添加一个空的数据
       let obj = [...data]
       setData([])
       obj[index]['name'].push('')
       // 然后在更新数据
       setData(obj);
   }}>+添加人名</Button>

3.3修改人名分析

修改路径和修改人名的代码类似,就不重复编写了,大家可以去看完整的代码。这里也“修改人名”举例子:

<Input style={{ width: 200, marginLeft: 10 }} value={nameItem} onChange={(e) => {
      // 采用了[...xxx]性质,
      let obj = [...data]
      setData([])
      // 修改对应的人名
      obj[index]['name'][nameIndex] = e.target.value
      // 然后在更新数据
      setData(obj)
  }} />

3.4删除人名分析

删除路径和删除人名的代码类似,就不重复编写了,大家可以去看完整的代码。这里也“删除人名”举例子:

<MinusCircleOutlined style={{ marginTop: 10, marginLeft: 5, display: item.name.length == 1 ? 'none' : '' }} onClick={() => {
    // 采用了[...xxx]性质,
    let obj = [...data]
    setData([])
    // 删除人名,使用js的数组用法: 根据下标删除
    obj[index]['name'].splice(nameIndex, 1);
    setData(obj)
}} />

3.5添加分组 

<Button type="dashed" style={{width:'400px',}} onClick={()=>{
       // 采用了[...xxx]性质,
      let obj = [...data]
      setData([])
      // 在原来的数组中,在添加一个对象
      obj.push({
          'name':[''],
          'path':['']
      })
      setData(obj);
   }}>+分组</Button>

3.6删除分组

<MinusCircleOutlined style={{ marginTop: 5, marginLeft: 8, display: data.length == 1 ? 'none' : '' }} onClick={() => {
    let flag = 0
     // 判断名字/路径的输入框中是否有值
     data[index]['name'].map(item=>{
         if(item != ''){
             flag = 1
         }
     })
     if(flag == 0){
         data[index]['path'].map(item=>{
             if(item != ''){
                 flag = 1
             }
         })
     }
     // 如果有值的话,则出现一个弹框提示用户这里还是有值的,是否要删除
     if(flag){
         confirm({
             title: '已经编辑了部分数据,确认要删除',
             icon: <ExclamationCircleOutlined />,
             centered:'true',
             okText:'确认',
             cancelText:'取消',
             onOk() {
                 let obj = [...data]
                 setData([])
                 obj.splice(index, 1);
                 setData(obj)
             },
             onCancel() {},
         });
     }else{
         let obj = [...data]
         setData([])
         obj.splice(index, 1);
         setData(obj)
     }
    
 }} />

4.完整代码

import React, { useState } from 'react';
import { Input, Row, Col, Button, Divider, Modal } from 'antd'
import { MinusCircleOutlined, ExclamationCircleOutlined } from '@ant-design/icons';
const { confirm } = Modal;
function Index() {

    const [data, setData] = useState([
        {
            'name': [''],
            'path': ['']
        }
    ])

    return (
        <div style={{ marginLeft: 50, marginTop: 100 }}>
            <div>
                {
                    data.map((item, index) => {
                        return <div>
                            <div style={{display:'flex'}}>
                                <span>第{index + 1}组</span>
                                <MinusCircleOutlined style={{ marginTop: 5, marginLeft: 8, display: data.length == 1 ? 'none' : '' }} onClick={() => {
                                    let flag = 0
                                    console.log(data[index]['name']);
                                    data[index]['name'].map(item=>{
                                        if(item != ''){
                                            flag = 1
                                            return false
                                        }
                                    })
                                    if(flag == 0){
                                        data[index]['path'].map(item=>{
                                            if(item != ''){
                                                flag = 1
                                                return false
                                            }
                                        })
                                    }

                                    if(flag){
                                        confirm({
                                            title: '已经编辑了部分数据,确认要删除',
                                            icon: <ExclamationCircleOutlined />,
                                            centered:'true',
                                            okText:'确认',
                                            cancelText:'取消',
                                            onOk() {
                                                let obj = [...data]
                                                setData([])
                                                obj.splice(index, 1);
                                                setData(obj)
                                            },
                                            onCancel() {},
                                        });
                                    }else{
                                        let obj = [...data]
                                        setData([])
                                        obj.splice(index, 1);
                                        setData(obj)
                                    }
                                   
                                }} />
                            </div>
                            <Divider />
                            <Row>
                                {
                                    item.name.map((nameItem, nameIndex) => {
                                        return <Col span={8}>
                                            <div style={{ display: 'flex', marginTop: 10 }}>
                                                <span >{'姓名' + (nameIndex + 1) + ':'}</span>
                                                <Input style={{ width: 200, marginLeft: 10 }} value={nameItem} onChange={(e) => {
                                                    let obj = [...data]
                                                    setData([])
                                                    obj[index]['name'][nameIndex] = e.target.value
                                                    setData(obj)
                                                }} />
                                                <MinusCircleOutlined style={{ marginTop: 10, marginLeft: 5, display: item.name.length == 1 ? 'none' : '' }} onClick={() => {
                                                    let obj = [...data]
                                                    setData([])
                                                    obj[index]['name'].splice(nameIndex, 1);
                                                    setData(obj)
                                                }} />
                                            </div>
                                        </Col>
                                    })
                                }
                            </Row>
                            <div style={{ display: 'flex', width: '100vw', justifyContent: 'center', marginTop: 10, marginBottom: 10 }}>
                                <Button type="dashed" width={200} onClick={() => {
                                    let obj = [...data]
                                    setData([])
                                    obj[index]['name'].push('')
                                    setData(obj);
                                }}>+添加人名</Button>
                            </div>
                            <Row>
                                {
                                    item.path.map((pathItem, pathIndex) => {
                                        return <Col span={8}>
                                            <div style={{ display: 'flex', marginTop: 10 }}>
                                                <span >{'路径' + (pathIndex + 1) + ':'}</span>
                                                <Input style={{ width: 200, marginLeft: 10 }} value={pathItem} onChange={(e) => {
                                                    let obj = [...data]
                                                    setData([])
                                                    obj[index]['path'][pathIndex] = e.target.value
                                                    setData(obj)
                                                }} />
                                                <MinusCircleOutlined style={{ marginTop: 10, marginLeft: 5, display: item.path.length == 1 ? 'none' : '' }} onClick={() => {
                                                    let obj = [...data]
                                                    setData([])
                                                    obj[index]['path'].splice(pathIndex, 1);
                                                    setData(obj)
                                                }} />
                                            </div>
                                        </Col>
                                    })
                                }
                            </Row>
                            <div style={{ display: 'flex', width: '100vw', justifyContent: 'center', marginTop: 10, marginBottom: 10 }}>
                                <Button type="dashed" width={200} onClick={() => {
                                    let obj = [...data]
                                    setData([])
                                    obj[index]['path'].push('')
                                    setData(obj);
                                }}>+添加路径</Button>
                            </div>

                        </div>
                    })
                }
            </div>
            <div style={{display:'flex', width:'100vw', justifyContent:'center', marginTop:10, marginBottom:10}}>
                <Button type="dashed" style={{width:'400px',}} onClick={()=>{
                    let obj = [...data]
                    setData([])
                    obj.push({
                        'name':[''],
                        'path':['']
                    })
                    setData(obj);
                }}>+分组</Button>
            </div>
        </div>
    )

}


export default Index

总结

好了,今天就分享到这里,希望大家在学习了一篇博文之后,可以封装出自己的组件来应对多种多样的需求。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • 详解React中多种组件通信方式的实现

    详解React中多种组件通信方式的实现

    在React中,组件之间的通信是一个非常重要的话题,React提供了几种方式来实现跨组件通信,下面小编将详细讲讲其中几种通信方式,并提供实际的代码示例,需要的可以参考下
    2023-11-11
  • 无废话快速上手React路由开发

    无废话快速上手React路由开发

    本文以简洁为目标,帮助快速上手react-router-dom默认你接触过路由相关的开发,通过实例代码讲解的很详细,对React路由相关知识感兴趣的朋友一起看看吧
    2021-05-05
  • React事件节流效果失效的原因及解决

    React事件节流效果失效的原因及解决

    这篇文章主要介绍了React事件节流效果失效的原因及解决,帮助大家更好的理解和学习使用React框架,感兴趣的朋友可以了解下
    2021-04-04
  • Webpack 4.x搭建react开发环境的方法步骤

    Webpack 4.x搭建react开发环境的方法步骤

    这篇文章主要介绍了Webpack 4.x搭建react开发环境的方法步骤,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-08-08
  • react-router 重新加回跳转拦截功能详解

    react-router 重新加回跳转拦截功能详解

    这篇文章主要为大家介绍了react-router 重新加回跳转拦截功能详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-02-02
  • react ant protable自定义实现搜索下拉框

    react ant protable自定义实现搜索下拉框

    这篇文章主要介绍了react ant protable自定义实现搜索下拉框,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-06-06
  • React Flux与Redux设计及使用原理

    React Flux与Redux设计及使用原理

    这篇文章主要介绍了React Flux与Redux设计及使用,Redux最主要是用作应用状态的管理。简言之,Redux用一个单独的常量状态树(state对象)保存这一整个应用的状态,这个对象不能直接被改变
    2023-03-03
  • React Native:react-native-code-push报错的解决

    React Native:react-native-code-push报错的解决

    这篇文章主要介绍了React Native:react-native-code-push报错的解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-10-10
  • 浅谈React碰到v-if

    浅谈React碰到v-if

    这篇文章主要介绍了浅谈React碰到v-if,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-11-11
  • react使用antd的上传组件实现文件表单一起提交功能(完整代码)

    react使用antd的上传组件实现文件表单一起提交功能(完整代码)

    最近在做一个后台管理项目,涉及到react相关知识,项目需求需要在表单中带附件提交,怎么实现这个功能呢?下面小编给大家带来了react使用antd的上传组件实现文件表单一起提交功能,一起看看吧
    2021-06-06

最新评论