- 后端
- HTML5
Made with ❤️ by 紫升
现在,您已经阅读了概述,冒险时间到了!
在本教程中,我们将使用 React 和 React DnD 构建一个国际象棋游戏。开个玩笑!编写一个完整的国际象棋游戏完全超出了本教程的范围。我们要构建的是一个带有国际象棋棋盘和孤独骑士的小应用程序。根据国际象棋的规则,骑士是可以拖动的。
如果您已经熟悉 React,请跳至最后一节: 添加拖放交互
我们将通过这个示例来演示 react-dnd 的数据驱动理念。您将学习如何创建拖放源和拖放目标,将它们与 React 组件连接起来,并根据拖放事件改变它们的外观。
现在,让我们构建一些东西!
本教程中的代码示例使用了函数组件和现代 JavaScript 语法。建议使用构建步骤将这些功能移植到目标环境中。我们推荐使用 create-react-app。
本网站提供了我们要构建的应用程序示例。
我们将首先创建一些 React 组件,不考虑拖放交互。我们的 "孤独骑士 "应用程序将由哪些组件组成?我能想到几个:
骑士,我们孤独的骑士棋子; 方格,棋盘上的一个方格; Board (棋盘),整个棋盘的 64 个方格。
让我们考虑一下他们的道具。
骑士可能不需要道具。它有一个位置,但骑士没有理由知道它的位置,因为它可以作为一个子节点被放置到 Square 中。
我们很想通过道具来确定 Square 的位置,但这同样没有必要,因为它在渲染时真正需要的信息只有颜色。我将默认把 Square 设为白色,并添加一个黑色布尔值道具。当然,Square 只接受一个子组件:当前在它上面的棋子。我选择白色作为默认背景色,以符合浏览器的默认设置。
棋盘很棘手。将 Squares 作为子代传递给它毫无意义,因为棋盘还能包含什么呢?因此,它可能拥有正方形。但是,它还需要拥有骑士,因为这家伙需要被放置在其中一个正方形内。这意味着棋盘需要知道马的当前位置。在真正的国际象棋游戏中,棋盘会接受一个描述所有棋子、颜色和位置的数据结构,但对我们来说,一个骑士位置参数就足够了。我们将使用双项数组作为坐标,其中 [0, 0] 指的是 A8 格。为什么是 A8 而不是 A1?为了与浏览器的坐标方向相匹配。我试过另一种方法,结果让我头疼不已。
当前状态放在哪里?我真的不想把它放在 Board 组件中。尽可能减少组件中的状态是个好主意,而且因为 Board 已经有了一些布局逻辑,我不想再让它承担管理状态的负担。
好消息是,这一点并不重要。我们只需编写组件,就好像状态存在于某个地方一样,并确保它们在通过道具接收状态时能正确呈现,之后再考虑管理状态的问题!
我更喜欢自下而上地开始,因为这样我总是在使用已经存在的东西。如果我先做棋盘,在完成方格之前,我不会看到我的成果。另一方面,我可以在不考虑木板的情况下直接制作并看到方形。我认为即时反馈回路很重要(你可以从我做的另一个项目中看出这一点)。
事实上,我打算从 "骑士 "开始。它根本没有任何道具,而且是最容易制作的:
import React from 'react';export default function Knight() {return <span>♘</span>;}
是的,♘ 是 Unicode 的骑士!它真漂亮。我们本可以把它的颜色作为道具,但在我们的例子中,我们不会有任何黑色骑士,所以没有必要这样做。 渲染似乎没有问题,但为了确保万无一失,我立即更改了入口点进行测试:
import React from 'react';import ReactDOM from 'react-dom';import Knight from './Knight';ReactDOM.render(<Knight />, document.getElementById('root'));
每当我编写另一个组件时,我都会这样做,这样我就总是有东西可以渲染。在一个较大的应用程序中,我会使用像 cosmos 这样的组件游乐场,这样我就不会在黑暗中编写组件了。
我在屏幕上看到了我的骑士!是时候继续实现 Square 了。这是我的第一次尝试:
import React from 'react';export default function Square({ black }) {const fill = black ? 'black' : 'white';return <div style={{ backgroundColor: fill }} />;}
现在,我改变入口点代码,看看骑士在广场内的样子:
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'),);
遗憾的是,屏幕上空空如也。我犯了几个错误:
我忘了给 Squareany 定尺寸,所以它只是折叠起来。我不想让它有任何固定尺寸,所以我给它**width:"100%"和height:"100%"**来填充容器。
我忘了把 {children} 放在 Square 返回的 div 内,所以它忽略了 Knight 。
即使改正了这两个错误,当 Square 变成黑色时,我仍然看不到我的骑士。这是因为页面正文的默认颜色是黑色,所以在黑色正方形上看不到。我本可以通过给 Knight 添加颜色道具来解决这个问题,但更简单的办法是在设置 backgroundColor 的地方设置相应的颜色样式。这个版本的 Square 修正了这些错误,而且在使用两种颜色时效果都很好:
import React from 'react';export default function Square({ black, children }) {const fill = black ? 'black' : 'white';const stroke = black ? 'white' : 'black';return (<divstyle={{backgroundColor: fill,color: stroke,width: '100%',height: '100%',}}>{children}</div>);}
终于可以开始使用棋盘了!我将从一个极其幼稚的版本开始,只画同样的一个正方形:
import React from 'react';import Square from './Square';import Knight from './Knight';export default function Board() {return (<div><Square black><Knight /></Square></div>);}
到目前为止,我唯一的目的就是让它呈现出来,这样我就可以开始调整它了:
import React from 'react';import ReactDOM from 'react-dom';import Board from './Board';ReactDOM.render(<Board knightPosition={[0, 0]} />,document.getElementById('root'),);
的确,我看到了同样的单个方块。我现在要添加一大堆方块!但我不知道从哪里开始。我应该在 render 中加入什么?某种循环?某个数组上的 map?
老实说,我现在不想考虑这些。我已经知道如何渲染一个有骑士或没有骑士的单个方格。通过 knightPosition,我还知道了骑士的位置。这意味着我可以编写 renderSquare 方法,而不必担心渲染 整个棋盘。
我对 renderSquare 的第一次尝试是这样的
function renderSquare(x, y, [knightX, knightY]) {const black = (x + y) % 2 === 1;const isKnightHere = knightX === x && knightY === y;const piece = isKnightHere ? <Knight /> : null;return <Square black={black}>{piece}</Square>;}
我已经可以试一试了,把电路板的效果图改为
export default function Board({ knightPosition }) {return (<divstyle={{width: '100%',height: '100%',}}>{renderSquare(0, 0, knightPosition)}{renderSquare(1, 0, knightPosition)}{renderSquare(2, 0, knightPosition)}</div>);}
这时,我意识到我忘了给我的方格设计任何布局。我要使用 Flexbox。我在根 div 中添加了一些样式,并将 Squares 封装到 div 中,这样我就可以对它们进行布局了。一般来说,让组件保持封装状态是个好主意,即使这意味着要添加封装 div,也不会影响它们的布局。
import React from 'react';import Square from './Square';import Knight from './Knight';function renderSquare(i, [knightX, knightY]) {const x = i % 8;const y = Math.floor(i / 8);const isKnightHere = x === knightX && y === knightY;const black = (x + y) % 2 === 1;const piece = isKnightHere ? <Knight /> : null;return (<div key={i} style={{ width: '12.5%', height: '12.5%' }}><Square black={black}>{piece}</Square></div>);}export default function Board({ knightPosition }) {const squares = [];for (let i = 0; i < 64; i++) {squares.push(renderSquare(i, knightPosition));}return (<divstyle={{width: '100%',height: '100%',display: 'flex',flexWrap: 'wrap',}}>{squares}</div>);}
它看起来非常棒!我不知道如何限制 Board 保持正方形的长宽比,但以后应该很容易添加。
想一想吧。通过改变骑士的位置,我们刚刚从一无所知到能够在一个漂亮的棋盘上移动骑士:
import React from 'react';import ReactDOM from 'react-dom';import Board from './Board';ReactDOM.render(<Board knightPosition={[7, 4]} />,document.getElementById('root'),);
声明性非常棒!这就是人们喜欢使用 React 的原因。
我们想让骑士可移动。要做到这一点,我们需要将当前的骑士位置保存在某种状态存储器中,并有某种方法来改变它。
由于设置这种状态需要花些心思,我们不会同时尝试实现拖动。相反,我们将从一个更简单的实现开始。我们将在点击特定方格时移动骑士,但前提是国际象棋规则允许这样做。实现这一逻辑应该能让我们对状态管理有足够的了解,因此我们可以在处理好状态管理后用拖放来代替点击。
React 对状态管理或数据流没有意见;你可以使用 Flux、Redux、Rx 甚至 Backbone,但要避免肥胖的模型,并将读取和写入分离开来。
在这个简单的例子中,我不想费心安装或设置 Redux,所以我打算采用更简单的模式。它的扩展性不如 Redux,但我也不需要它。我还没有决定我的状态管理器的 API,但我打算把它叫做 Game,而且它肯定需要有某种方式向我的 React 代码发送数据变化信号。
既然知道了这么多,我就可以用一个还不存在的假想 Game 重写我的 index.jsx。请注意,这次我是盲写代码,还不能运行。这是因为我还在摸索 API:
import React from 'react';import ReactDOM from 'react-dom';import Board from './Board';import { observe } from './Game';const root = document.getElementById('root');observe((knightPosition) =>ReactDOM.render(<Board knightPosition={knightPosition} />, root),);
observe我导入的这个函数是什么?这只是我能想到的订阅变化状态的最简单的方法。我本可以做到这一点,但当我所需要的只是一个单一的变化事件时,EventEmitter到底为什么还要去那里呢?我本可以创建 Game 一个对象模型,但当我需要的只是值流时,为什么要这样做呢?
为了验证这个订阅 API 是否有意义,我将编写一个Game发出随机位置的假值:
export function observe(receive) {const randPos = () => Math.floor(Math.random() * 8);setInterval(() => receive([randPos(), randPos()]), 500);}
这显然不是很有用。如果我们想要一些互动性,就需要一种方法来修改组件中的游戏状态。目前,我打算保持简单,暴露一个直接修改内部状态的 moveKnight 函数。在一个中等复杂度的应用程序中,不同的状态存储空间可能都有兴趣更新它们的状态以响应用户的一个操作,因此这种方法并不合适,但在我们的情况下,这样做就足够了:
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();}
现在,让我们回到我们的组件。此时我们的目标是将骑士移动到被点击的方格中。有一种方法是从方格本身调用 moveKnight。然而,这需要我们传递方格的位置。这里有一个很好的经验法则:
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();}
现在,让我们回到我们的组件。此时我们的目标是将骑士移动到被点击的方格中。有一种方法是从方格本身调用 moveKnight。然而,这需要我们传递方格的位置。这里有一个很好的经验法则:
如果一个组件在渲染时不需要某些数据,那么它就根本不需要这些数据。
正方形在渲染时不需要知道自己的位置。因此,此时最好避免将其与 moveKnight 方法耦合。取而代之的是,我们将在将正方形包裹在棋盘内的 div 中添加一个 onClick 处理程序:
import React from 'react';import Square from './Square';import Knight from './Knight';import { moveKnight } from './Game';/* ... */function renderSquare(i, knightPosition) {/* ... */return <div onClick={() => handleSquareClick(x, y)}>{/* ... */}</div>;}function handleSquareClick(toX, toY) {moveKnight(toX, toY);}
我们也可以为 Square 添加一个 onClick,并用它来代替,但既然我们稍后会移除点击处理程序,转而使用拖放界面,又何必多此一举呢。
现在缺少的最后一块是国际象棋规则检查。骑士不能随意移动到任意方格,它只能进行 L 形移动。我正在为游戏添加一个 canMoveKnight(toX, toY)函数,并将初始位置改为 B1,以符合国际象棋规则:
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));}
最后,我在 handleSquareClick 中添加了 canMoveKnight 检查:
import { canMoveKnight, moveKnight } from './Game';/* ... */function handleSquareClick(toX, toY) {if (canMoveKnight(toX, toY)) {moveKnight(toX, toY);}}
目前运行良好!
这正是促使我编写本教程的原因。现在,我们将看到 React DnD 如何轻松地为现有组件添加拖放交互。
这部分内容假定您至少对概述中介绍的概念有一定程度的了解,例如后端、收集函数、类型、项目、拖动源和拖放目标。如果您没有完全理解,也没关系,但请确保在进入编码过程之前至少给自己一个机会。
我们将首先安装 React DnD 及其 HTML5 后端:
npm install react-dnd react-dnd-html5-backend
今后,您可能想探索其他第三方后端,如触摸后端,但这不在本教程的范围之内。
我们需要在应用程序中设置的第一件事就是 DndProvider。它应安装在应用程序的顶部附近。我们需要用它来指定我们将使用 HTML5Backend。
import React from 'react';import { DndProvider } from 'react-dnd';import { HTML5Backend } from 'react-dnd-html5-backend';function Board() {/* ... */return <DndProvider backend={HTML5Backend}>...</DndProvider>;}
接下来,我要为可拖动的物品类型创建常量。我们的游戏中只有一种物品类型,那就是 KNIGHT。我将创建一个常量模块来导出它:
export const ItemTypes = {KNIGHT: 'knight',};
准备工作已经完成。让我们把骑士拖动起来吧!
useDrag 接受一个 memoization 函数,该函数返回一个规范对象。在这个对象中,item.type 被设置为我们刚刚定义的常量,因此现在我们需要编写一个收集函数。
const [{ isDragging }, drag] = useDrag(() => ({type: ItemTypes.KNIGHT,collect: (monitor) => ({isDragging: !!monitor.isDragging(),}),}));
让我们来分析一下:
import React from 'react'import { ItemTypes } from './Constants'import { useDrag } from 'react-dnd'function Knight() {const [{isDragging}, drag] = useDrag(() => ({type: ItemTypes.KNIGHT,collect: monitor => ({isDragging: !!monitor.isDragging(),}),}))return (<divref={drag}style={{opacity: isDragging ? 0.5 : 1,fontSize: 25,fontWeight: 'bold',cursor: 'move',}}>♘</div>,)}export default Knight
现在,"骑士 "是一个拖动源,但还没有处理空投的空投目标。我们现在要将正方形设为拖放目标。
这一次,我们无法避免将位置传递给方块。毕竟,如果正方形不知道自己的位置,它又怎么知道被拖动的骑士应该移动到哪里呢?另一方面,这仍然让人感觉不妥,因为在我们的应用程序中,方块作为一个实体并没有改变,既然它过去很简单,为什么要把它复杂化呢 ?面对这种两难境地,是时候将智能组件和傻瓜组件分开了。
我将介绍一个名为 BoardSquare 的新组件。它可以渲染老式的正方形,但也能意识到自己的位置。事实上,它封装了 Board 内的 renderSquare 方法曾经做过的一些逻辑。在合适的时候,React 组件通常会从这类呈现子方法中提取出来。
下面是我提取的 BoardSquare:
import React from 'react';import Square from './Square';export default function BoardSquare({ x, y, children }) {const black = (x + y) % 2 === 1;return <Square black={black}>{children}</Square>;}
我还更换了 Board 来使用它:
/* ... */import BoardSquare from './BoardSquare';function renderSquare(i, knightPosition) {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}>{renderPiece(x, y, knightPosition)}</BoardSquare></div>);}function renderPiece(x, y, [knightX, knightY]) {if (x === knightX && y === knightY) {return <Knight />;}}
现在让我们用一个 useDrophook 来包装 BoardSquare。我将编写一个只处理下降事件的下降目标规范:
const [, drop] = useDrop(() => ({accept: ItemTypes.KNIGHT,drop: () => moveKnight(x, y),}),[x, y],);
看到了吗?drop 方法的作用域中包含了 BoardSquare 的道具,因此它知道当骑士下降时应将其移动到哪里。在真正的应用程序中,我可能还会使用 monitor.getItem()来获取拖动源从 beginDrag 返回的被拖动项,但由于整个应用程序中只有一个可拖动项,所以我并不需要它。
在我的收集函数中,我将询问监视器指针当前是否位于 BoardSquare 上,以便突出显示它:
const [{ isOver }, drop] = useDrop(() => ({accept: ItemTypes.KNIGHT,drop: () => moveKnight(x, y),collect: (monitor) => ({isOver: !!monitor.isOver(),}),}),[x, y],);
更改呈现功能以连接下拉目标并显示高亮叠加后,BoardSquare 就变成了这样:
import React from 'react'import Square from './Square'import { canMoveKnight, moveKnight } from './Game'import { ItemTypes } from './Constants'import { useDrop } from 'react-dnd'function BoardSquare({ x, y, children }) {const black = (x + y) % 2 === 1const [{ isOver }, drop] = useDrop(() => ({accept: ItemTypes.KNIGHT,drop: () => moveKnight(x, y),collect: monitor => ({isOver: !!monitor.isOver(),}),}), [x, y])return (<divref={drop}style={{position: 'relative',width: '100%',height: '100%',}}><Square black={black}>{children}</Square>{isOver && (<divstyle={{position: 'absolute',top: 0,left: 0,height: '100%',width: '100%',zIndex: 1,opacity: 0.5,backgroundColor: 'yellow',}}/>)}</div>,)}export default BoardSquare
看起来效果不错!要完成本教程,只剩下一个改动了。我们要突出显示代表有效棋步的棋盘方格,只有当落子发生在其中一个有效棋盘方格上时,才会对落子进行处理。
值得庆幸的是,这在 React DnD 中很容易实现。我只需在下棋目标规范中定义一个 canDrop 方法即可:
const [{ isOver, canDrop }, drop] = useDrop(() => ({accept: ItemTypes.KNIGHT,canDrop: () => canMoveKnight(x, y),drop: () => moveKnight(x, y),collect: (monitor) => ({isOver: !!monitor.isOver(),canDrop: !!monitor.canDrop(),}),}),[x, y],);
我还在收集函数中添加了 monitor.canDrop(),并在组件中添加了一些叠加渲染代码:
import React from 'react';import { useDrop } from 'react-dnd';import { ItemTypes } from './Constants';import { canMoveKnight, moveKnight } from './Game';import Square from './Square';function BoardSquare({ x, y, children }) {const black = (x + y) % 2 === 1;const [{ isOver, canDrop }, drop] = useDrop(() => ({accept: ItemTypes.KNIGHT,drop: () => moveKnight(x, y),canDrop: () => canMoveKnight(x, y),collect: (monitor) => ({isOver: !!monitor.isOver(),canDrop: !!monitor.canDrop(),}),}),[x, y],);return (<divref={drop}style={{position: 'relative',width: '100%',height: '100%',}}><Square black={black}>{children}</Square>{isOver && !canDrop && <Overlay color="red" />}{!isOver && canDrop && <Overlay color="yellow" />}{isOver && canDrop && <Overlay color="green" />}</div>);}export default BoardSquare;
最后要演示的是拖动预览自定义。当然,浏览器会截图 DOM 节点,但如果我们想显示自定义图片呢?
我们又是幸运的,因为使用 React DnD 可以轻松做到这一点。我们只需使用 useDrag 提供的预览 ref 即可。
const [{ isDragging }, drag, preview] = useDrag(() => ({type: ItemTypes.KNIGHT,collect: (monitor) => ({isDragging: !!monitor.isDragging(),}),}));
react-dnd 还提供了一个实用组件 DragPreviewImage,它可以使用此 ref 将图像显示为拖动预览。
const knightImage = '';render() {return (<><DragPreviewImage connect={preview} src={knightImage} /><divref={drag}style={{...knightStyle,opacity: isDragging ? 0.5 : 1,}}>♘</div></>)}}
本教程将引导您创建 React 组件,对这些组件和应用程序状态进行设计决策,最后添加拖放交互。本教程的目的是向您展示 React DnD 与 React 的理念非常契合,而且您应该首先考虑应用程序的架构,然后再深入实施复杂的交互。
祝您拖放愉快。
现在去玩吧!