Tic Tac Toe

React 官方教程-井字棋

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
import { useState } from 'react';
/*
React provides a special function called useState that you can call from your component to let it “remember” things.
Let’s store the current value of the Square in state, and change it when the Square is clicked.
*/

// 九宫格 单个宫格
// value: X, O, null
function Square({ value, onSquareClick }) {
  return (
    <button className="square" onClick={onSquareClick}>
      {value}
    </button>
  );
}

function Board({ xIsNext, squares, onPlay }) {
  function handleClick(i) {
    // 如果已经判定胜负, 或者被点击的单元格已经被点击过, 不做处理
    if (calculateWinner(squares) || squares[i]) {
      return;
    }
    // 点击后的棋盘
    const nextSquares = squares.slice();
    if (xIsNext) {
      nextSquares[i] = 'X';
    } else {
      nextSquares[i] = 'O';
    }
    onPlay(nextSquares);
  }

  const winner = calculateWinner(squares);
  let status;
  if (winner) {
    status = 'Winner: ' + winner;
  } else {
    status = 'Next player: ' + (xIsNext ? 'X' : 'O');
  }

  return (
    <>
      <div className="status">{status}</div>
      <div className="board-row">
        <Square value={squares[0]} onSquareClick={() => handleClick(0)} />
        <Square value={squares[1]} onSquareClick={() => handleClick(1)} />
        <Square value={squares[2]} onSquareClick={() => handleClick(2)} />
      </div>
      <div className="board-row">
        <Square value={squares[3]} onSquareClick={() => handleClick(3)} />
        <Square value={squares[4]} onSquareClick={() => handleClick(4)} />
        <Square value={squares[5]} onSquareClick={() => handleClick(5)} />
      </div>
      <div className="board-row">
        <Square value={squares[6]} onSquareClick={() => handleClick(6)} />
        <Square value={squares[7]} onSquareClick={() => handleClick(7)} />
        <Square value={squares[8]} onSquareClick={() => handleClick(8)} />
      </div>
    </>
  );
}

export default function Game() {
  // 设置棋盘记录状态
  const [history, setHistory] = useState([Array(9).fill(null)]);
  // [null, null, null, null, null, null, null, null, null]
  // 第 N 步
  const [currentMove, setCurrentMove] = useState(0);
  // 判断下一个是玩家 X 或者 O
  const xIsNext = currentMove % 2 === 0;
  // 当前步对应的棋盘
  const currentSquares = history[currentMove];

  function handlePlay(nextSquares) {
    // 三个点 `...`, 展开语法
    // nextHistory 是二维数组, 每一行对应一步执行完后的棋盘
    const nextHistory = [...history.slice(0, currentMove + 1), nextSquares];
    setHistory(nextHistory);
    setCurrentMove(nextHistory.length - 1);
  }
  // 这里 jumpTo 基本等同于 setCurrentMove, 为什么要这样写?
  // 推测 setCurrentMove 没有对外暴露, 待完善
  function jumpTo(nextMove) {
    setCurrentMove(nextMove);
  }

  const moves = history.map((squares, move) => {
    let description;
    if (move > 0) {
      description = 'Go to move #' + move;
    } else {
      description = 'Go to game start';
    }
    return (
      <li key={move}>
        <button onClick={() => jumpTo(move)}>{description}</button>
      </li>
    );
  });

  return (
    <div className="game">
      <div className="game-board">
        <Board xIsNext={xIsNext} squares={currentSquares} onPlay={handlePlay} />
      </div>
      <div className="game-info">
        <ol>{moves}</ol>
      </div>
    </div>
  );
}

// 枚举出所有可能的, 一方获胜的情况
function calculateWinner(squares) {
  const lines = [
    [0, 1, 2],
    [3, 4, 5],
    [6, 7, 8],
    [0, 3, 6],
    [1, 4, 7],
    [2, 5, 8],
    [0, 4, 8],
    [2, 4, 6],
  ];
  for (let i = 0; i < lines.length; i++) {
    const [a, b, c] = lines[i];
    // a=b=c 且不为 null
    if (squares[a] && squares[a] === squares[b] && squares[a] === squares[c]) {
      return squares[a];
    }
  }
  return null;
}
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
<!-- 以下是生成的 HTML -->

<div class="game">
  <div class="game-board">
    <div class="status">Next player: X</div>
    <div class="board-row">
      <button class="square">
      </button>
      <button class="square">
      </button>
      <button class="square">
      </button>
    </div>
    <div class="board-row">
      <button class="square">
      </button>
      <button class="square">
      </button>
      <button class="square">
      </button>
    </div>
    <div class="board-row">
      <button class="square">
      </button>
      <button class="square">
      </button>
      <button class="square">
      </button>
    </div>
  </div>
  <div class="game-info">
    <ol>
      <li>
        <button>Go to game start</button>
      </li>
    </ol>
  </div>
</div>

Tutorial: Tic-Tac-Toe – React