react-dropzone vs react-dnd vs react-file-drop vs react-dropzone-uploader
React Drag-and-Drop Libraries Comparison
1 Year
react-dropzonereact-dndreact-file-dropreact-dropzone-uploaderSimilar Packages:
What's React Drag-and-Drop Libraries?

These libraries provide various functionalities for implementing drag-and-drop features in React applications. They cater to different use cases, from simple file uploads to complex drag-and-drop interfaces. Choosing the right library depends on the specific requirements of your project, such as the complexity of the drag-and-drop interactions, the need for file uploads, and the desired user experience.

Package Weekly Downloads Trend
Github Stars Ranking
Stat Detail
Package
Downloads
Stars
Size
Issues
Publish
License
react-dropzone4,040,97710,713567 kB597 days agoMIT
react-dnd2,253,64921,265231 kB460-MIT
react-file-drop36,17017620.8 kB3-MIT
react-dropzone-uploader26,176449-1565 years agoMIT
Feature Comparison: react-dropzone vs react-dnd vs react-file-drop vs react-dropzone-uploader

Customization

  • react-dropzone:

    react-dropzone provides basic customization options, such as styling the drop area and handling events, but it is primarily focused on simplicity. It allows for easy integration of file uploads but does not support complex drag-and-drop interactions beyond file selection.

  • react-dnd:

    react-dnd offers extensive customization options, allowing developers to define their own drag-and-drop behaviors and interactions. It provides a powerful API that supports complex use cases, such as nested drag sources and drop targets, making it suitable for applications that require tailored drag-and-drop experiences.

  • react-file-drop:

    react-file-drop offers limited customization options, focusing on ease of use. It allows for basic styling of the drop area but does not support extensive customization for drag-and-drop behaviors.

  • react-dropzone-uploader:

    react-dropzone-uploader allows for moderate customization, including the ability to define upload behavior, styling, and progress indicators. It is designed to provide a more comprehensive file upload experience while still being user-friendly and visually appealing.

File Handling

  • react-dropzone:

    react-dropzone is specifically designed for file uploads and provides built-in support for handling files dropped onto the drop area. It simplifies the process of managing file uploads and integrates easily with various file handling libraries.

  • react-dnd:

    react-dnd does not handle file uploads directly; it is primarily focused on drag-and-drop interactions. Developers need to implement their own file handling logic if they want to use it for file uploads, making it less suitable for applications that require this functionality out of the box.

  • react-file-drop:

    react-file-drop focuses on drag-and-drop file uploads but does not provide advanced file handling features. It allows users to drop files but requires additional implementation for handling the uploaded files.

  • react-dropzone-uploader:

    react-dropzone-uploader enhances file handling capabilities by providing features like file previews, progress tracking, and multiple file uploads. It is well-suited for applications that require a rich file upload experience with visual feedback.

User Experience

  • react-dropzone:

    react-dropzone offers a simple and intuitive user experience for file uploads. Its drag-and-drop area is easy to understand, and it provides visual feedback when files are dragged over the drop zone, making it user-friendly for basic file upload scenarios.

  • react-dnd:

    react-dnd provides a highly interactive user experience, allowing developers to create complex drag-and-drop interfaces that feel natural and responsive. It supports animations and custom drag previews, enhancing the overall user experience in applications with intricate drag-and-drop requirements.

  • react-file-drop:

    react-file-drop provides a straightforward user experience for file uploads but lacks advanced features. It is easy to implement and use, making it suitable for applications that require basic drag-and-drop functionality without additional complexity.

  • react-dropzone-uploader:

    react-dropzone-uploader delivers a polished user experience with features like progress bars and file previews. It is designed to keep users informed about their uploads, making it ideal for applications where users need to upload multiple files and track their progress.

Learning Curve

  • react-dropzone:

    react-dropzone is relatively easy to learn and implement, making it suitable for developers who need a quick solution for file uploads. Its API is straightforward, allowing for rapid integration into projects.

  • react-dnd:

    react-dnd has a steeper learning curve due to its flexibility and extensive API. Developers may need to invest time in understanding its concepts and how to implement complex drag-and-drop interactions effectively.

  • react-file-drop:

    react-file-drop is simple to learn and implement, making it a good choice for developers who want to add basic drag-and-drop functionality without much overhead.

  • react-dropzone-uploader:

    react-dropzone-uploader has a moderate learning curve, as it offers more features than react-dropzone. However, its documentation is comprehensive, making it accessible for developers looking to implement a robust file upload solution.

Community and Support

  • react-dropzone:

    react-dropzone also has a solid community and good documentation, providing developers with the resources needed to implement file uploads effectively. Its popularity ensures that developers can find help and examples easily.

  • react-dnd:

    react-dnd has a strong community and is well-maintained, with extensive documentation and examples available. This support makes it easier for developers to find solutions and best practices for implementing drag-and-drop functionality.

  • react-file-drop:

    react-file-drop has a smaller community compared to the others, which may result in less available support and fewer resources. However, its simplicity may reduce the need for extensive documentation.

  • react-dropzone-uploader:

    react-dropzone-uploader benefits from a growing community and decent documentation, although it may not be as extensive as react-dropzone. Developers can still find support and examples for common use cases.

How to Choose: react-dropzone vs react-dnd vs react-file-drop vs react-dropzone-uploader
  • react-dropzone:

    Select react-dropzone for a straightforward file upload interface with drag-and-drop support. It is best suited for applications where users need to upload files easily and quickly, without complex drag-and-drop interactions.

  • react-dnd:

    Choose react-dnd if you need a highly customizable drag-and-drop solution that supports complex interactions and nesting of draggable items. It is ideal for applications requiring advanced drag-and-drop capabilities, such as kanban boards or sortable lists.

  • react-file-drop:

    Use react-file-drop for a lightweight and easy-to-implement drag-and-drop file upload solution. It is suitable for simpler applications where you want to add drag-and-drop functionality without additional complexity.

  • react-dropzone-uploader:

    Opt for react-dropzone-uploader if you require a more feature-rich file upload component that includes progress tracking, previews, and multiple file uploads. This package is perfect for applications that need a robust file upload experience with visual feedback.

README for react-dropzone

react-dropzone logo

react-dropzone

npm Tests codecov Open Collective Backers Open Collective Sponsors Gitpod Contributor Covenant

Simple React hook to create a HTML5-compliant drag'n'drop zone for files.

Documentation and examples at https://react-dropzone.js.org. Source code at https://github.com/react-dropzone/react-dropzone/.

Installation

Install it from npm and include it in your React build process (using Webpack, Browserify, etc).

npm install --save react-dropzone

or:

yarn add react-dropzone

Usage

You can either use the hook:

import React, {useCallback} from 'react'
import {useDropzone} from 'react-dropzone'

function MyDropzone() {
  const onDrop = useCallback(acceptedFiles => {
    // Do something with the files
  }, [])
  const {getRootProps, getInputProps, isDragActive} = useDropzone({onDrop})

  return (
    <div {...getRootProps()}>
      <input {...getInputProps()} />
      {
        isDragActive ?
          <p>Drop the files here ...</p> :
          <p>Drag 'n' drop some files here, or click to select files</p>
      }
    </div>
  )
}

Or the wrapper component for the hook:

import React from 'react'
import Dropzone from 'react-dropzone'

<Dropzone onDrop={acceptedFiles => console.log(acceptedFiles)}>
  {({getRootProps, getInputProps}) => (
    <section>
      <div {...getRootProps()}>
        <input {...getInputProps()} />
        <p>Drag 'n' drop some files here, or click to select files</p>
      </div>
    </section>
  )}
</Dropzone>

If you want to access file contents you have to use the FileReader API:

import React, {useCallback} from 'react'
import {useDropzone} from 'react-dropzone'

function MyDropzone() {
  const onDrop = useCallback((acceptedFiles) => {
    acceptedFiles.forEach((file) => {
      const reader = new FileReader()

      reader.onabort = () => console.log('file reading was aborted')
      reader.onerror = () => console.log('file reading has failed')
      reader.onload = () => {
      // Do whatever you want with the file contents
        const binaryStr = reader.result
        console.log(binaryStr)
      }
      reader.readAsArrayBuffer(file)
    })
    
  }, [])
  const {getRootProps, getInputProps} = useDropzone({onDrop})

  return (
    <div {...getRootProps()}>
      <input {...getInputProps()} />
      <p>Drag 'n' drop some files here, or click to select files</p>
    </div>
  )
}

Dropzone Props Getters

The dropzone property getters are just two functions that return objects with properties which you need to use to create the drag 'n' drop zone. The root properties can be applied to whatever element you want, whereas the input properties must be applied to an <input>:

import React from 'react'
import {useDropzone} from 'react-dropzone'

function MyDropzone() {
  const {getRootProps, getInputProps} = useDropzone()

  return (
    <div {...getRootProps()}>
      <input {...getInputProps()} />
      <p>Drag 'n' drop some files here, or click to select files</p>
    </div>
  )
}

Note that whatever other props you want to add to the element where the props from getRootProps() are set, you should always pass them through that function rather than applying them on the element itself. This is in order to avoid your props being overridden (or overriding the props returned by getRootProps()):

<div
  {...getRootProps({
    onClick: event => console.log(event),
    role: 'button',
    'aria-label': 'drag and drop area',
    ...
  })}
/>

In the example above, the provided {onClick} handler will be invoked before the internal one, therefore, internal callbacks can be prevented by simply using stopPropagation. See Events for more examples.

Important: if you omit rendering an <input> and/or binding the props from getInputProps(), opening a file dialog will not be possible.

Refs

Both getRootProps and getInputProps accept a custom refKey (defaults to ref) as one of the attributes passed down in the parameter.

This can be useful when the element you're trying to apply the props from either one of those fns does not expose a reference to the element, e.g:

import React from 'react'
import {useDropzone} from 'react-dropzone'
// NOTE: After v4.0.0, styled components exposes a ref using forwardRef,
// therefore, no need for using innerRef as refKey
import styled from 'styled-components'

const StyledDiv = styled.div`
  // Some styling here
`
function Example() {
  const {getRootProps, getInputProps} = useDropzone()
  <StyledDiv {...getRootProps({ refKey: 'innerRef' })}>
    <input {...getInputProps()} />
    <p>Drag 'n' drop some files here, or click to select files</p>
  </StyledDiv>
}

If you're working with Material UI v4 and would like to apply the root props on some component that does not expose a ref, use RootRef:

import React from 'react'
import {useDropzone} from 'react-dropzone'
import RootRef from '@material-ui/core/RootRef'

function PaperDropzone() {
  const {getRootProps, getInputProps} = useDropzone()
  const {ref, ...rootProps} = getRootProps()

  <RootRef rootRef={ref}>
    <Paper {...rootProps}>
      <input {...getInputProps()} />
      <p>Drag 'n' drop some files here, or click to select files</p>
    </Paper>
  </RootRef>
}

IMPORTANT: do not set the ref prop on the elements where getRootProps()/getInputProps() props are set, instead, get the refs from the hook itself:

import React from 'react'
import {useDropzone} from 'react-dropzone'

function Refs() {
  const {
    getRootProps,
    getInputProps,
    rootRef, // Ref to the `<div>`
    inputRef // Ref to the `<input>`
  } = useDropzone()
  <div {...getRootProps()}>
    <input {...getInputProps()} />
    <p>Drag 'n' drop some files here, or click to select files</p>
  </div>
}

If you're using the <Dropzone> component, though, you can set the ref prop on the component itself which will expose the {open} prop that can be used to open the file dialog programmatically:

import React, {createRef} from 'react'
import Dropzone from 'react-dropzone'

const dropzoneRef = createRef()

<Dropzone ref={dropzoneRef}>
  {({getRootProps, getInputProps}) => (
    <div {...getRootProps()}>
      <input {...getInputProps()} />
      <p>Drag 'n' drop some files here, or click to select files</p>
    </div>
  )}
</Dropzone>

dropzoneRef.open()

Testing

react-dropzone makes some of its drag 'n' drop callbacks asynchronous to enable promise based getFilesFromEvent() functions. In order to test components that use this library, you need to use the react-testing-library:

import React from 'react'
import Dropzone from 'react-dropzone'
import {act, fireEvent, render} from '@testing-library/react'

test('invoke onDragEnter when dragenter event occurs', async () => {
  const file = new File([
    JSON.stringify({ping: true})
  ], 'ping.json', { type: 'application/json' })
  const data = mockData([file])
  const onDragEnter = jest.fn()

  const ui = (
    <Dropzone onDragEnter={onDragEnter}>
      {({ getRootProps, getInputProps }) => (
        <div {...getRootProps()}>
          <input {...getInputProps()} />
        </div>
      )}
    </Dropzone>
  )
  const { container } = render(ui)

  await act(
    () => fireEvent.dragEnter(
      container.querySelector('div'),
      data,
    )
  );
  expect(onDragEnter).toHaveBeenCalled()
})

function mockData(files) {
  return {
    dataTransfer: {
      files,
      items: files.map(file => ({
        kind: 'file',
        type: file.type,
        getAsFile: () => file
      })),
      types: ['Files']
    }
  }
}

NOTE: using Enzyme for testing is not supported at the moment, see #2011.

More examples for this can be found in react-dropzone's own test suites.

Caveats

Required React Version

React 16.8 or above is required because we use hooks (the lib itself is a hook).

File Paths

Files returned by the hook or passed as arg to the onDrop cb won't have the properties path or fullPath. For more inf check this SO question and this issue.

Not a File Uploader

This lib is not a file uploader; as such, it does not process files or provide any way to make HTTP requests to some server; if you're looking for that, checkout filepond or uppy.io.

Using <label> as Root

If you use <label> as the root element, the file dialog will be opened twice; see #1107 why. To avoid this, use noClick:

import React, {useCallback} from 'react'
import {useDropzone} from 'react-dropzone'

function MyDropzone() {
  const {getRootProps, getInputProps} = useDropzone({noClick: true})

  return (
    <label {...getRootProps()}>
      <input {...getInputProps()} />
    </label>
  )
}

Using open() on Click

If you bind a click event on an inner element and use open(), it will trigger a click on the root element too, resulting in the file dialog opening twice. To prevent this, use the noClick on the root:

import React, {useCallback} from 'react'
import {useDropzone} from 'react-dropzone'

function MyDropzone() {
  const {getRootProps, getInputProps, open} = useDropzone({noClick: true})

  return (
    <div {...getRootProps()}>
      <input {...getInputProps()} />
      <button type="button" onClick={open}>
        Open
      </button>
    </div>
  )
}

File Dialog Cancel Callback

The onFileDialogCancel() cb is unstable in most browsers, meaning, there's a good chance of it being triggered even though you have selected files.

We rely on using a timeout of 300ms after the window is focused (the window onfocus event is triggered when the file select dialog is closed) to check if any files were selected and trigger onFileDialogCancel if none were selected.

As one can imagine, this doesn't really work if there's a lot of files or large files as by the time we trigger the check, the browser is still processing the files and no onchange events are triggered yet on the input. Check #1031 for more info.

Fortunately, there's the File System Access API, which is currently a working draft and some browsers support it (see browser compatibility), that provides a reliable way to prompt the user for file selection and capture cancellation.

Also keep in mind that the FS access API can only be used in secure contexts.

NOTE You can enable using the FS access API with the useFsAccessApi property: useDropzone({useFsAccessApi: true}).

File System Access API

When setting useFsAccessApi to true, you're switching to the File System API (see the file system access RFC).

What this essentially does is that it will use the showOpenFilePicker method to open the file picker window so that the user can select files.

In contrast, the traditional way (when the useFsAccessApi is not set to true or not specified) uses an <input type="file"> (see docs) on which a click event is triggered.

With the use of the file system access API enabled, there's a couple of caveats to keep in mind:

  1. The users will not be able to select directories
  2. It requires the app to run in a secure context
  3. In Electron, the path may not be set (see #1249)

Supported Browsers

We use browserslist config to state the browser support for this lib, so check it out on browserslist.dev.

Need image editing?

React Dropzone integrates perfectly with Pintura Image Editor, creating a modern image editing experience. Pintura supports crop aspect ratios, resizing, rotating, cropping, annotating, filtering, and much more.

Checkout the Pintura integration example.

Support

Backers

Support us with a monthly donation and help us continue our activities. [Become a backer]

Sponsors

Become a sponsor and get your logo on our README on Github with a link to your site. [Become a sponsor]

Hosting

react-dropzone.js.org hosting provided by netlify.

Contribute

Checkout the organization CONTRIBUTING.md.

License

MIT