Upload Images and Files in React with Preview, Progress Percentage Bar using react-dropzone

In this React tutorial, we’ll learn how to upload single or multiple files by simply dragging and dropping on the uploader zone using the react-dropzone package library. We can display a preview of files and percentage progress bar of files being uploaded.

To make file uploader easy and quick for users, we can add a drop zone where users simply need to select, drag, and drop files that need to be uploaded.

We can display important information meanwhile they are uploading like a preview of files, percentage of upload progress.

The files can be easily converted into an encoded base64 format to upload on the server. Using the react-dropzone we can add a number of configuration and customization to both functionality and style.

Let’s start with the implementation and try it with examples using different configurations available.

[lwptoc]

Create a React Application

First, we’ll create a new React application using npx create-react-app command

$ npx create-react-app react-dropzone-file-uploader-app

Move inside the react app

$ cd react-dropzone-file-uploader-app

Run application

$ npm start

 

Install react-dropzone Package

After creating the React application ready, install the react-dropzone package by running below npm command

$ npm install --save react-dropzone-uploader

Adding React Dropzone Uploader

React Dropzone Uploader is created by importing the Dropzone and also CSS style to apply the basic layout provided by default.

import Dropzone from 'react-dropzone-uploader'
import 'react-dropzone-uploader/dist/styles.css'

The Drop zone is created by adding <Dropzone/> component to create a simple and most basic form of upload container.

<Dropzone
    getUploadParams={getUploadParams}
    onChangeStatus={handleChangeStatus}
    onSubmit={handleSubmit}
    accept="image/*,audio/*,video/*"
/>

Above component is having following method and options properties:

  • getUploadParams:  Define a function triggered when the Submit button is clicked to send payload information with the post URL.
  • onChangeStatus: Return the current status of files being uploaded.
  • onSubmit: Triggered when the submit button is clicked to return uploaded files array.
  • accept: This property is used to limit the type of file that can be uploaded using a Dropzone uploader.

 

Create a new component file ‘~src/components/simple-dropzone.component.js‘ and update it with below code

// components/simple-dropzone.component.js
import React from "react";

import Dropzone from 'react-dropzone-uploader'
import 'react-dropzone-uploader/dist/styles.css'

const SimpleDropZone = () => {

    // Payload data and url to upload files
    const getUploadParams = ({ meta }) => { return { url: 'https://httpbin.org/post' } }

    // Return the current status of files being uploaded
    const handleChangeStatus = ({ meta, file }, status) => { console.log(status, meta, file) }

    // Return array of uploaded files after submit button is clicked
    const handleSubmit = (files, allFiles) => {
        console.log(files.map(f => f.meta))
        allFiles.forEach(f => f.remove())
    }

    return (
        <Dropzone
            getUploadParams={getUploadParams}
            onChangeStatus={handleChangeStatus}
            onSubmit={handleSubmit}
            accept="image/*,audio/*,video/*"
        />
    );
};

export default SimpleDropZone;

 

Render Drop Zone Uploader in App

To render the Dropzone, open the App.js file to import the SimpleDropZone component. Then add inside the return block

import React from 'react';
import './App.css';

import SimpleDropZone from './components/simple-dropzone.component';

function App() {
  return (
    <div className="App">
      <SimpleDropZone />
    </div>
  );
}

export default App;

Now run the React application by executing $ npm start to see it working:

 

 

Show Warning Message for Un-Accepted Files

The inputContent and styles property can be used to show a custom style message to the user if un-accepted files are drop to upload.

 

<Dropzone
            getUploadParams={getUploadParams}
            onChangeStatus={handleChangeStatus}
            onSubmit={handleSubmit}
            accept="image/*,audio/*,video/*"
            
            inputContent={(files, extra) => (extra.reject ? 'Only Image, audio and video files allowed!' : 'Select and Drop Files')}
            styles={{
                dropzoneReject: { borderColor: '#F19373', backgroundColor: '#F1BDAB' },
                inputLabel: (files, extra) => (extra.reject ? { color: '#A02800' } : {}),
            }}
 />

 

 

Validation for Minimum and Maximum Files

The Dropzone can have validation for the minimum and maximum files that can be uploaded by using the maxFiles property.

<Dropzone
      onChangeStatus={handleChangeStatus}
      onSubmit={handleSubmit}
      maxFiles={3}
      inputContent="Drop 3 Files"
      inputWithFilesContent={files => `${3 - files.length} more`}
      submitButtonDisabled={files => files.length < 3}
    />

The Submit button will remain disabled until 3 files are not added.

 

Customize Style of Dropzone

The <Dropzone/> component is a group of sections. These sections include the dropzone, preview section, submit button, and input control. These sections can be modified by defining custom JSX layout to the following properties:

  •   LayoutComponent: Customize the complete layout od Dropzone.
  •   PreviewComponent: Customize the preview section layout.
  •   InputComponent: Customize the choice to select input control.
  •   SubmitButtonComponent: Customize the Submit button layout.

 

Custom Input Control

Let’s see how to customize the style of Choose File button and add custom events needed for it to work.

First, we need to install the html5-file-selector npm package to fetch the array of files selected by input file control, then import it in the component.

$ npm install html5-file-selector

Import the package

import { getDroppedOrSelectedFiles } from 'html5-file-selector'

Next, define the InputChooseFile JSX component to define in the InputComponent property. We also need to add the getFilesFromEvent property to handle the file change event.

The final component will look like this:

// components/simple-dropzone.component.js
import React from "react";

import Dropzone from 'react-dropzone-uploader'
import 'react-dropzone-uploader/dist/styles.css'
import { getDroppedOrSelectedFiles } from 'html5-file-selector'


const SimpleDropZone = () => {

    const getUploadParams = ({ meta }) => {
        console.log(meta);
        return { url: 'https://httpbin.org/post' }
    }

    const handleChangeStatus = ({ meta, file }, status) => { console.log(status, meta, file) }

    const handleSubmit = (files, allFiles) => {
        console.log(files.map(f => f.meta))
        allFiles.forEach(f => f.remove())
    }

    const getFilesFromEvent = e => {
        return new Promise(resolve => {
            getDroppedOrSelectedFiles(e).then(chosenFiles => {
                resolve(chosenFiles.map(f => f.fileObject))
            })
        })
    }

    const InputChooseFile = ({ accept, onFiles, files, getFilesFromEvent }) => {
        const text = files.length > 0 ? 'Add more files' : 'Choose files to upload'

        const buttonStyle = {
            backgroundColor: '#67b0ff',
            color: '#fff',
            cursor: 'pointer',
            padding: 15,
            borderRadius: 30
        }

        return (
            <label style={buttonStyle}>
                {text}
                <input
                    style={{ display: 'none' }}
                    type="file"
                    accept={accept}
                    multiple
                    onChange={e => {
                        getFilesFromEvent(e).then(chosenFiles => {
                            onFiles(chosenFiles)
                        })
                    }}
                />
            </label>
        )
    }

    return (
        <Dropzone
            getUploadParams={getUploadParams}
            onChangeStatus={handleChangeStatus}
            onSubmit={handleSubmit}
            InputComponent={InputChooseFile}
            getFilesFromEvent={getFilesFromEvent}
            classNames
        />
    );
};

export default SimpleDropZone;

The result will look like this:

 

 

 

Properties of Dropzone

  • accept: Type of files that can be uploaded using Dropzone.
  • multiple: Boolean value to enable/disable multiple file upload.
  • minSizeBytes: The minimum size of files in bytes allowed.
  • maxSizeBytes: The maximum size of files in bytes allowed.
  • maxFiles: The number of files uploaded in the Dropzone.
  • autoUpload: Boolean value to control auto-upload event after files are dropped.
  • initialFiles: An array of base64 string to upload initially.

 

Conclusion

The React Dropzone Uploaded provides an awesome solution creating fully-features file uploaded component. There are a number of customization properties available to change the look according to needs. You can check more examples and features on official documentation here.

Leave a Comment

Your email address will not be published. Required fields are marked *