Drag and Drop in HTML5

In this post, we will see how to implement drag and drop functionality in browsers using the native drag and drop interface in HTML5.

The idea is simple:

  • Customize an element to make it draggable
  • Allow dropping the draggable element in the droppable area.
html drag drop

Declare two boxes

<div className="flex gap-8">
  <div
    id="box1"
    className="w-[300px] h-[300px] border border-black flex items-center justify-center"
  />
  <div
    id="box2"
    className="w-[300px] h-[300px] border border-black flex items-center justify-center"
  />
</div>

These two boxes will serve as our “drop zones.” The user will be able to drag an item from one box and drop it into the other.

Add a div that we want to make as draggable inside “box1”

  <div
    id="box1"
    className="w-[300px] h-[300px] border border-black flex items-center justify-center"
  />
    <div
      id="draggable1"
      className="w-[50px] h-[50px] bg-red-500 cursor-move"
    />
  </div>

This red square will be the item that can be dragged between the boxes.

Making the Element Draggable

To make an element draggable, we need to add the draggable attribute to it and handle the dragstart event. The dragstart event will trigger when the user starts dragging the item. Here’s how we can implement it:

const handleOnDragStart = (event) => {
  event.dataTransfer.setData("text/plain", event.target.id);
};

In this function, we use event.dataTransfer.setData() to store the ID of the dragged element. This ID will later help us identify which element was dragged and where it needs to be dropped.

Next, update the draggable1 div to make it draggable and add the onDragStart event handler:

<div
  id="draggable1"
  className="w-[50px] h-[50px] bg-red-500 cursor-move"
  draggable="true"
  onDragStart="{handleOnDragStart}"
/>

Dropping the Element

const handleOnDrop = (event) => {
  event.preventDefault();
  const data = event.dataTransfer.getData("text/plain");
  event.target.appendChild(document.getElementById(data));
};

const handleDragOver = (event) => {
  event.preventDefault();
};
  • The handleOnDrop function prevents the default behavior (which is important for allowing the drop), retrieves the ID of the dragged element from event.dataTransfer, and appends the dragged element to the drop target.
  • The handleDragOver function ensures that the dragged element can be dropped by preventing the default behavior.

Finally, apply these event handlers to the boxes:

<div
  id="box1"
  onDrop="{handleOnDrop}"
  onDragOver="{handleDragOver}"
  className="w-[300px] h-[300px] border border-black flex items-center justify-center"
>
  <div
    id="draggable1"
    className="w-[50px] h-[50px] bg-red-500 cursor-move"
    draggable="true"
    onDragStart="{handleOnDragStart}"
  />
</div>

<div
  id="box2"
  onDrop="{handleOnDrop}"
  onDragOver="{handleDragOver}"
  className="w-[300px] h-[300px] border border-black flex items-center justify-center"
/>

You can add visual cues to highlight when the drag operation is in progress. We will reduce the opacity of the component. This can be done by tracking when the drag operation is being performed in a state variable and changing the opacity. This is how your react component should look like

import { useState } from "react";

const SimpleDragDrop = () => {
  const [isDragging, setIsDragging] = useState(false);

  const handleOnDragStart = (event) => {
    setIsDragging(true);
    event.dataTransfer.setData("text/plain", event.target.id);
  };

  const handleOnDrop = (event) => {
    event.preventDefault();
    setIsDragging(false);
    const data = event.dataTransfer.getData("text/plain");
    event.target.appendChild(document.getElementById(data));
  };

  const handleDragOver = (event) => {
    event.preventDefault();
  };

  return (
    <div className="flex gap-8">
      <div
        id="box1"
        onDrop={handleOnDrop}
        onDragOver={handleDragOver}
        className="flex h-[300px] w-[300px] items-center justify-center border border-main"
      >
        <div
          id="draggable1"
          className={`h-[50px] w-[50px] cursor-move bg-red-500 ${isDragging ? "opacity-50" : "opacity-100"}`}
          draggable={true}
          onDragStart={handleOnDragStart}
        />
      </div>
      <div
        id="box2"
        onDrop={handleOnDrop}
        onDragOver={handleDragOver}
        className="flex h-[300px] w-[300px] items-center justify-center border border-main"
      />
    </div>
  );
};

export default SimpleDragDrop;

Demo

This example demonstrates how easy it is to implement drag-and-drop functionality with just a few lines of code. Feel free to expand on this by adding more drag-and-drop targets or customizing the appearance and behavior of the elements further!

Read more