Tutorial 

Now that you've read the overview, it's the adventure time! Even if you have not, you can skip it for now, because ⅔ of our time we'll be busy identifying and building normal React components, just like in the classic Thinking in React tutorial. Adding the drag and drop support is just the icing on the cake.

In this tutorial, we're going to build a Chess game with React and React DnD. Just kidding! Writing a full-blown Chess game is totally out of scope of this tutorial. What we're going to build is a tiny app with a Chess board and a lonely Knight. The Knight will be draggable according to the Chess rules.

We will use this example to demonstrate the data-driven approach of React DnD. You will learn how to create a drag source and a drop target, wire them together with your components, and change their appearance in response to the drag and drop events.

If you're new to React and know a thing or two about it, but yet have to gain some experience building components, this tutorial can also serve as an introduction to the React mode of thinking and the React workflow. If you are a seasoned React developer and only came here for the drag and drop part, feel free to skip to the third and the final chapter of this tutorial.

Enough talk! It's time to set up a build workflow for our little project. I use Webpack, you might be using Browserify. I don't want to get into that now, so just set up an empty React project in whatever way is most convenient for you. If you're feeling lazy, you are free to clone React Hot Boilerplate and work on top of it. In fact, that's what I'm going to do myself.

In this tutorial, the code examples are available simultaneously in ES5, ES6, and ES7. If you want to follow along using ES6 or ES7, you will need to set up a compilation step using Babel. It's easy to make it work with the tool of your choice so we're going to skip this step, too, and assume you've dealt with it and you are ready to write code now. The boilerplate project I linked to before already includes Babel.

The app we're going to build is available as an example on this website.

Identifying the Components 

We're going to start by creating some React components first, with no thoughts of the drag and drop interaction. Which components is our Lonely Knight app going to be made of? I can think of a few:

  • Knight, our lonely knight piece;
  • Square, a single square on the board;
  • Board, the whole board with 64 squares.

Let's consider their props.

  • Knight probably needs no props. It has a position, but there's no reason for the Knight to know it, because it can be positioned by being placed into a Square as a child.

  • It is tempting to give Square its position via props, but this, again, is not necessary, because the only information it really needs for the rendering is the color. I'm going to make Square white by default, and add a black boolean prop. And of course Square may accept a single child: the chess piece that is currently on it. I chose white as the default background color to match the browser defaults.

  • The Board is tricky. It makes no sense to pass Squares as children to it, because what else could a board contain? Therefore it probably owns the Squares. But then, it also needs to own the Knight because this guy needs to be placed inside one of those Squares. This means that the Board needs to know the knight's current position. In a real Chess game, the Board would accept a data structure describing all the pieces, their colors and positions, but for us, a knightPosition prop will suffice. We will use two-item arrays as coordinates, with [0, 0] referring to the A8 square. Why A8 instead of A1? To match the browser coordinate orientation. I tried it another way and it just messed with my head too much.

Where will the current state live? I really don't want to put it into the Board component. It's a good idea to have as little state in your components as possible, and because the Board will already have some layout logic, I don't want to also burden it with managing the state.

The good news is, it doesn't matter at this point. We're just going to write the components as if the state existed somewhere, and make sure that they render correctly when they receive it via props, and think about managing the state afterwards!

Creating the Components 

I prefer to start bottom-up, because this way I'm always working with something that already exists. If I were to build the Board first, I wouldn't see my results until I'm done with the Square. On the other hand, I can build and see the Square right away without even thinking of the Board. I think that the immediate feedback loop is important (you can tell that by another project I work on).

In fact I'm going to start with the Knight. It doesn't have any props at all, and it's the easiest one to build:

import React, { Component } from 'react'; export default class Knight extends Component { render() { return <span></span>; } }

Yes, ♘ is the Unicode knight! It's gorgeous. We could've made its color a prop, but in our example we're not going to have any black knights, so there is no need for that.

It seems to render fine, but just to be sure, I immediately changed my entry point to test it:

import React from 'react'; import ReactDOM from 'react-dom'; import Knight from './Knight'; ReactDOM.render(<Knight />, document.getElementById('root'));

Screenshot

I'm going to do this every time I work on another component, so that I always have something to render. In a larger app, I would use a component playground like cosmos so I'd never write the components in the dark.

I see my Knight on the screen! Time to go ahead and implement the Square now. Here is my first stab:

import React, { Component } from 'react'; import PropTypes from 'prop-types'; export default class Square extends Component { static propTypes = { black: PropTypes.bool }; render() { const { black } = this.props; const fill = black ? 'black' : 'white'; return <div style={{ backgroundColor: fill }} />; } }

Now I change the entry point code to see how the Knight looks inside a Square:

import React from 'react'; import ReactDOM from 'react-dom'; import Knight from './Knight'; import Square from './Square'; ReactDOM.render( <Square black> <Knight /> </Square>, document.getElementById('root') );

Sadly, the screen is empty. I made a few mistakes:

  • I forgot to give Square any dimensions so it just collapses. I don't want it to have any fixed size, so I'll give it width: '100%' and height: '100%' to fill the container.

  • I forgot to put {this.props.children} inside the div returned by the Square, so it ignores the Knight passed to it.

Even after correcting these two mistakes, I still can't see my Knight when the Square is black. That's because the default page body text color is black, so it is not visible on the black Square. I could have fixed this by giving Knight a color prop, but a much simpler fix is to set a corresponding color style in the same place where I set backgroundColor. This version of Square corrects the mistakes and works equally great with both colors:

import React, { Component } from 'react'; import PropTypes from 'prop-types'; export default class Square extends Component { static propTypes = { black: PropTypes.bool }; render() { const { black } = this.props; const fill = black ? 'black' : 'white'; const stroke = black ? 'white' : 'black'; return ( <div style={{ backgroundColor: fill, color: stroke, width: '100%', height: '100%' }}> {this.props.children} </div> ); } }

Screenshot

Finally, time to get started with the Board! I'm going to start with an extremely naïve version that just draws the same single square:

import React, { Component } from 'react'; import PropTypes from 'prop-types'; import Square from './Square'; import Knight from './Knight'; export default class Board extends Component { static propTypes = { knightPosition: PropTypes.arrayOf( PropTypes.number.isRequired ).isRequired }; render() { return ( <div> <Square black> <Knight /> </Square> </div> ); } }

My only intention so far is to make it render, so that I can start tweaking it:

import React from 'react'; import ReactDOM from 'react-dom'; import Board from './Board'; ReactDOM.render( <Board knightPosition={[0, 0]} />, document.getElementById('root') );

Indeed, I can see the same single square. I'm now going to add a whole bunch of them! But I don't know where to start. What do I put in render? Some kind of a for loop? A map over some array?

To be honest, I don't want to think about it now. I already know how to render a single square with or without a knight. I also know the knight's position thanks to the knightPosition prop. This means I can write the renderSquare method and not worry about rendering the whole board just yet.

My first attempt at renderSquare looks like this:

renderSquare(x, y) { const black = (x + y) % 2 === 1; const [knightX, knightY] = this.props.knightPosition; const piece = (x === knightX && y === knightY) ? <Knight /> : null; return ( <Square black={black}> {piece} </Square> ); }

I can already give it a whirl by changing render to be

render() { return ( <div style={{ width: '100%', height: '100%' }}> {this.renderSquare(0, 0)} {this.renderSquare(1, 0)} {this.renderSquare(2, 0)} </div> ); }

Screenshot

At this point, I realize that I forgot to give my squares any layout. I'm going to try Flexbox because why not. I added some styles to the root div, and also wrapped the Squares into divs so I could lay them out. Generally it's a good idea to keep components encapsulated and ignorant of how they're being laid out, even if this means adding wrapper divs.

import React, { Component } from 'react'; import PropTypes from 'prop-types'; import Square from './Square'; import Knight from './Knight'; export default class Board extends Component { static propTypes = { knightPosition: PropTypes.arrayOf( PropTypes.number.isRequired ).isRequired }; renderSquare(i) { const x = i % 8; const y = Math.floor(i / 8); const black = (x + y) % 2 === 1; const [knightX, knightY] = this.props.knightPosition; const piece = (x === knightX && y === knightY) ? <Knight /> : null; return ( <div key={i} style={{ width: '12.5%', height: '12.5%' }}> <Square black={black}> {piece} </Square> </div> ); } render() { const squares = []; for (let i = 0; i < 64; i++) { squares.push(this.renderSquare(i)); } return ( <div style={{ width: '100%', height: '100%', display: 'flex', flexWrap: 'wrap' }}> {squares} </div> ); } }

Screenshot

It looks pretty awesome! I don't know how to constrain the Board to maintain a square aspect ratio, but this should be easy to add later.

Think about it for a moment. We just went from nothing to being able to move the Knight on a beautiful Board by changing the knightPosition:

import React from 'react'; import ReactDOM from 'react-dom'; import Board from './Board'; ReactDOM.render( <Board knightPosition={[7, 4]} />, document.getElementById('root') );

Screenshot

Declarative much? That's why people love working with React.

Adding the State 

We want to make the Knight draggable. It's a noble goal, but we need to see past it. What we really mean is that we want to keep the current knightPosition in some kind of state storage, and have some way to change it.

Because setting up this state requires some thought, we won't try to implement dragging at the same time. Instead, we'll start with a simpler implementation. We will move the Knight when you click a particular Square, but only if this is allowed by the Chess rules. Implementing this logic should give us enough insight into managing the state, so we can replace clicking with the drag and drop once we've dealt with that.

React is not opinionated about the state management or the data flow; you can use Flux, Redux, Rx or even Backbone nah, avoid fat models and separate your reads from writes.

I don't want to bother with installing or setting up Redux for this simple example, so I'm going to follow a simpler pattern. It won't scale as well as Redux, but I also don't need it to. I have not decided on the API for my state manager yet, but I'm going to call it Game, and it will definitely need to have some way of signaling data changes to my React code.

Since I know this much, I can rewrite my index.js with a hypothetical Game that doesn't exist yet. Note that this time, I'm writing my code in blind, not being able to run it yet. This is because I'm still figuring out the API:

import React from 'react'; import ReactDOM from 'react-dom'; import Board from './Board'; import { observe } from './Game'; const rootEl = document.getElementById('root'); observe(knightPosition => ReactDOM.render( <Board knightPosition={knightPosition} />, rootEl ) );

What is this observe function I import? It's just the most minimal way I can think of to subscribe to a changing state. I could've made it an EventEmitter but why on Earth even go there when all I need is a single change event? I could have made Game an object model, but why do that, when all I need is a stream of values?

Just to verify that this subscription API makes some sense, I'm going to write a fake Game that emits random positions:

export function observe(receive) { setInterval(() => receive([ Math.floor(Math.random() * 8), Math.floor(Math.random() * 8) ]), 500); }

Nothing feels as good as being back into the rendering game!

Screenshot

This is obviously not very useful. If we want some interactivity, we're going to need a way to modify the Game state from our components. For now, I'm going to keep it simple and expose a moveKnight function that directly modifies the internal state. This is not going to fare well in a moderately complex app where different state storages may be interested in updating their state in response to a single user action, but in our case this will suffice:

let knightPosition = [0, 0]; let observer = null; function emitChange() { observer(knightPosition); } export function observe(o) { if (observer) { throw new Error('Multiple observers not implemented.'); } observer = o; emitChange(); } export function moveKnight(toX, toY) { knightPosition = [toX, toY]; emitChange(); }

Now, let's go back to our components. Our goal at this point is to move the Knight to a Square that was clicked. One way to do that is to call moveKnight from the Square itself. However, this would require us to pass the Square its position. Here is a good rule of thumb:

If a component doesn't need some data for rendering, it doesn't need that data at all.

The Square does not need to know its position to render. Therefore, it's best to avoid coupling it to the moveKnight method at this point. Instead, we are going to add an onClick handler to the div that wraps the Square inside the Board:

import React from 'react'; import PropTypes from 'prop-types'; import Square from './Square'; import Knight from './Knight'; import { moveKnight } from './Game'; /* ... */ renderSquare(i) { const x = i % 8; const y = Math.floor(i / 8); const black = (x + y) % 2 === 1; const [knightX, knightY] = this.props.knightPosition; const piece = (x === knightX && y === knightY) ? <Knight /> : null; return ( <div key={i} style={{ width: '12.5%', height: '12.5%' }} onClick={() => this.handleSquareClick(x, y)}> <Square black={black}> {piece} </Square> </div> ); } handleSquareClick(toX, toY) { moveKnight(toX, toY); }

We could have also added an onClick prop to Square and used it instead, but since we're going to remove the click handler in favor of the drag and drop interface later anyway, why bother.

The last missing piece right now is the Chess rule check. The Knight can't just move to an arbitrary square, it is only allowed to make L-shaped moves. I'm adding a canMoveKnight(toX, toY) function to the Game and changing the initial position to A2 to match the Chess rules:

let knightPosition = [1, 7]; /* ... */ export function canMoveKnight(toX, toY) { const [x, y] = knightPosition; const dx = toX - x; const dy = toY - y; return (Math.abs(dx) === 2 && Math.abs(dy) === 1) || (Math.abs(dx) === 1 && Math.abs(dy) === 2); }

Finally, I'm adding a canMoveKnight check to the handleSquareClick method:

import { canMoveKnight, moveKnight } from './Game'; /* ... */ handleSquareClick(toX, toY) { if (canMoveKnight(toX, toY)) { moveKnight(toX, toY); } }

Screenshot

Working great so far!

Adding the Drag and Drop Interaction 

This is the part that actually prompted me to write this tutorial. We are now going to see how easy React DnD makes it to add some drag and drop interaction to your existing components.

This part assumes you are at least somewhat familiar with the concepts presented in the overview, such as the backends, the collecting functions, the types, the items, the drag sources, and the drop targets. If you didn't understand everything, it's fine, but make sure you at least give it a chance before jumping into the coding process.

We're going to start by installing React DnD and the HTML5 backend for it:

npm install --save react-dnd react-dnd-html5-backend

In the future, you might want to explore alternative third-party backends, such as the touch backend, but this is out of scope of this tutorial.

The first thing we need to set up in our app is the DragDropContext. We need it to specify that we're going to use the HTML5 backend in our app.

Because the Board is the top-level component in our app, I'm going to put the DragDropContext on it:

import React, { Component } from 'react'; import { DragDropContext } from 'react-dnd'; import HTML5Backend from 'react-dnd-html5-backend'; @DragDropContext(HTML5Backend) export default class Board extends Component { /* ... */ }

Next, I'm going to create the constants for the draggable item types. We're only going to have a single item type in our game, a KNIGHT. I'm creating a Constants module that exports it:

export const ItemTypes = { KNIGHT: 'knight' };

The preparation work is done now. Let's make the Knight draggable!

The DragSource higher-order component accepts three parameters: type, spec, and collect. Our type is the constant we just defined, so now we need to write a drag source specification and a collecting function. For the Knight, the drag source specification is going to be ridiculously simple:

const knightSource = { beginDrag(props) { return {}; } };

This is because there is nothing to describe: there is literally a single draggable object in the whole application! If we had a bunch of chess pieces, it might be a good idea to use the props parameter and return something like { pieceId: props.id }. In our case, an empty object will suffice.

Next, we're going to write a collecting function. What props does the Knight need? It will sure need a way to specify the drag source node. It would also be nice to slightly dim the Knight's opacity while it is being dragged. Therefore, it needs to know whether it is currently being dragged.

Here is the collecting function I wrote for it:

function collect(connect, monitor) { return { connectDragSource: connect.dragSource(), isDragging: monitor.isDragging() } }

Let's take a look at the whole Knight component now, including the DragSource call and the updated render function:

import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { ItemTypes } from './Constants'; import { DragSource } from 'react-dnd'; const knightSource = { beginDrag(props) { return {}; } }; function collect(connect, monitor) { return { connectDragSource: connect.dragSource(), isDragging: monitor.isDragging() } } @DragSource(ItemTypes.KNIGHT, knightSource, collect) export default class Knight extends Component { static propTypes = { connectDragSource: PropTypes.func.isRequired, isDragging: PropTypes.bool.isRequired }; render() { const { connectDragSource, isDragging } = this.props; return connectDragSource( <div style={{ opacity: isDragging ? 0.5 : 1, fontSize: 25, fontWeight: 'bold', cursor: 'move' }}></div> ); } }

Screenshot

The Knight is now a drag source, but there are no drop targets to handle the drop yet. We're going to make the Square a drop target now.

This time, we can't avoid passing the position to the Square. After all, how can the Square know where to move the dragged knight if the Square doesn't know its own position? On the other hand, it still feels wrong because the Square as an entity in our application has not changed, and if it used to be simple, why complicate it? When you face this dilemma, it's time to separate the smart and dumb components.

I'm going to introduce a new component called the BoardSquare. It renders the good old Square, but is also aware of its position. In fact, it's encapsulating some of the logic that the renderSquare method inside the Board used to do. React components are often extracted from such render submethods when the time is right.

Here is the BoardSquare I extracted:

import React, { Component } from 'react'; import PropTypes from 'prop-types'; import Square from './Square'; export default class BoardSquare extends Component { static propTypes = { x: PropTypes.number.isRequired, y: PropTypes.number.isRequired }; render() { const { x, y } = this.props; const black = (x + y) % 2 === 1; return ( <Square black={black}> {this.props.children} </Square> ); } }

I also changed the Board to use it:

renderSquare(i) { const x = i % 8; const y = Math.floor(i / 8); return ( <div key={i} style={{ width: '12.5%', height: '12.5%' }}> <BoardSquare x={x} y={y}> {this.renderPiece(x, y)} </BoardSquare> </div> ); } renderPiece(x, y) { const [knightX, knightY] = this.props.knightPosition; if (x === knightX && y === knightY) { return <Knight />; } }

Let's now wrap the BoardSquare with a DropTarget. I'm going to write a drop target specification that only handles the drop event:

const squareTarget = { drop(props, monitor) { moveKnight(props.x, props.y); } };

See? The drop method receives the props of the BoardSquare so it knows where to move the knight when it drops. In a real app, I might also use monitor.getItem() to retrieve the dragged item that the drag source returned from beginDrag, but since we only have a single draggable thing in the whole application, I don't need it.

In my collecting function, I'm going to obtain the function to connect my drop target node, and I'm also going to ask the monitor whether the pointer is currently over the BoardSquare so I can highlight it:

function collect(connect, monitor) { return { connectDropTarget: connect.dropTarget(), isOver: monitor.isOver() }; }

After changing the render function to connect the drop target and show the highlight overlay, here is what BoardSquare came to be:

import React, { Component } from 'react'; import PropTypes from 'prop-types'; import Square from './Square'; import { canMoveKnight, moveKnight } from './Game'; import { ItemTypes } from './Constants'; import { DropTarget } from 'react-dnd'; const squareTarget = { drop(props) { moveKnight(props.x, props.y); } }; function collect(connect, monitor) { return { connectDropTarget: connect.dropTarget(), isOver: monitor.isOver() }; } @DropTarget(ItemTypes.KNIGHT, squareTarget, collect) export default class BoardSquare extends Component { static propTypes = { x: PropTypes.number.isRequired, y: PropTypes.number.isRequired, isOver: PropTypes.bool.isRequired }; render() { const { x, y, connectDropTarget, isOver } = this.props; const black = (x + y) % 2 === 1; return connectDropTarget( <div style={{ position: 'relative', width: '100%', height: '100%' }}> <Square black={black}> {this.props.children} </Square> {isOver && <div style={{ position: 'absolute', top: 0, left: 0, height: '100%', width: '100%', zIndex: 1, opacity: 0.5, backgroundColor: 'yellow', }} /> } </div> ); } }

Screenshot

This is starting to look good! There is just one change left to complete this tutorial. We want to highlight the BoardSquares that represent the valid moves, and only process the drop if it happened over one of those valid BoardSquares.

Thankfully, it is really easy to do with React DnD. I just need to define a canDrop method in my drop target specification:

canDrop(props) { return canMoveKnight(props.x, props.y); }

I'm also adding monitor.canDrop() to my collecting function, as well as some overlay rendering code to the component:

import React, { Component } from 'react'; import PropTypes from 'prop-types'; import Square from './Square'; import { canMoveKnight, moveKnight } from './Game'; import { ItemTypes } from './Constants'; import { DropTarget } from 'react-dnd'; const squareTarget = { canDrop(props) { return canMoveKnight(props.x, props.y); }, drop(props) { moveKnight(props.x, props.y); } }; function collect(connect, monitor) { return { connectDropTarget: connect.dropTarget(), isOver: monitor.isOver(), canDrop: monitor.canDrop() }; } @DropTarget(ItemTypes.KNIGHT, squareTarget, collect) export default class BoardSquare extends Component { static propTypes = { x: PropTypes.number.isRequired, y: PropTypes.number.isRequired, isOver: PropTypes.bool.isRequired, canDrop: PropTypes.bool.isRequired }; renderOverlay(color) { return ( <div style={{ position: 'absolute', top: 0, left: 0, height: '100%', width: '100%', zIndex: 1, opacity: 0.5, backgroundColor: color, }} /> ); } render() { const { x, y, connectDropTarget, isOver, canDrop } = this.props; const black = (x + y) % 2 === 1; return connectDropTarget( <div style={{ position: 'relative', width: '100%', height: '100%' }}> <Square black={black}> {this.props.children} </Square> {isOver && !canDrop && this.renderOverlay('red')} {!isOver && canDrop && this.renderOverlay('yellow')} {isOver && canDrop && this.renderOverlay('green')} </div> ); } }

Screenshot

Final Touches 

This tutorial guided you through creating the React components, making the design decisions about them and the app's data layer, and finally adding the drag and drop interaction. My intention was to show you that React DnD fits great with the philosophy of React, and that you should think about the app architecture first before diving into implementing the complex interactions.

The last thing I want to demonstrate is drag preview customization. Sure, the browser will screenshot the DOM node, but what if we want to show something different?

We are lucky again, because it is easy to do with React DnD. We just need to add a connect.dragPreview() to the collecting function of the Knight:

function collect(connect, monitor) { return { connectDragSource: connect.dragSource(), connectDragPreview: connect.dragPreview(), isDragging: monitor.isDragging() } }

This lets us use connectDragPreview in render method, just like we used connectDragSource, or even in componentDidMount with a custom image:

componentDidMount() { const img = new Image(); img.src = ''; img.onload = () => this.props.connectDragPreview(img); }

Happy dragging and dropping.

Screenshot

Now go and play with it!