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
總結
好瞭,今天就分享到這裡,希望大傢在學習瞭一篇博文之後,可以封裝出自己的組件來應對多種多樣的需求。
以上就是本文的全部內容,希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。