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.
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 Square
s as children to it, because what else could a board contain? Therefore it probably owns the Square
s. But then, it also need to own the Knight
because this guy needs to be placed inside one of those Square
s. 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!
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 the easiest one to build:
var React = require('react');
var Knight = React.createClass({
render: function () {
return <span>♘</span>;
}
});
module.exports = Knight;
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:
var React = require('react');
var Knight = require('./Knight');
React.render(<Knight />, document.getElementById('root'));
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:
var React = require('react');
var PropTypes = React.PropTypes;
var Square = React.createClass({
propTypes = {
black: PropTypes.bool
},
render: function () {
var black = this.props.black;
var fill = black ? 'black' : 'white';
return <div style={{ backgroundColor: fill }} />;
}
});
module.exports = Square;
Now I change the entry point code to see how the Knight
looks inside a Square
:
var React = require('react');
var Knight = require('./Knight');
var Square = require('./Square');
React.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:
var React = require('react');
var PropTypes = React.PropTypes;
var Square = React.createClass({
propTypes: {
black: PropTypes.bool
},
render: function () {
var black = this.props.black;
var fill = black ? 'black' : 'white';
var stroke = black ? 'white' : 'black';
return (
<div style={{
backgroundColor: fill,
color: stroke,
width: '100%',
height: '100%'
}}>
{this.props.children}
</div>
);
}
});
module.exports = Square;
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:
var React = require('react');
var PropTypes = React.PropTypes;
var Square = require('./Square');
var Knight = require('./Knight');
var Board = React.createClass({
propTypes: {
knightPosition: PropTypes.arrayOf(
PropTypes.number.isRequired
).isRequired
},
render: function () {
return (
<div>
<Square black>
<Knight />
</Square>
</div>
);
}
});
module.exports = Board;
My only intention so far is to make it render, so that I can start tweaking it:
var React = require('react');
var Board = require('./Board');
React.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: function (x, y) {
var black = (x + y) % 2 === 1;
var knightX = this.props.knightPosition[0];
var knightY = this.props.knightPosition[1];
var 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: function () {
return (
<div style={{
width: '100%',
height: '100%'
}}>
{this.renderSquare(0, 0)}
{this.renderSquare(1, 0)}
{this.renderSquare(2, 0)}
</div>
);
}
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 Square
s into div
s 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 div
s.
var React = require('react');
var PropTypes = React.PropTypes;
var Square = require('./Square');
var Knight = require('./Knight');
var Board = React.createClass({
propTypes: {
knightPosition: PropTypes.arrayOf(
PropTypes.number.isRequired
).isRequired
},
renderSquare: function (i) {
var x = i % 8;
var y = Math.floor(i / 8);
var black = (x + y) % 2 === 1;
var knightX = this.props.knightPosition[0];
var knightY = this.props.knightPosition[1];
var 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: function () {
var 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>
);
}
});
module.exports = Board;
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
:
var React = require('react');
var Board = require('./Board');
React.render(
<Board knightPosition={[4, 7]} />,
document.getElementById('root')
);
Declarative much? That's why people love working with React.
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, 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 Flux for this simple example, so I'm going to follow a simpler pattern. It won't scale as well as Flux, 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:
var React = require('react');
var Board = require('./Board');
var observe = require('./Game').observe;
var rootEl = document.getElementById('root');
observe(function (knightPosition) {
React.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:
exports.observe = function (receive) {
setInterval(function () {
receive([
Math.floor(Math.random() * 8),
Math.floor(Math.random() * 8)
]);
}, 500);
};
Nothing feels as good as being back into the rendering game!
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:
var knightPosition = [0, 0];
var observer = null;
function emitChange() {
observer(knightPosition);
}
exports.observe = function (o) {
if (observer) {
throw new Error('Multiple observers not implemented.');
}
observer = o;
emitChange();
}
exports.moveKnight = function (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
:
var React = require('react');
var PropTypes = React.PropTypes;
var Square = require('./Square');
var Knight = require('./Knight');
var moveKnight = require('./Game').moveKnight;
/* ... */
renderSquare: function (i) {
var x = i % 8;
var y = Math.floor(i / 8);
var black = (x + y) % 2 === 1;
var knightX = this.props.knightPosition[0];
var knightY = this.props.knightPosition[1];
var piece = (x === knightX && y === knightY) ?
<Knight /> :
null;
return (
<div key={i}
style={{ width: '12.5%', height: '12.5%' }}
onClick={this.handleSquareClick.bind(this, x, y)}>
<Square black={black}>
{piece}
</Square>
</div>
);
},
handleSquareClick: function (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:
var knightPosition = [1, 7];
/* ... */
exports.canMoveKnight = function (toX, toY) {
const x = knightPosition[0];
const y = knightPosition[1];
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:
handleSquareClick: function (toX, toY) {
if (canMoveKnight(toX, toY)) {
moveKnight(toX, toY);
}
}
Working great so far!
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 running npm install --save react-dnd
. 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:
var React = require('react');
var DragDropContext = require('react-dnd').DragDropContext;
var HTML5Backend = require('react-dnd/modules/backends/HTML5');
var Board = React.createClass({
/* ... */
});
module.exports = DragDropContext(HTML5Backend)(Board);
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:
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:
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:
var React = require('react');
var PropTypes = React.PropTypes;
var ItemTypes = require('./Constants').ItemTypes;
var DragSource = require('react-dnd').DragSource;
var knightSource = {
beginDrag: function (props) {
return {};
}
};
function collect(connect, monitor) {
return {
connectDragSource: connect.dragSource(),
isDragging: monitor.isDragging()
}
}
var Knight = React.createClass({
propTypes: {
connectDragSource: PropTypes.func.isRequired,
isDragging: PropTypes.bool.isRequired
},
render: function () {
var connectDragSource = this.props.connectDragSource;
var isDragging = this.props.isDragging;
return connectDragSource(
<div style={{
opacity: isDragging ? 0.5 : 1,
fontSize: 25,
fontWeight: 'bold',
cursor: 'move'
}}>
♘
</div>
);
}
});
module.exports = DragSource(ItemTypes.KNIGHT, knightSource, collect)(Knight);
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:
var React = require('react');
var PropTypes = React.PropTypes;
var Square = require('./Square');
var BoardSquare = React.createClass({
propTypes: {
x: PropTypes.number.isRequired,
y: PropTypes.number.isRequired
},
render: function () {
var x = this.props.x;
var y = this.props.y;
var black = (x + y) % 2 === 1;
return (
<Square black={black}>
{this.props.children}
</Square>
);
}
});
module.exports = BoardSquare;
I also changed the Board
to use it:
renderSquare: function (i) {
var x = i % 8;
var 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: function (x, y) {
var knightX = this.props.knightPosition[0]
var knightY = this.props.knightPosition[1];
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:
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:
var React = require('react');
var PropTypes = React.PropTypes;
var Square = require('./Square');
var canMoveKnight = require('./Game').canMoveKnight;
var moveKnight = require('./Game').moveKnight;
var ItemTypes = require('./Constants').ItemTypes;
var DropTarget = require('react-dnd').DropTarget;
var squareTarget = {
drop: function (props) {
moveKnight(props.x, props.y);
}
};
function collect(connect, monitor) {
return {
connectDropTarget: connect.dropTarget(),
isOver: monitor.isOver()
};
}
var BoardSquare = React.createClass({
propTypes: {
x: PropTypes.number.isRequired,
y: PropTypes.number.isRequired,
isOver: PropTypes.bool.isRequired
},
render: function () {
var x = this.props.x;
var y = this.props.y;
var connectDropTarget = this.props.connectDropTarget;
var isOver = this.props.isOver;
var 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>
);
}
});
module.exports = DropTarget(ItemTypes.KNIGHT, squareTarget, collect)(BoardSquare);
This is starting to look good! There is just one change left to complete this tutorial. We want to highlight the BoardSquare
s that represent the valid moves, and only process the drop if it happened over one of those valid BoardSquare
s.
Thankfully, it is really easy to do with React DnD. I just need to define a canDrop
method in my drop target specification:
I'm also adding monitor.canDrop()
to my collecting function, as well as some overlay rendering code to the component:
var React = require('react');
var PropTypes = React.PropTypes;
var Square = require('./Square');
var canMoveKnight = require('./Game').canMoveKnight;
var moveKnight = require('./Game').moveKnight;
var ItemTypes = require('./Constants').ItemTypes;
var DropTarget = require('react-dnd').DropTarget;
var squareTarget = {
canDrop: function (props) {
return canMoveKnight(props.x, props.y);
},
drop: function (props) {
moveKnight(props.x, props.y);
}
};
function collect(connect, monitor) {
return {
connectDropTarget: connect.dropTarget(),
isOver: monitor.isOver(),
canDrop: monitor.canDrop()
};
}
var BoardSquare = React.createClass({
propTypes: {
x: PropTypes.number.isRequired,
y: PropTypes.number.isRequired,
isOver: PropTypes.bool.isRequired,
canDrop: PropTypes.bool.isRequired
},
renderOverlay: function (color) {
return (
<div style={{
position: 'absolute',
top: 0,
left: 0,
height: '100%',
width: '100%',
zIndex: 1,
opacity: 0.5,
backgroundColor: color,
}} />
);
},
render: function () {
var x = this.props.x;
var y = this.props.y;
var connectDropTarget = this.props.connectDropTarget;
var isOver = this.props.isOver;
var 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>
);
}
}
module.exports = DropTarget(ItemTypes.KNIGHT, squareTarget, collect)(BoardSquare);
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: function () {
var connectDragPreview = this.props.connectDragPreview;
var img = new Image();
img.src = '';
img.onload = function () {
connectDragPreview(img);
};
},
Happy dragging and dropping.
Now go and play with it!