ReactJS:如何使用 Formik 处理图像/文件上传?

ReactJS: How to handle Image / File upload with Formik?

我正在使用 ReactJS 为我的网站设计个人资料页面。 现在我的问题是如何从本地机器上传图像并将其保存到数据库并在配置文件页面中显示它

import React, {Component} from 'react';
import { connect } from 'react-redux';
import { AccountAction } from '../../actions/user/AccountPg1Action';
import { Formik, Form, Field, ErrorMessage } from 'formik';
import * as Yup from 'yup';

class AccountInfo extends Component {
  constructor(props) {
    super(props) 
    this.state = {
      currentStep: 1,
      userAccountData: {
        userid: '',
        useravtar: '',
        attachement_id: '',
   }
  }
 }

handleFileUpload = (event) => {
  this.setState({useravtar: event.currentTarget.files[0]})
};

handleChange = event => {
    const {name, value} = event.target
    this.setState({
      [name]: value
    })    
  }

handleSubmit = event => {
    let that = this;
    const { AccountAction } = that.props;
    event.preventDefault();

    let accountInputs = {
      userid: 49,
      useravtar: that.state.image,
      attachement_id: 478,
}
    that.setState({
      userAccountData: accountInputs,
    })

    AccountAction(accountInputs)
  }
AccountInfoView = () => {
console.log(this.state.useravtar)
    return (
      <section id="account_sec" className="second_form">
      <div className="container">
      <React.Fragment>
        <Formik
          initialValues={‌{
            file: null,
            email: '',
            phone: ''
          }}
          validationSchema={accountInfoSchema}
          render={(values) => {
          return(
        <Form onSubmit={this.handleSubmit}>
        <Step1 
          currentStep={this.state.currentStep} 
          handleChange={this.handleChange}
          file= {this.state.useravtar}
          handleFileUpload={this.handleFileUpload}
          />
          </Form>
        );
      }}
      />
      </React.Fragment>
      )
  }

  render() {    

    return (
      <div>{this.authView()}</div>
    )
  }
}

function Step1(props) {
console.log(props.useravtar)
  if (props.currentStep !== 1) {
    return null
  } 

  return(
    <div className="upload">
        <label htmlFor="profile">
          <div className="imgbox">
            <img src="images/trans_116X116.png" alt="" />
            <img src={props.useravtar} className="absoImg" alt="" />
          </div>
        </label>
<input id="file" name="file" type="file" accept="image/*" onChange={props.handleFileUpload}/>
        <span className="guide_leb">Add your avatar</span>
      </div>
  )
}

当我在 event.target.file[0] 的 handleChange 操作中执行控制台时,它会以未定义的方式响应。

此外,在 handleSubmit 操作中执行 console.log(this.state.useravtar) 它显示的路径名类似于 c:/fakepath/imgname.jpg

P.S:我有多种形式,所以我以 Step 的方式使用它。我正在使用 Redux Reducer 来存储数据。

我已经提到 this link 但我的要求不是这样的。

Formik默认不支持文件上传,但您可以尝试以下方法

<input id="file" name="file" type="file" onChange={(event) => {
  setFieldValue("file", event.currentTarget.files[0]);
}} />

此处 "file" 代表您用于保存文件的密钥

在提交时,您可以使用

获取文件的文件名、大小等
onSubmit={(values) => {
        console.log({ 
              fileName: values.file.name, 
              type: values.file.type,
              size: `${values.file.size} bytes`
            })

如果你想将文件设置为组件状态,那么你可以使用

onChange={(event) => {
  this.setState({"file": event.currentTarget.files[0]})};
}}

根据您的代码,您必须按如下方式处理文件上传

在AccountInfo中添加一个函数来处理文件上传

handleFileUpload = (event) => {
this.setState({WAHTEVETKEYYOUNEED: event.currentTarget.files[0]})};
}

并将相同的函数传递给 Step1 组件,如下所示

    <Step1 
      currentStep={this.state.currentStep} 
      handleChange={this.handleChange}
      file= {this.state.image}
      handleFileUpload={this.handleFileUpload}
      />

在上传文件的第 1 步组件中,将输入更改为

<input id="file" name="file" type="file" accept="image/*" onChange={props.handleFileUpload}/>

如果您需要预览上传的图像,那么您可以创建一个 blob 并将其作为图像源传递,如下所示

<img src={URL.createObjectURL(FILE_OBJECT)} /> 

EDIT-1

由于安全问题,URL.createObjectURL 方法已被弃用,我们需要对媒体元素使用 srcObject,要使用它,您可以使用 ref 分配 srcObject,例如

假设您正在使用 class 个组件,

构造函数

在构造函数中你可以使用

constructor(props) {
  super(props)
  this.imageElRef = React.createRef(null)
}

手柄更改功能

handleFileUpload = (event) => {
  let reader = new FileReader();
let file = event.target.files[0];
reader.onloadend = () => {
  this.setState({
    file: reader.result
  });
};
reader.readAsDataURL(file);
}

元素

<img src={this.state.file} /> 

您可以使用 Formik 上传带有验证的单个或多个文件,如下所示:

import "./App.css";
import { useEffect, useState } from "react";
import * as Yup from "yup";
import { Formik, Field, Form, ErrorMessage, useField } from "formik";
import axios from "axios";


function App() {
  return (
    <Formik
      initialValues={{
        profile: [],
      }}
      validationSchema={Yup.object({
        profile:Yup.array().min(1,"select at least 1 file")
      })}
      onSubmit={(values, props) => {
        let data = new FormData();
        values.profile.forEach((photo, index) => {
          data.append(`photo${index}`, values.profile[index]);
        });
        axios
          .post("you_api_for_file_upload", data, {
            headers: {
              "Content-Type": "multipart/form-data",
            },
          })
          .then((response) => {
            console.log(response);
          })
          .catch((err) => {
            console.log(err);
          });
      }}
    >
      {(formik) => {
        return (
          <>
            <Form>
              <input
                id="file"
                name="profile"
                type="file"
                onChange={(event) => {
                  const files = event.target.files;
                  let myFiles =Array.from(files);
                  formik.setFieldValue("profile", myFiles);
                }}
                multiple
              />
              <ErrorMessage name="profile"/>
              <button type="submit" disabled={formik.isSubmitting}>
                Submit
              </button>
            </Form>
          </>
        );
      }}
    </Formik>
  );
}

export default App;

注意:您可以根据需要自定义 min(your choice, "your message")

   validationSchema={Yup.object({
        profile:Yup.array().min(1,"select at least 1 file")
      })}

这是我用 FormikMaterial UI

解决问题的方法

在您的 JS 文件中,只需像下面这样声明一个变量 avatarPreview

  const [avatarPreview, setAvatarPreview] = useState('/avatars/default.png');



           <Box
            display='flex'
            textAlign='center'
            justifyContent='center'
            flexDirection='column'>
           
            <ImageAvatar size='md' src={avatarPreview || user?.avatar} />

            <Button
              variant='contained'
              component='label'
              startIcon={<CloudUploadIcon />}>
              Choose Avatar
              <input
                name='avatar'
                accept='image/*'
                id='contained-button-file'
                type='file'
                hidden
                onChange={(e) => {
                  const fileReader = new FileReader();
                  fileReader.onload = () => {
                    if (fileReader.readyState === 2) {
                      setFieldValue('avatar', fileReader.result);
                      setAvatarPreview(fileReader.result);
                    }
                  };
                  fileReader.readAsDataURL(e.target.files[0]);
                }}
              />
            </Button>
          </Box>

默认预览:

选择头像后:

要使用后端处理 formik 中的文件,您只需添加下面给出的输入(您可以将 avatar 更改为您想要的任何内容):

<input
 type="file"
 name="avatar"
 onChange={(event) => {
 setFieldValue('avatar', event.currentTarget.files[0]);
 }}
 />

onSubmit 您可以使用值 obj 访问文件,例如 values.avatar.

上server-side(express)访问我们使用的文件req.file 您还必须使用 multer 来自动检测文件以在 server-side

上处理它

要使用后端处理 formik 中的文件,您只需添加下面给出的输入(您可以将 avatar 更改为您想要的任何内容):

<input
 type="file"
 name="avatar"
 onChange={(event) => {
 setFieldValue('avatar', event.currentTarget.files[0]);
 }}
 />

onSubmit 您可以使用值 obj 访问文件,例如 values.avatar.

在 server-side (express) 上访问我们使用的文件 req.file 您还必须使用 multer & cloudinary 来检测文件以处理它在 server-side

我使用了简单的技巧

<Field name="image">
{ (form , meta , value ) =>
const {setFieldValue} = form 
return (  
<input name="image" type="file" onChange={(e) => setFieldValue(e.target.file[0])}
)
}
</Field>

for multiple image
<input name="image" type="file" onChange={(e) => setFieldValue(e.target.files)}
or use loop
for( i=0 ; i < e.target.file; i++){
setFieldValue([...image , e.target.file])
}