您好,登錄后才能下訂單哦!
本篇文章給大家分享的是有關(guān)antd form實(shí)現(xiàn)表單數(shù)據(jù)回顯的方法,小編覺得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。
index頁面有一個(gè)表格,有一個(gè)新增按鈕,點(diǎn)擊新增按鈕或表格編輯彈出表單模塊,如果是編輯,會(huì)回顯對應(yīng)的表格中的數(shù)據(jù)
//index頁面 import React from 'react' import { Table, Button, message, Input, Select, Modal, } from 'antd'; const Option = Select.Option; import AddOrEdit from './AddOrEdit ' class List extends React.Component { constructor(props) { super(props); } state = { id: "", selectOpt: { getPopupContainer: () => { return this.refs.myForm } }, tableOption: { //表頭 columns: [ { title: '員工姓名', key: 'workerName', dataIndex: 'workerName' }, { title: '員工工號(hào)', key: 'workNumber', dataIndex: 'workNumber' }, { title: "操作", key: 'action', className: 'columnsCenter', render: (text, record) => { return ( <a title="編輯" type="edit" onClick={this.addOrEditClick.bind(this, record)}>編輯</a> ) } } ], dataSource: [], //表格總數(shù)據(jù) loading: false, scroll: { x: 1600 }, //分頁初始數(shù)據(jù) pagination: { current: 1, pageSize: 10, total: 0, showTotal: total => `共 ${total} 條` } }, //編輯表單詳細(xì)信息 dataForm: { data: [], model: { id: { value: undefined }, workerName: { value: undefined },//員工姓名 workNumber: { value: undefined }//員工工號(hào) }, param: {} }, //form表單模塊 modalOption: { title: "新增/修改", visible: false, footer: null, destroyOnClose: true, width: 900 }, } //編輯數(shù)據(jù)回顯 addOrEditClick(record) { const self = this; let { modalOption, dataForm } = this.state; dataForm.param = JSON.parse(JSON.stringify(dataForm.model)); //如果是編輯彈出表單并且數(shù)據(jù)回顯,否則就是新增 if (record.id) { api.get(APIS.yluAssessTarget + record.id).then(function (response) { const data = response.data.data; dataForm.param.id.value = data.id; //給 dataForm.param.workerName.value = data.workerName; dataForm.param.workNumber.value = data.workNumber; modalOption.visible = true; self.setState({ modalOption, dataForm }); }); } else { modalOption.visible = true; self.setState({ modalOption }); } } //分頁 onTableChange(pagination, filters, sorte) { if (pagination.current) { let { tableOption, searchObj } = this.state; tableOption.pagination.current = pagination.current; tableOption.pagination.pageSize = pagination.pageSize; this.setState({ tableOption }); } this.loadTable(); }; /** * 初始化列表 */ loadTable() { let self = this, { tableOption } = this.state; tableOption.loading = true; self.setState({ tableOption }); api.post(APIS.yluAssessTargetSearch + '?current=' + tableOption.pagination.current + '&pageSize=' + tableOption.pagination.pageSize, { data: { companyName: "查詢參數(shù)"//分公司名 } }).then(function (response) { tableOption.dataSource = response.data.data. tableOption.pagination.total = response.data.data.total; }).finally(() => { tableOption.loading = false; self.setState({ tableOption, notDataLevelGroup, searchObj }); }); } render() { const self = this; let { tableOption, modalOption, dataForm } = this.state; return ( <div> <Button size="small" type="primary" onClick={this.addOrEditClick.bind(this, 0)} >新增</Button> <Table {...tableOption} bordered onChange={this.onTableChange.bind(this)} /> <Modal {...modalOption} > {modalOption.visible ? <AddOrEdit {...dataForm} /> : null} </Modal> </div> ) } componentDidMount() { this.loadTable(); } }
//form表單頁,點(diǎn)擊編輯或新增的時(shí)候會(huì)彈出,并且編輯的時(shí)候下拉框value值就為表格當(dāng)前的這一條數(shù)據(jù)對應(yīng)的值 //對select里面的value值用getFieldDecorator進(jìn)行綁定 import React from 'react' import { Table, Button, Select, Form, message, Row, Col, } from 'antd'; const Option = Select.Option; class AddOrEdit extends React.Component { //提交保存 handleSubmit(e) { e.preventDefault(); const self = this; this.props.form.validateFieldsAndScroll((err, values) => { if (!err) { //提交保存 api.post(APIS.yluAssessTarget, { data: { ...values, } }).then(res => { message.success(res.data.message) }).finally(() => { self.setState({ addOrEditFooterLoading: false }) }) } }); } render() { const { workerNameList, workNumberList} = this.state; const { getFieldDecorator } = this.props.form; const reqMeg = '不能為空'; const renderOption = (arr, code, name) => arr ? arr.map((item, index) => { return (<Option key={index + item[code]} value={typeof (item[code]) === 'number' ? item[code].toString() : item[code]}>{item[name]}</Option>) }) : null return ( <Form styleName="form_box" onSubmit={this.handleSubmit.bind(this)} > <Col {...span24}> <FormItem label="員工姓名" {...formItemLayout} hasFeedback> {getFieldDecorator('workerName', { rules: [{ required: true, message: reqMeg }] })(<Select placeholder="請選擇" > {renderOption(workerNameList, 'workCode', 'name')} </Select>)} </FormItem> </Col> <Col {...span24}> <FormItem label="員工工號(hào)" {...formItemLayout} hasFeedback> {getFieldDecorator('workNumber', { rules: [{ required: true, message: reqMeg }] })(<Select placeholder="請選擇" > {renderOption(workNumberList, 'workNumber', 'name')} </Select>)} </FormItem> </Col> </Form> ) } } export default AddOrEdit;
補(bǔ)充知識(shí):Ant Design Vue 中a-upload組件通過axios實(shí)現(xiàn)文件列表上傳與更新回顯的前后端處理方案
前言
在企業(yè)應(yīng)用的快速開發(fā)中,我們需要盡快的完成一些功能。如果您使用了Ant Design Vue,在進(jìn)行表單的文件上傳相關(guān)功能開發(fā)的時(shí)候,您肯定迫不及待地需要找到一篇包治百病的文章,正是如此,才有了該文的誕生,愿以此文解君憂。
方案設(shè)計(jì)
前端方案設(shè)計(jì)
重寫a-upload的文件上傳方式,使用axios來進(jìn)行上傳
選擇一個(gè)文件后立即進(jìn)行上傳,前端記錄上傳成功后的name和uid,并構(gòu)建一個(gè)File實(shí)例,用于a-upload組件的已上傳文件列表的回顯
提交前讓文件列表轉(zhuǎn)化為要后端要處理的uid列表
后端方案設(shè)計(jì)
提供一個(gè)統(tǒng)一上傳單個(gè)文件的接口,每個(gè)文件選擇后自動(dòng)上傳都將上傳到該接口,并寫入到數(shù)據(jù)庫的file數(shù)據(jù)表中
對表單數(shù)據(jù)新建完成后,對上傳的文件列表進(jìn)行當(dāng)前實(shí)體記錄的綁定
對表單數(shù)據(jù)更新完成后,檢測該實(shí)體記錄的所有文件列表,對沒有在該列表中的uid的文件列表進(jìn)行刪除,然后對該列表中所有的uid文件記錄進(jìn)行當(dāng)前實(shí)體記錄的綁定
新建與更新的一致性處理方案
因?yàn)楦卤韱卧谧x取舊數(shù)據(jù)后,需要與新選擇文件進(jìn)行同樣的格式化處理,這里的處理流程一樣,進(jìn)行回顯的數(shù)據(jù)是一樣的,提交表單也都是提交file表中已經(jīng)存在的uid列表,所以這里的數(shù)據(jù)結(jié)構(gòu)是一致的,處理起來將會(huì)更加簡潔明了。
讓代碼說話
為了讓各位看官老爺們便于理解,直接上代碼,希望能將整件事說明白。
構(gòu)建表單
<a-form :form="form"> <a-form-item label="名稱" > <a-input v-decorator="['name', {rules: [{required: true, message: '請輸入名稱!'}]}]" /> </a-form-item> <a-form-item> <a-upload :multiple="true" :fileList="downloadFiles" :remove="handleDownloadFileRemove" :customRequest="downloadFilesCustomRequest" > <a-button class="upload-btn"> <a-icon type="upload" > 相關(guān)下載 </a-button> </a-upload> </a-form-item> </a-form>
編寫js代碼
請求后端接口的token、header以及baseUrl等我已默認(rèn)您已經(jīng)在axios的統(tǒng)一設(shè)置中已經(jīng)配置好了
為了簡化axios相關(guān)操作,我們將axios進(jìn)行了如下封裝(您也可以按此完全使用axios來直接對數(shù)據(jù)進(jìn)行提交等):
const dibootApi = { get (url, params) { return axios.get(url, { params }) }, upload(url, formData) { return service({ url, method: 'POST', data: formData }) } } export default dibootApi
我們默認(rèn)為demo實(shí)體中需要上傳一些文件列表
export default { name: 'demoForm', data () { title: '新建', // 該表單的功能標(biāo)題 form: this.$form.createForm(this), // 表單數(shù)據(jù)初始化,沒什么好說的 model: {}, // 如果是更新表單,之前的數(shù)據(jù)放到這里,來做數(shù)據(jù)初始化顯示之用 downloadFiles: [] // 已經(jīng)上傳的文件列表 }, methods: { // 初始打開的表單時(shí)的處理(如果是更新表單,這里首先需要讀取出相關(guān)數(shù)據(jù)) async open (id) { if (id === undefined) { // 沒有id數(shù)據(jù)則認(rèn)為是新建 this.model = {} this.afterOpen() } else { // 否則作為更新處理 const res = await dibootApi.get(`/${this.name}/${id}`) if (res.code === 0) { this.model = res.data this.title = '編輯' this.afterOpen(id) } else { this.$notification.error({ message: '獲取數(shù)據(jù)失敗', description: res.msg }) } } }, // 更新表單在讀取數(shù)據(jù)完成后的操作 afterOpen (id) { // 獲取該記錄信息后,回顯文件列表相關(guān)操作 dibootApi.post(`/demo/getFiles/${id}`).then(res => { if (res.code === 0){ if (res.data.downloadFile !== undefined){ res.data.downloadFile.forEach(data => { this.downloadFiles.push(this.fileFormatter(data)) }) } } }) }, // 重寫a-upload的文件上傳處理方式 downloadFilesCustomRequest (data) { this.saveFile(data) }, // 上傳并保存文件 saveFile (data){ const formData = new FormData() formData.append('file', data.file) dibootApi.upload('/demo/upload', formData).then((res) => { if (res.code === 0){ let file = this.fileFormatter(res.data) // 上傳單個(gè)文件后,將該文件會(huì)先到a-upload組件的已上傳文件列表中的操作 this.downloadFiles.push(file) } else { this.$message.error(res.msg) } }) }, // 對上傳成功返回的數(shù)據(jù)進(jìn)行格式化處理,格式化a-upload能顯示在已上傳列表中的格式(這個(gè)格式官方文檔有給出的) fileFormatter(data) { let file = { uid: data.uuid, // 文件唯一標(biāo)識(shí),建議設(shè)置為負(fù)數(shù),防止和內(nèi)部產(chǎn)生的 id 沖突 name: data.name, // 文件名 status: 'done', // 狀態(tài)有:uploading done error removed response: '{"status": "success"}', // 服務(wù)端響應(yīng)內(nèi)容 } return file }, // 沒錯(cuò),刪除某個(gè)已上傳的文件的時(shí)候,就是調(diào)用的這里 handleDownloadFileRemove (file) { const index = this.downloadFiles.indexOf(file) const newFileList = this.downloadFiles.slice() newFileList.splice(index, 1) this.downloadFiles = newFileList }, // 表單校驗(yàn)就靠他了,不過這里面還是可以對需要提交的一些數(shù)據(jù)做些手腳的 validate () { return new Promise((resolve, reject) => { this.form.validateFields((err, fieldsValue) => { if (!err) { // 設(shè)置上傳文件列表 const downloadFiles = this.downloadFiles.map(o => { return o.uid }) const values = { ...fieldsValue, 'downloadFiles': downloadFiles } resolve(values) } else { reject(err) } }) }) }, // 表單提交的相關(guān)操作 async onSubmit () { const values = await this.validate() try { let result = {} if (this.model.id === undefined) { // 新增該記錄 result = await this.add(values) } else { // 更新該記錄 values['id'] = this.model.id result = await this.update(values) } // 執(zhí)行提交成功的一系列后續(xù)操作 this.submitSuccess(result) } catch (e) { // 執(zhí)行提交失敗的一系列后續(xù)操作 this.submitFailed(e) } }, // 新增數(shù)據(jù)的操作 async add (values) { .... }, // 更新數(shù)據(jù)的操作 async update (values) { ... } } }
編寫SpringBoot相關(guān)的接口代碼
DemoController
/*** * 獲取文件信息列表 * @param id * @return * @throws Exception */ @PostMapping("/getFiles/{id}") public JsonResult getFilesMap(@PathVariable("id") Serializable id) throws Exception{ List<File> files = fileService.getEntityList( Wrappers.<File>lambdaQuery() .eq(File::getRelObjType, Demo.class.getSimpleName()) .eq(File::getRelObjId, id) ); return new JsonResult(Status.OK, files); } /*** * 上傳文件 * @param file * @param request * @return * @throws Exception */ @PostMapping("/upload") public JsonResult upload(@RequestParam("file") MultipartFile file) throws Exception { File fileEntity = demoService.uploadFile(file); return new JsonResult(Status.OK, fileEntity, "上傳文件成功"); } /*** * 創(chuàng)建成功后的相關(guān)處理 * @param entity * @return */ @Override protected String afterCreated(BaseEntity entity) throws Exception { DemoDTO demoDTO = (DemoDTO) entity; // 更新文件關(guān)聯(lián)信息 demoService.updateFiles(new ArrayList<String>(){{ addAll(demoDTO.getDownloadFiles()); }}, demoDTO.getId(), true); return null; } /*** * 更新成功后的相關(guān)處理 * @param entity * @return */ @Override protected String afterUpdated(BaseEntity entity) throws Exception { DemoDTO demoDTO = (DemoDTO) entity; // 更新文件關(guān)聯(lián)信息 demoService.updateFiles(new ArrayList<String>(){{ addAll(demoDTO.getDownloadFiles()); }}, demoDTO.getId(), false); return null; }
DemoService
@Override public File uploadFile(MultipartFile file) { if(V.isEmpty(file)){ throw new BusinessException(Status.FAIL_OPERATION, "請上傳圖片"); } String fileName = file.getOriginalFilename(); String ext = fileName.substring(fileName.lastIndexOf(".")+1); String newFileName = S.newUuid() + "." + ext; //TODO: 需要對合法的文件類型進(jìn)行驗(yàn)證 if(FileHelper.isImage(ext)){ throw new BusinessException(Status.FAIL_OPERATION, "請上傳合法的文件類型"); }; // 說明:此處為我們的處理流程,看官們需要根據(jù)自己的需求來對文件進(jìn)行保存及處理(之后我們的File組件開源之后也可以按照此處的處理) String filePath = FileHelper.saveFile(file, newFileName); if(V.isEmpty(filePath)){ throw new BusinessException(Status.FAIL_OPERATION, "圖片上傳失敗"); } File fileEntity = new File(); fileEntity.setRelObjType(Demo.class.getSimpleName()); fileEntity.setFileType(ext); fileEntity.setName(fileName); fileEntity.setPath(filePath); String link = "/file/download/" + D.getYearMonth() + "_" + newFileName; fileEntity.setLink(link); boolean success = fileService.createEntity(fileEntity); if (!success){ throw new BusinessException(Status.FAIL_OPERATION, "上傳文件失敗"); } return fileEntity; } @Override public void updateFiles(List<String> uuids, Long currentId, boolean isCreate) { // 如果不是創(chuàng)建,需要?jiǎng)h除不在列表中的file記錄 if (!isCreate){ fileService.deleteEntities(Wrappers.<File>lambdaQuery().notIn(File::getUuid, uuids)); } // 進(jìn)行相關(guān)更新 boolean success = fileService.updateEntity( Wrappers.<File>lambdaUpdate() .in(File::getUuid, uuids) .set(File::getRelObjType, Demo.class.getSimpleName()) .set(File::getRelObjId, currentId)); if (!success){ throw new BusinessException(Status.FAIL_OPERATION, "更新文件信息失敗"); } }
以上就是antd form實(shí)現(xiàn)表單數(shù)據(jù)回顯的方法,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見到或用到的。希望你能通過這篇文章學(xué)到更多知識(shí)。更多詳情敬請關(guān)注億速云行業(yè)資訊頻道。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。