Using React Beautiful DnD to Implement Drag and Drop

Pieces 🌟 - Oct 5 '22 - - Dev Community

The React front-end application world is an ocean of infinite possibilities for building your application. You can have your users interact with your product in so many ways! However, some recognizable patterns are used in most applications. For example, because drag and drop functionality is so frequently used, we can use React Beautiful DnD to make our applications more user-friendly.

When to Use Drag and Drop in React

These are common use cases of drag and drop functionality:

  • Reordering items in a list
  • Creating a file dropper
  • Creating a Kanban board
  • Moving items between lists

Therefore, you should have a good working knowledge of how to build React drag and drop in your application.

Today, we’ll cover this important skill. We’ll build drag and drop functionality into our application using React Beautiful DnD. We’ll also discuss troubleshooting with this library.

What is React Beautiful DnD?

React Beautiful DnD is a React drag and drop library that provides the tools to build drag and drop functionality into your application. This powerful library provides you flexibility and offers many features.

Why Use React Beautiful DnD?

React Beautiful DnD is only one drag-and-drop library of the many available on the market. However, we’ll use it because it’s simple and easy to use.

React Beautiful DnD is a very lightweight library and is straightforward to implement in your application. It’s also very customizable, so you can adapt it to your needs.

Here are some other libraries that you can use to build drag and drop in React:

  • react-dnd: This is very popular, but it’s a bit complex to use.
  • @dnd-kit/core: This is a very lightweight library, but it’s not as customizable as React Beautiful DnD.

What Are We Going to Build?

As a React Beautiful DnD example, we’re going to build a very simple application that’ll have a list of items that we can drag and drop to reorder.

The final product will look something like this.

The final product that we'll build with React Beautiful DnD.

Prerequisites

You’ll need a basic knowledge of React, and that’s it! Let's get started.

Set Up the Application with React Beautiful Dnd

First, create a boilerplate React.JS application:

npx create-react-app drag-n-drop-demo
Enter fullscreen mode Exit fullscreen mode

Then, install the dependencies:

npm install react-beautiful-dnd
Enter fullscreen mode Exit fullscreen mode

1. Create a Simple Task List

First, let’s make a simple list of tasks as a starting point:

const initialTasks = [
  {
    id: 1,
    title: "Task 1",
  },
  {
    id: 2,
    title: "Task 2",
  },
  {
    id: 3,
    title: "Task 3",
  },
];
Enter fullscreen mode Exit fullscreen mode

Save this code

2. Create the Context

The first step in using drag and drop functionality is to declare a particular area on the screen as droppable.

The way we do this is by creating context. We will use this context to wrap the area on the screen where we want drag and drop:

<DragDropContext onDragEnd={onDragEnd}>
  All drag-and-drop functionality happens inside this context
</DragDropContext>
Enter fullscreen mode Exit fullscreen mode

You’ll notice a special function called onDragEnd passed into the context. We’ll use this function to update the state of our application.

For now, let’s use the following code:

const onDragEnd = (result) => {
 console.log(result);
};
Enter fullscreen mode Exit fullscreen mode

Basically, this function will be called after a user finishes a drag and drop operation.

3. Create the Droppable Area

Let's create a React Beautiful DnD component that will be used to wrap the area on the screen that will be droppable:

<Droppable droppableId="tasks">
  {(provided) => (
    <div ref={provided.innerRef} {...provided.droppableProps}>
      { Your draggable items go here }
      {provided. placeholder} // This is a placeholder that will be used to show the space where the item will be dropped
    </div>
  )}
</Droppable>
Enter fullscreen mode Exit fullscreen mode

Save this code

The droppableId is a unique identifier for the droppable area. This is used to identify the area when a drag and drop operation is completed.

The provided object contains two properties:

  • The innerRef refers to the DOM element that wraps the droppable area.
  • The droppableProps is a set of props that will be applied to the DOM element.

4. Create the Draggable Items

Now, let's go over the task items individually and make them draggable:

{
  tasks.map((task, index) => (
    <Draggable key={task.id} draggableId={task.id.toString()} index={index}>
      {(provided) => (
        <div
          {...provided.draggableProps}
          {...provided.dragHandleProps}
          ref={provided.innerRef}
        >
          <div>{task.title}</div>
        </div>
      )}
    </Draggable>
  ));
}
Enter fullscreen mode Exit fullscreen mode

Save this code

The React Draggable component takes in two props. The draggableId is a unique identifier for the draggable item, and the index is the position of the item in the list.

The provided object contains three properties. The draggableProps is a set of props that will be applied to the DOM element. The dragHandleProps is a set of props applied to the DOM element used to drag the item. The innerRef refers to the DOM element that wraps the draggable item.

Now, you should have a working drag-and-drop functionality. However, you’ll notice that the items will return to their original position after updating them.

Let's solve that issue!

5. Update the State with React Beautiful Dnd

The above issue occurs because the application's state is not updated when the drag and drop operation is completed. Let's do that:

const onDragEnd = (result) => {
 if (!result.destination) return;

 const items = Array.from(tasks);
 const [reorderedItem] = items.splice(result.source.index, 1);
  items.splice(result.destination.index, 0, reorderedItem);

  setTasks(items);
};
Enter fullscreen mode Exit fullscreen mode

Save this code

The result object contains the source and destination properties. The source property includes the index and droppableId of the dragged item.

The destination property contains the index and droppableId of the dropped item.

6. Complete Code

import React, { useState } from "react";
import { DragDropContext, Droppable, Draggable } from "react-beautiful-dnd";

const initialTasks = [
  {
    id: 1,
    title: "Task 1",
  },
  {
    id: 2,
    title: "Task 2",
  },
  {
    id: 3,
    title: "Task 3",
  },
];

function App() {
 const [tasks, setTasks] = useState(initialTasks);

 const onDragEnd = (result) => {
 if (!result.destination) return;

 const items = Array.from(tasks);
 const [reorderedItem] = items.splice(result.source.index, 1);
    items.splice(result.destination.index, 0, reorderedItem);

    setTasks(items);
  };

 return (
    <div className="App">
      <DragDropContext onDragEnd={onDragEnd}>
        <Droppable droppableId="tasks">
          {(provided) => (
            <div ref={provided.innerRef} {...provided.droppableProps}>
              {tasks.map((task, index) => (
                <Draggable
                  key={task.id}
                  draggableId={task.id.toString()}
                  index={index}
                >
                  {(provided) => (
                    <div
                      {...provided.draggableProps}
                      {...provided.dragHandleProps}
                      ref={provided.innerRef}
                    >
                      <div>{task.title}</div>
                    </div>
                  )}
                </Draggable>
              ))}
              {provided.placeholder}
            </div>
          )}
        </Droppable>
      </DragDropContext>
    </div>
  );
}

export default App;
Enter fullscreen mode Exit fullscreen mode

Save this code

Common Issues with React Beautiful Dnd Examples

  1. The draggable item is not moving.

Sometimes, you’ll notice that the items are not moving. If you open the console, you’ll see errors like the following:

Unable to find draggable with id: 1

This is a known issue and usually happens due to the use of React.StrictMode on your index.js file.

To fix this issue, you have to remove the React.StrictMode.

Instead of this:

ReactDOM.render(
  <React.StrictMode>
    <App />
  </React.StrictMode>,
  document.getElementById("root")
);
Enter fullscreen mode Exit fullscreen mode

Use this:

ReactDOM.render(<App />, document.getElementById("root"));
Enter fullscreen mode Exit fullscreen mode

This should solve the issue.

  1. The draggable item is not moving smoothly.

If you notice that the draggable item is not moving smoothly, you can try to add the following CSS to your application:

.react-beautiful-dnd-draggable {
  transition: transform 0.2s;
}
Enter fullscreen mode Exit fullscreen mode
  1. The draggable item is not moving to the correct position.

This is usually caused by the draggable item not having a designated height. You can fix this by adding a height value to the draggable object.

References

. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .