我正在使用带有react和Express的react-image-crop与Crop上传个人资料图片

时间:2018-12-04 06:40:26

标签: javascript reactjs mongodb express

尝试使用https://www.npmjs.com/package/react-image-crop上传裁剪的图像。

并遵循了https://codesandbox.io/s/72py4jlll6

我可以从src上传基本文件,但请注意裁剪的图像src。谁能告诉我如何获取裁剪后的图像并制作文件并上传。

这是我现在的ProfilePic组件-

          import React, { Component } from "react";
          import { connect } from "react-redux";
          import PropTypes from "prop-types";
          import ReactCrop from "react-image-crop";
          import "react-image-crop/dist/ReactCrop.css";
          import { setProfilePic } from "../../actions/authAction";

          // https://codesandbox.io/s/72py4jlll6
          class ProfilePic extends Component {
            constructor(props) {
              super(props);
              this.state = {
                src: null,
                crop: { x: 10, y: 10, aspect: 1, width: 50 },
                selectedFile: null,
                croppedImageUrl: null
              };

              this.onProfileImageSubmit = this.onProfileImageSubmit.bind(this);
            }
            b64toBlob(b64Data, contentType, sliceSize) {
              contentType = contentType || "";
              sliceSize = sliceSize || 512;

              var byteCharacters = atob(b64Data);
              var byteArrays = [];

              for (var offset = 0; offset < byteCharacters.length; offset += sliceSize) {
                var slice = byteCharacters.slice(offset, offset + sliceSize);

                var byteNumbers = new Array(slice.length);
                for (var i = 0; i < slice.length; i++) {
                  byteNumbers[i] = slice.charCodeAt(i);
                }

                var byteArray = new Uint8Array(byteNumbers);

                byteArrays.push(byteArray);
              }

              var blob = new Blob(byteArrays, { type: contentType });
              return blob;
            }
            onProfileImageSubmit(e) {
              e.preventDefault();

              var ImageURL = this.state.src;
              var block = ImageURL.split(";");
              var contentType = block[0].split(":")[1];
              var realData = block[1].split(",")[1];
              var blob = this.b64toBlob(realData, contentType);

              console.log(blob);

              const data = new FormData();

              data.append("profilePic", blob);
              this.props.setProfilePic(data);
            }
            onSelectFile = e => {
              this.setState({ selectedFile: e.target.files });
              if (e.target.files && e.target.files.length > 0) {
                const reader = new FileReader();
                reader.addEventListener("load", () =>
                  this.setState({ src: reader.result })
                );
                reader.readAsDataURL(e.target.files[0]);
              }
            };

            onImageLoaded = (image, pixelCrop) => {
              this.imageRef = image;

              // Make the library regenerate aspect crops if loading new images.
              const { crop } = this.state;

              if (crop.aspect && crop.height && crop.width) {
                this.setState({
                  crop: { ...crop, height: null }
                });
              } else {
                this.makeClientCrop(crop, pixelCrop);
              }
            };

            onCropComplete = (crop, pixelCrop) => {
              this.makeClientCrop(crop, pixelCrop);
            };

            onCropChange = crop => {
              this.setState({ crop });
            };

            async makeClientCrop(crop, pixelCrop) {
              if (this.imageRef && crop.width && crop.height) {
                const croppedImageUrl = await this.getCroppedImg(
                  this.imageRef,
                  pixelCrop,
                  "newFile.jpeg"
                );

                this.setState({ croppedImageUrl });
              }
            }

            getCroppedImg(image, pixelCrop, fileName) {
              const canvas = document.createElement("canvas");
              canvas.width = pixelCrop.width;
              canvas.height = pixelCrop.height;
              const ctx = canvas.getContext("2d");

              ctx.drawImage(
                image,
                pixelCrop.x,
                pixelCrop.y,
                pixelCrop.width,
                pixelCrop.height,
                0,
                0,
                pixelCrop.width,
                pixelCrop.height
              );

              return new Promise((resolve, reject) => {
                canvas.toBlob(blob => {
                  blob.name = fileName;
                  window.URL.revokeObjectURL(this.fileUrl);
                  this.fileUrl = window.URL.createObjectURL(blob);
                  resolve(this.fileUrl);
                }, "image/jpeg");
              });
            }
            render() {
              const { croppedImageUrl } = this.state;
              return (
                <div>
                  <div
                    className="modal fade"
                    id="update-header-photo"
                    tabindex="-1"
                    role="dialog"
                    aria-labelledby="update-header-photo"
                    aria-hidden="false"
                  >
                    <div
                      className="modal-dialog window-popup update-header-photo"
                      role="document"
                    >
                      <div className="modal-content">
                        <a
                          href="#"
                          className="close icon-close"
                          data-dismiss="modal"
                          aria-label="Close"
                        >
                          <i class="fa fa-user mr-3" aria-hidden="true" />
                        </a>
                        <div className="modal-header">
                          <h6 className="title">Update Header Photo</h6>
                        </div>
                        <div className="modal-body">
                          <a
                            href="#"
                            className="upload-photo-item"
                            style={{ width: "100%" }}
                          >
                            <form onSubmit={this.onProfileImageSubmit}>
                              <div>
                                <input
                                  type="file"
                                  onChange={this.onSelectFile}
                                  name="profilePic"
                                />
                              </div>
                              {this.state.src && (
                                <ReactCrop
                                  src={this.state.src}
                                  crop={this.state.crop}
                                  onImageLoaded={this.onImageLoaded}
                                  onComplete={this.onCropComplete}
                                  onChange={this.onCropChange}
                                />
                              )}
                              {croppedImageUrl && (
                                <img alt="Crop" src={croppedImageUrl} />
                              )}

                              <h6>Upload Photo</h6>
                              <span>Browse your computer.</span>
                              <input
                                className="btn btn-primary btn-lg full-width"
                                type="submit"
                                value="save"
                              />
                            </form>
                          </a>
                        </div>
                      </div>
                    </div>
                  </div>
                </div>
              );
            }
          }
          export default connect(
            null,
            { setProfilePic }
          )(ProfilePic);

1 个答案:

答案 0 :(得分:0)

getCroppedImg()函数上添加this.setState({blob:blob})

canvas.toBlob(blob => {
    blob.name = fileName;
    window.URL.revokeObjectURL(this.fileUrl);
    this.fileUrl = window.URL.createObjectURL(blob);
    resolve(this.fileUrl);
    this.setState({blob:blob}) //added by SO huy nguyen 
}, "image/jpeg");

并修改onProfileImageSubmit()

onProfileImageSubmit(e) {
  e.preventDefault();           
  const data = new FormData();
  data.append("profilePic", this.state.blob);
  this.props.setProfilePic(data);
}

我们不需要将base64转换为Blob。