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

总结

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

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

(0)

相关推荐

  • React事件处理的机制及原理

    React中的事件处理 在React元素中绑定事件有两点需要注意: (1)在React中,事件命名采用驼峰命名方式,而不是DOM元素中的小写字母命名方式.例如onclick要写成onClick,onchange要写成onChange等. (2)处理事件的响应函数要以对象的形式赋值给事件属性,而不是DOM中的字符串形式.例如在DOM中绑定一个点击事件应该写成: <button onclick="clickButton()"> Click </button> 而在R

  • React实现复杂搜索表单的展开收起功能

    给时间时间,让过去过去. 上节我们写过了[搜索]表单,以及查询.重置功能.本节对于需要展开收起效果的查询表单 进行概述,主要涉及前端样式知识. 样式效果如下: 思路:在Search组件中定义两个组件renderAdvancedForm,renderSimpleForm,其中renderSimpleForm中只有五个查询选项,而在renderAdvancedForm包含所有的搜索选项.点击'展开''收起'按钮调用onClick={toggleForm}切换form显示样式即可. 1. 写rende

  • React学习笔记之事件处理(二)

    之前已经给大家介绍了React中的条件渲染(传送门),本文将给大家关于React中事件处理的相关内容,分享出来供大家参考学习,下面来一起看看详细的介绍: React的事件处理和DOM的事件处理是很相似的,只是有一些语法上的区别: React的事件名是驼峰的,不是小写的 在JSX语法中,你传递一个fucntion作为时间处理器,而不是一个string 举个例子: <button onClick={activateLasers}> Activate Lasers </button> 而

  • react中事件处理与柯里化的实现

    目录 1. 事件处理 阻止默认行为 合成事件 2. 柯里化 柯里化的目的 一个简单的例子 1. 事件处理 React 中元素也可接受.处理事件,但是在语法上有一点不同. 在React 中所有事件的命名采用的是小驼峰,而非原生 DOM 的纯小写,所有事件需要我们传入一个函数,而非字符串. 例如: const Button = () => { const handleClick = () => { console.log('click') } return <button onClick={

  • React的事件处理你了解吗

    目录 一.React的事件处理 1.与DOM事件处理的不同之处 (1)React事件的命名方式:小驼峰方式,DOM的命名方式是小写 (2)事件处理函数是以对象的方式赋值,而不是以字符串的方式赋值 (3)阻止默认事件的方式不同 2.React中事件处理函数的定义 (1)使用ES6的箭头数 (2)在构造函数中进行绑定:将事件处理函数作为类的成员函数 (3)在render函数中绑定this (4)React中事件处理函数 (5)注意事项 3.事件处理中的参数传递 (1)直接传递参数 (2)在定义UI控

  • react 表单数据形式配置化设计

    目录 前言 实现 一般实现 配置化的实现 实现思路 具体实现 使用示例 总结 前言 在日常的中后台系统开发中,表单是和我们打交道非常多的名词.但是在一般的表单实现中.我们会做着很多重复的工作,不停在写 FormItem...,以及为组件加上“请输入/请选择”等无脑的 placeholder 文本和“请输入xx/请选择xx”等必填提示.其次表单一般都存在编辑页和详情页,而为了代码更好的维护性通常会将编辑和详情用一套代码实现.此时我们的代码里就会出现“isEdit ? 表单组件 : 纯文本”这样无脑

  • React表单容器的通用解决方案

    目录 1. 前话 2. 正文 2.1 表单容器定义 2.2 表单容器定义实现 2.3 表单容器呈现实现 2.3.1 弹窗表单容器 2.3.1 抽屉表单容器 2.4 表单容器用例 3. 最后 1. 前话 提问:ToB中台类系统的后端开发主要做啥? ‍♂️:CRUD 再次提问:那前端开发呢? ‍♂️:增删查改 开个玩笑哈啊哈哈哈 虽然没有具体数据统计,但作者仍主观地认为中台类的系统的前端内容至少一半都是增删查改

  • React事件处理和表单的绑定详解

    目录 一.事件处理 1.1 React事件 1.2 事件对象 1.3 事件传参 1.4 函数组件事件处理 二.表单 一.事件处理 1.1 React事件 React 元素的事件处理和 DOM 元素类似.但是有一点语法上的不同: React 事件绑定属性的命名采用驼峰式写法,而不是小写. 例如onclick属性应该写成onClick. 如果采用 JSX的语法你需要传入一个函数作为事件处理函数,而不是一个字符串(DOM 元素的写法).例如已经定义了一个名为handleClick的函数,应该这样调用:

  • react实现动态表单

    本文实例为大家分享了react实现动态表单的具体代码,供大家参考,具体内容如下 1.小要求 在工作中,我们也会碰到这样子的需求:在填写信息的时候,可以填写多个人名.多个需求.以及动态生产一个分组.今天我们就以: 可以动态的添加/删除人名.路径以及可以添加/删除一个分组的需求来开始今天的学习之旅.需求如下图所示: 2.技术点分析 1.数据结构2.react+antd 动态编辑表格数据提及的知识点3.js操作数据的方法: 添加数据.根据下标删除数据 3.代码分析 3.1 数据结构分析 /** *  

  • 简单谈谈Struts动态表单(DynamicForm)

    动态表单的含义是不要手动定义,直接在配置文件中进行定义. 1.手动进行定义 <form-beans > <form-bean name="userForm" type="org.apache.struts.action.DynaActionForm"> <!-- 该表单的属性是配置出来的 --> <form-property name="username" type="java.lang.Str

  • 动态表单验证的操作方法和TP框架里面的ajax表单验证

    动态验证的操作方法 function yz() { if(empty($_POST)) { $this->display(); } else { $db=D("Info"); $shu=array( array("Code","require","代号不能为空",0,"regex",3), ); if(!$db->validate($shu)->create()) { echo $db-&

  • javascript创建动态表单的方法

    本文实例讲述了javascript创建动态表单的方法.分享给大家供大家参考.具体实现方法如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"> <head>

  • jQuery实现动态表单验证时文本框抖动效果完整实例

    本文实例讲述了jQuery实现动态表单验证时文本框抖动效果.分享给大家供大家参考.具体如下: 这里使用jQuery实现的动态表单验证特效,当用户输入错误或没有输入的时候点击提交按钮,有问题的输入框会抖动几下,以提示用户此项有问题,文本框抖动功能都有意思,这个表单比较典型,想实现jquery Ajax表单功能的可以参考. 运行效果截图如下: 在线演示地址如下: http://demo.jb51.net/js/2015/jquery-table-txt-check-shake-style-codes

  • BootStrap 动态表单效果

    html部分 <!-- The template for adding new field --> <div class="form-group hide" id="bookTemplate"> <label class="col-sm-3 control-label">承包商</label> <div class="col-sm-2"> <form:input

  • asp.net 动态表单之数据分页

    但是问题来了,不同科系的同学的科目是不一样的,那么我们在数据库设计的时候通常是把学生.某科成绩作为一条记录,那么这个时候我们就需要做一个行转列的逻辑处理了. 解决方法: 使用GridView来生成表单,这个实现起来会比较麻烦,如果要在列表里面显示链接就更不可能了: 生成html再输出到页面中,这个实现起来比较灵活.方便: 基本功能点: 动态生成表头: 数据进行分页: 查询数据: 对每个成绩进行超链接,查看明细:  页面代码 复制代码 代码如下: <div id="dataDiv1"

  • angular动态表单制作

    源码:https://github.com/Karin001/ngx-dynamic-form 动态表单使用场景 有时候我们需要一个灵活的表单,这个表单可以根据用户的选择,或者服务器返回的信息进行重新配置,比如:增加或删除一组input元素.一组select元素,等等. 在这样的情况下,如果一开始就在模板里写下所有的表单,利用一个ngif树状结构进行选择控制,程序会变得比较冗余. 这时.程序最好是能够根据用户的选择(driven by configuration)或者服务器的响应,自动生成所需要

  • 基于Vue+elementUI实现动态表单的校验功能(根据条件动态切换校验格式)

    前言 开发过程中遇到了一个需求,根据用户选择的联系方式,动态改变输入框的检验条件,并且整个表单是可以增加的 在线访问:动态表单校验 github(欢迎star): https://github.com/Mrblackant. .. 思考几个问题 1.整个表单是可新增的,所以要遍历生成: 2.联系方式(手机/座机)的切换,是要切换后边不同类型输入框还是只改变校验规则(本篇是动态改变校验规则) 实现 1.elementui的form表单实现校验的时候要给当前el-form-item加上prop属性,

  • 详解ng-alain动态表单SF表单项设置必填和正则校验

    在使用动态表单时对表单项进行非空校验及正则校验. 使用手机号进行校验,示例如下: 动态表单的基本使用:https://ng-alain.com/form/getting-started/zh 基于基本示例,增加手机号必填与正则校验的例子: @Component({ selector: 'app-home', template: ` <sf [schema]="schema" [ui]="ui" (formSubmit)="submit($event)

随机推荐