React Drag and Drop: The Missing Example

React Drag and Drop: The Missing Example

August 28, 2024 by Dave Gray📖 7 min read

TLDR: Learn React drag and drop with the example missing from most tutorials.

React Drag and Drop

Drag and drop can be a very useful feature in your React application.

In a music app, you may want to allow users to reorder songs in a playlist.

In a workout app, you may want to allow users to reorder exercises in a workout plan.

Or in a simple todo app, you may want to allow users to reorder tasks in a list.

In all of these cases, adding drag and drop functionality to your application is a great way to make it more user-friendly.

dndkit

dndkit is a drag and drop library for React, and I'll be using it in this article.

I provided a tutorial for react-beautiful-dnd in the past, but at the time of this article, dndkit is the prevalent drag and drop library for React.

If you are interested, I also have a drag and drop tutorial for Vanilla JS.

Let's get started with dndkit by installing it with npm:

npm i @dnd-kit/core @dnd-kit/sortable @dnd-kit/utilities

The Missing Example

Most drag and drop code examples show data with an id field and other fields like name or description.

[
        {
            "id": 1,
            "artist": "The Beatles",
            "title": "Hey Jude",
        },
        {
            "id": 2,
            "artist": "Neil Young",
            "title": "My My, Hey Hey",
        },
        {
            "id": 3,
            "artist": "The Rolling Stones",
            "title": "Wild Horses",
        }
]

This makes a nice, simple example, but it ignores how it will likely integrate with a full stack application using a database.

The id field is often also the primary key for a table in the database.

If we made changes to our playlist above, it would be better to save the playlist order with an additional field like sequence or order.

[
        {
            "id": 1,
            "artist": "The Beatles",
            "title": "Hey Jude",
            "sequence": 1,
        },
        {
            "id": 2,
            "artist": "Neil Young",
            "title": "My My, Hey Hey",
            "sequence": 2,
        },
        {
            "id": 3,
            "artist": "The Rolling Stones",
            "title": "Wild Horses",
            "sequence": 3,
        }
]

Now, when we make changes to the playlist sequence, we can save it back to our database without impacting a primary key.

With this in mind, let's set up a drag and drop example with dndkit.

List Component Imports

The example List component will receive data that is shaped liked the example data with the sequence field.


export type Item = {
    id: number,
    artist: string,
    title: string,
    sequence: number
}

type Props = {
    data: Item[]
}

Let's start with the necessary imports:

// List.tsx 
import { useState } from 'react'

import {
    closestCenter, DragEndEvent, DragOverlay, DragStartEvent, PointerSensor, TouchSensor, useSensor,
    useSensors, DndContext
} from '@dnd-kit/core'

import { arrayMove, SortableContext, verticalListSortingStrategy } from '@dnd-kit/sortable'

I'm not going to go into detail describing each import. That would be like republishing the docs for dndkit, but feel free to read the docs and dive deeper on anything you want.

Client-side React vs Next.js

Before I share the rest of the component code, I need to highlight one important difference between client-side React and Next.js.

A traditional React app is all client-side code and can use the DndContext I show imported above.

Next.js will do a server-side render of your component first - even if you use the "use client" directive. This will cause a hydration mismatch warning when you drag and drop because you are moving around DOM elements.

If you're using Next.js, you will want to use a dynamic import instead of importing DndContext directly.

Add this to your code:

import dynamic from 'next/dynamic' 
// rest of imports here

const DndContextWithNoSSR = dynamic(() => import('@dnd-kit/core').then(mod => mod.DndContext), { ssr: false })

Now, instead of the DndContext import, you will use DndContextWithNoSSR.

List Component State & Functions

Below is the logic for the List component. It boils down to two state variables and four functions.

For the most part, these examples are basic variations of what you will find in the documentation. However, what I believe is worthy to highlight is how the sequence and id properties interact.

export function List({ data }: Props) {
    const [items, setItems] = useState(data)
    const [activeItem, setActiveItem] = useState<Item | undefined>(undefined)

    // for input methods detection
    const sensors = useSensors(useSensor(PointerSensor), useSensor(TouchSensor))

    const removeItem = (id: number) => {
        const updated = items.filter(item => item.id !== id).map((item, i) => ({ ...item, sequence: i + 1 }))

        setItems(updated)
    }

    // triggered when dragging starts
    const handleDragStart = (event: DragStartEvent) => {
        const { active } = event
        setActiveItem(items?.find(item => item.sequence === active.id))
    }

    const handleDragEnd = (event: DragEndEvent) => {
        const { active, over } = event

        if (!over) return

        const activeItem = items.find(ex => ex.sequence === active.id)
        const overItem = items.find(ex => ex.sequence === over.id)

        if (!activeItem || !overItem) {
            return
        }

        const activeIndex = items.findIndex(ex => ex.sequence === active.id)
        const overIndex = items.findIndex(ex => ex.sequence === over.id)

        if (activeIndex !== overIndex) {
            setItems(prev => {
                const updatedWorkoutData = arrayMove(prev, activeIndex, overIndex).map((ex, i) => ({ ...ex, sequence: i + 1 }))

                return updatedWorkoutData
            })
        }
        setActiveItem(undefined)
    }

    const handleDragCancel = () => {
        setActiveItem(undefined)
    }
// rest of component code here

List Component Return Value

Not too leave anything out, here is the return value of the List component.

Note, this example is using the DndContextWithNoSSR import for Next.js. If you are using client-side React, you can use DndContext instead of DndContextWithNoSSR.

return (
        <div className="flex flex-col gap-2 w-1/2 mx-auto">
            {items?.length ? (
                <DndContextWithNoSSR
                    sensors={sensors}
                    collisionDetection={closestCenter}
                    onDragStart={handleDragStart}
                    onDragEnd={handleDragEnd}
                    onDragCancel={handleDragCancel}
                >
                    <SortableContext
                        items={items.map(item => item.sequence)}
                        strategy={verticalListSortingStrategy}
                    >
                        {items.map(item => (
                            <SortableRow
                                key={item.id}
                                item={item}
                                removeItem={removeItem}
                            />
                        ))}
                    </SortableContext>

                    <DragOverlay adjustScale style={{ transformOrigin: "0 0 " }}>
                        {activeItem ? (
                            <SortableRow
                                item={activeItem}
                                removeItem={removeItem}
                                forceDragging={true}
                            />
                        ) : null}
                    </DragOverlay>
                </DndContextWithNoSSR>
            ) : null}
        </div>
    )
}

SortableRow Component

You've surely noticed there is a SortableRow component in the example above that I did not show with the other imports.

Let's take a look at it and don't forget to import it in your List component after you create it.

This component doesn't stray far from the typical examples I have seen, but again, I want to highlight where the sequence value comes into play.

import { XIcon } from 'lucide-react'

import { useSortable } from '@dnd-kit/sortable'
import { CSS } from '@dnd-kit/utilities'

import { Button } from './ui/button'

import type { Item } from "./List"

type Props = {
    item: Item,
    removeItem: (id: number) => void,
    forceDragging?: boolean,
}

export function SortableRow({ item, removeItem, forceDragging = false }: Props) {

    const {
        attributes,
        isDragging,
        listeners,
        setNodeRef,
        setActivatorNodeRef,
        transform,
        transition
    } = useSortable({
        id: item.sequence,
    })

    const parentStyles = {
        transform: CSS.Transform.toString(transform),
        transition: transition || undefined,
        opacity: isDragging ? "0.4" : "1",
        lineHeight: "4",
    }

    const draggableStyles = {
        cursor: isDragging || forceDragging ? "grabbing" : "grab",
    }

    return (
        <article
            className="flex flex-col w-full gap-2 [&:not(:first-child)]:pt-2" ref={setNodeRef}
            style={parentStyles}
        >
            <div className="bg-secondary w-full rounded-md flex items-center gap-2 overflow-hidden">

                <div className="bg-ring w-12 h-full flex items-center">
                    <p className="w-full text-center text-secondary">{item.sequence}</p>
                </div>

                <div
                    ref={setActivatorNodeRef}
                    className="flex-grow p-2"
                    style={draggableStyles}
                    {...attributes} {...listeners}
                >
                    <h2 className="text-lg">
                        {item.title}
                    </h2>
                    <p className="text-sm">{item.artist}</p>
                </div>

                <div className="w-12 h-full flex items-center">
                    <Button
                        type="button"
                        size="icon"
                        variant="outline"
                        onClick={() => removeItem(item.id)}
                    >
                        <XIcon className="text-red-500" />
                    </Button>
                </div>

            </div>

        </article>
    )
}

Where to Go Next

Admittedly, this article is a bit of a code dump. If you are trying to learn more about each detail of dndkit, I recommend you read the docs as they do hold a lot of information.

My goal in sharing this article was to provide what I could not quickly find - an example of using dndkit where you do not change the id value of your data and instead, work with a separate sequence value.

If you were looking for the same thing, I hope this article has helped you.

Related:

← Back to home

Last Updated on August 28, 2024