ant design vue动态循环生成表单以及自定义校验规则详解
在ant design vue开发中有时候会利用后台数据循环生成表单,需要我们绑定prop以及自定义校验事件
以下是我用官网提供的方法结合自身项目写出来的总结
一、首先在data里定义表单数据
// 循环生成的人员表单数据 addManForm:{ manObjList:[ { person_info_company_guid:undefined,//所属公司 person_info_team_guid:undefined,//班组 person_info_plan_sum:'',//计划投入数量 person_info_plan_time:undefined,//计划投入时间 person_info_sum:'',//实际投入数量 person_info_time:undefined,//实际投入时间 person_info_remark:'', currentParentGuid:'',//当前选中的行,用于添加的的父级id } ] },
二、在对话框内循环生成表单
<a-modal :title="addManTitle" :visible="manModalVisible" @ok="manModalOk" @cancel="manModalCancel" okText="确定" cancelText="取消" :mask=false :zIndex=1000 :width=850> <a-form-model ref="addManFormRef" :model="addManForm" :label-col="{span:7}" :wrapper-col="{ span: 16 }"> <a-row> </a-row> <div class="manContent"> <div class="manSingle" v-for="(item,index) in addManForm.manObjList" :key="index"> <div class="title"> <div class="iconOperate"> <a-icon type="up" v-if="manArrowState && index==0" @click="manFold"/> <a-icon type="down" v-if="!manArrowState && index==0" @click="manUnfold"/> <i class="iconfont iconicon-test" v-if="!btnVisible && index==0" @click="addManObjList"></i> <i class="iconfont iconshanchu" v-if="!btnVisible && index==0" @click="delManObjList"></i> </div> </div> <div class="manContainer"> <a-row> <a-col :span="12"> <a-form-model-item label="所属公司:" :prop="'manObjList.'+index+'.person_info_company_guid'" :rules="[{ required: true, message: '所属公司不能为空', trigger: 'change'}]"> <a-tree-select v-model="item.person_info_company_guid" :tree-data="companySelectData" style="width:100%" placeholder="请选择所属公司" allow-clear :disabled="modalDisabled" tree-default-expand-all/> </a-form-model-item> </a-col> <a-col :span="12"> <a-form-model-item label="所属班组:" :prop="'manObjList.'+index+'.person_info_team_guid'" :rules="[{ required: true, message: '所属班组不能为空', trigger: 'change'}]"> <a-tree-select v-model="item.person_info_team_guid" :tree-data="teamSelectData" style="width:100%" placeholder="请选择所属班组" allow-clear :disabled="modalDisabled" tree-default-expand-all/> </a-form-model-item> </a-col> </a-row> <a-row> <a-col :span="12"> <a-form-model-item label="计划投入数量:" :prop="'manObjList.'+index+'.person_info_plan_sum'" :rules="[{type: 'number', message: '仅支持输入数字', trigger: 'blur' ,transform(value){ if(value){ // 将输入的值转为数字 var val = Number(value) if(/^(?!0+(?:\.0+)?$)(?:[1-9]\d*|0)(?:\.\d{1,2})?$/.test(val)) return val // 返回false即为校验失败 return false } }}, {required:true,message:'计划投入数量不能为空',trigger: 'blur'}]"> <a-input v-model="item.person_info_plan_sum" :disabled="modalDisabled" placeholder="请输入计划投入数量" /> </a-form-model-item> </a-col> <a-col :span="12"> <a-form-model-item label="计划投入时间:" :prop="'manObjList.'+index+'.person_info_plan_time'" :rules="[{ required: true, message: '计划投入时间不能为空', trigger: 'change'}, {validator:manPlanTime}]"> <a-date-picker v-model="item.person_info_plan_time" format="YYYY-MM-DD" :disabled="modalDisabled" style="width:100%"/> </a-form-model-item> </a-col> </a-row> <a-row> <a-col :span="12"> <a-form-model-item label="实际投入数量:" :prop="'manObjList.'+index+'.person_info_sum'" :rules="[{type: 'number', message: '仅支持输入数字', trigger: 'blur' ,transform(value){ if(value){ // 将输入的值转为数字 var val = Number(value) if(/^(?!0+(?:\.0+)?$)(?:[1-9]\d*|0)(?:\.\d{1,2})?$/.test(val)) return val // 返回false即为校验失败 return false } }}, {required:true,message:'实际投入数量不能为空',trigger: 'blur'} ]"> <a-input v-model="item.person_info_sum" :disabled="modalDisabled" placeholder="请输入实际投入数量"/> </a-form-model-item> </a-col> <a-col :span="12"> <a-form-model-item label="实际投入时间:" :prop="'manObjList.'+index+'.person_info_time'" :rules="[{ required: true, message: '实际投入时间不能为空', trigger: 'change'}, {validator:manActualTime}]" > <a-date-picker v-model="item.person_info_time" placeholder="请选择实际投入时间" style="width:100%" :disabled="modalDisabled"/> </a-form-model-item> </a-col> </a-row> <a-row> <a-col :span="12"> <a-form-model-item label="备注:" :prop="'manObjList.'+index+'.person_info_remark'" :rules="[{max: 200, message: '备注长度不能大于200', trigger: 'blur' }]"> <a-input type="textarea" v-model="item.person_info_remark" placeholder="请输入备注" :disabled="modalDisabled"/> </a-form-model-item> </a-col> </a-row> </div> </div> </div> </a-form-model> </a-modal>
fold和unfold方法控制展开折叠,addManObjList添加一组数据,delManObjList删除一组数据
注:prop要动态绑定,数组+索引+字段名=》manObjList.’+index+’.person_info_remark’
三、添加数据方法
// 添加一个人员信息表单 addManObjList(){ this.addManForm.manObjList.push( { person_info_company_guid:undefined,//所属公司 person_info_team_guid:undefined,//班组 person_info_plan_sum:'',//计划投入数量 person_info_plan_time:undefined,//计划投入时间 person_info_sum:'',//实际投入数量 person_info_time:undefined,//实际投入时间 man_deviation_number:'',//投入数量偏差 man_deviation_time:'',//投入时间偏差 person_info_remark:'', currentParentGuid:'',//当前选中的行,用于添加的的父级id }) },
循环表单中的自定义校验
在a-form-model-item中绑定动态rules
:rules="[ {required: true, message: '计划投入时间不能为空', trigger: 'change'}, {validator:manPlanTime} ]"
// 人员的校验规则,计划时间 manPlanTime(rule,value,callback){ let planTimeArr = this.addManForm.manObjList.filter(item=>{ return item.person_info_plan_time == value }) if(value && planTimeArr[0].person_info_time){ if(this.$moment(value).format('YYYY-MM-DD') > this.$moment(planTimeArr[0].person_info_time).format('YYYY-MM-DD')){ callback(new Error('计划投入时间不得大于实际投入时间')) return false } callback() } callback() },
写在最后
这样就能完成一个循环生成表单且完成自定义表单校验,但还有一个小瑕疵,动态生成的日期选择框绑定的是UTC格式的日期,而不是组件本身绑定的moment对象的日期格式。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
相关文章
vue - vue.config.js中devServer配置方式
今天小编就为大家分享一篇vue - vue.config.js中devServer配置方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧2019-10-10uniapp Vue3中如何解决web/H5网页浏览器跨域的问题
存在跨域问题的原因是因为浏览器的同源策略,也就是说前端无法直接发起跨域请求,同源策略是一个基础的安全策略,但是这也会给uniapp/Vue开发者在部署时带来一定的麻烦,这篇文章主要介绍了在uniapp Vue3版本中如何解决web/H5网页浏览器跨域的问题,需要的朋友可以参考下2024-06-06
最新评论