• react antd实现upload上传文件前form校验,同时请求带data


    最近的需求,两个下拉框是必填项,点击上传按钮,如果有下拉框没选要有提示,如图

    如果直接使用antd的Upload组件,一点击文件选择的窗口就打开了,哪怕在Button里再加点击事件,也只是(几乎)同时处理两个方法,beforeUpload更是不行,文件都选择完了才执行。

    在网上查找,终于看到这位vue选手的一个思路,两个同位置同样式的按钮按某个state切换:

    AntD框架上传文件前校验信息:选择文件前进行内容校验并提示

    改成react写法,把除了文件以外的payload获取放到beforeUpload里

    1. const [uploadData, setUpLoadData] = useState(null);
    2. const [selectMonth, setSelectMonth] = useState(false);
    3. const [selectaaa, setSelectaaa] = useState(false);
    4. const handleBeforeUpload = file => {
    5. if (!/.xls?$/.test(file.name) && !/.xlsx?$/.test(file.name)) {
    6. message.error('...');
    7. return false;
    8. }
    9. form.validateFields((err, fieldsValue) => {
    10. if (err) return;
    11. const month = moment(fieldsValue.month).format('YYYY-MM');
    12. setUpLoadData({...fieldsValue, month: month});
    13. });
    14. return true;
    15. };

    组件里加onChange监听:

    1. <FormItem label="月份" {...formItemLayout}>
    2. {getFieldDecorator('month', {
    3. rules: [{ required: true, message: '请选择月份' }],
    4. })(
    5. <MonthPicker onChange={value => value ? setSelectMonth(true) : setSelectMonth(false)}/>
    6. )}
    7. FormItem>

    最后上传按钮那边做两个按钮,用最上面定义的两个state来确定展示哪个按钮(validate里调用form.validateFields方法做必填提示):

    1. {selectDepartment && selectMonth ? (
    2. <Upload
    3. action={uploadUrl}
    4. data={uploadData}
    5. showUploadList={false}
    6. onChange={handleUpload}
    7. beforeUpload={handleBeforeUpload}
    8. withCredentials
    9. headers={}
    10. >
    11. <Button type="primary" style={}>
    12. <Icon type='cloud-upload' /> 上传数据
    13. Button>
    14. Upload>
    15. ) : (
    16. <Button type="primary" style={} onClick={()=>validate()}>
    17. <Icon type='cloud-upload' /> 上传数据
    18. Button>
    19. )
    20. }

    就可以做到最开始提到的效果啦。

    又看到一篇文章(https://zhuanlan.zhihu.com/p/590018131?utm_id=0),可以更优雅地监听form,不用再使用onChange方法和useState,改完发现自己项目antd版本太低了用不了,从antd@4.20.0 开始,antd Form 添加了一个新的 API ->Form.useWatch

    1. const [form] = Form.useForm();
    2. const aaa = Form.useWatch('aaa', form);
    3. const month = Form.useWatch('month', form);

    在form item标签里加上name属性:

    <FormItem label="月份" {...formItemLayout} name='month'>

    效果应该是一样的。

    另外发现使用dataform格式post方法上传文件,其他数据参数不需要用new FormData()再append进去了,直接传字典就行。

  • 相关阅读:
    ARM_LIB_HEAP 与 RTT_HEAP的区别
    python 将二维数组的数据保存到csv文件中
    Ubuntu篇——crontab修改编辑器
    MySQL - WITH RECURSIVE AS 递归查询
    一、Flume使用
    pg14安装_rpm方式
    为什么自学或是培训完软件测试后,找不到工作?原因可能是这几种
    Seata分布式事务
    Springboot中认证和授权的实现(使用jwt)
    应用在液晶背光领域中的环境光传感芯片
  • 原文地址:https://blog.csdn.net/six66667/article/details/133928193