logo

Drag and Drop feature in React

How to Implement Drag n' Drop feature without using any library.

Introduction

In this blog post, I will show you how to implement a simple Drag and Drop feature in a React application without using any library.

The Drag and Drop feature is a common interaction pattern that allows users to move elements around the page. It is a great way to make your application more interactive and user-friendly.

Getting Started

To implement the Drag and Drop feature, we will use the HTML5 Drag and Drop API. This API provides a set of events and methods that allow you to create a custom Drag and Drop behavior.

First, let's create a new React application using Create React App.

npx create-react-app drag-n-drop --template typescript

And let's navigate to the newly created project.

cd drag-n-drop

Now we can start the development server.

npm start

This will start the development server and open the application in your default web browser.

Now that we have our React application set up, let's move on to the next step. But before, let's open our project in our favorite code editor. I personally use Visual Studio Code, but you can use any code editor you prefer.

code .

Next, let's create a new component called Draggable that will represent the draggable element.

We'll create it in the src folder.

import React from 'react'

const Draggable: React.FC = () => {
  return (
    <div
      className="draggable"
      draggable
      onDragStart={(e) => {
        e.dataTransfer.setData('text/plain', 'Hello, World!')
      }}
    >
      Drag me!
    </div>
  )
}

export default Draggable

In this component, we use the draggable attribute to make the element draggable. This attribute will tell the browser that the element can be dragged.

We're also reacting to the onDragStart event. This event is fired by the Drag n Drop API when the user starts dragging the element and can be used to set some properties of the drag operation.

For example, we use the setData method of the dataTransfer object (that is coming directly from the event itself) to set the data that will be transferred when the element is dragged.

This works by setting the data type and the data itself. In this case, we set the data type to text/plain and the data to Hello, World!.

to set the data that will be transferred when the element is dragged.

Also, we're setting the class name of the element to draggable to style it later.

Next, let's create a new component called Droppable that will represent the droppable area.

We'll create it in the src folder.

import React from 'react'

const Droppable: React.FC = () => {
  return (
    <div
      className="droppable"
      onDragOver={(e) => {
        e.preventDefault()
      }}
      onDrop={(e) => {
        e.preventDefault()
        const data = e.dataTransfer.getData('text/plain')
        console.log(data)
      }}
    >
      Drop here!
    </div>
  )
}

export default Droppable

In this component, we're reacting to the onDragOver event.

This event is fired by the Drag n Drop API when the user drags an element over a valid drop target and can be used to specify what happens when the element is dragged over the target.

In this case, we're calling the preventDefault method of the event to prevent the default behavior of the browser, which is to not allow dropping.

This is very important, because by default, the browser does not allow dropping elements on other elements. So we need to call preventDefault in the onDragOver handler to allow the onDrop handler to be fired.

We're also reacting to the onDrop event.

This event is fired by the Drag n Drop API when the user drops an element on a valid drop target and can be used to specify what happens when the element is dropped.

In this case, we're still calling the preventDefault and then we're using the getData method of the dataTransfer object to get the data that was set when the element was dragged.

Also, we're setting the class name of the element to droppable to style it later.

Now that we have our Draggable and Droppable components, let's use them in the App component.

We'll open the App.tsx file and replace its content with the following code.

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

import Draggable from './Draggable'
import Droppable from './Droppable'

const App: React.FC = () => {
  return (
    <div className="container">
      <Draggable />
      <Droppable />
    </div>
  )
}

export default App

In this component, we're using the Draggable and Droppable components that we created earlier. We're also setting the class name of the container to container to style it later.

Now that we have our components set up, let's style them.

We'll open the App.css file and replace its content with the following code.

.container {
  display: flex;
  justify-content: center;
  align-items: center;
  gap: 2rem;
  height: 100vh;
}

.draggable {
  padding: 16px;
  background-color: #f0f0f0;
  cursor: move;
}

.droppable {
  padding: 16px;
  background-color: #f0f0f0;
  cursor: pointer;
}

In this file, we're styling the container, draggable, and droppable elements.

Now that we have our components set up and styled, let's test the Drag and Drop feature.

We can do this by dragging the Draggable element and dropping it on the Droppable element.

When we do this, we should see the data that was set when the element was dragged logged to the console.

If everything is working correctly, you should see Hello, World! logged to the console, like the following:

Demo

The Drag and Drop feature is now working correctly! 🎉 The two components are indeed communicating with each other, and the data is being transferred from the draggable element to the droppable element. Starting from this simple example, you can build more complex interactions and behaviors.

Conclusion

In this blog post, we learned how to implement a simple Drag and Drop feature in a React application without using any library. We used the HTML5 Drag and Drop API to create a custom Drag and Drop behavior and we created a Draggable component and a Droppable component to represent the draggable element and the droppable area.

I hope you found this blog post helpful and that you learned something new. If you have any questions or feedback, feel free to reach out to me on Bluesky. I'm always happy to help!

You can see the final code on my GitHub profile

Reach Out

Have a question or want to work together?

Made with ❤ by Andrea Coluzzi