@oathompsonjones/mini-games 1.0.6 → 1.0.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/TODO.md ADDED
@@ -0,0 +1,6 @@
1
+ [ ] Write this program in Rust (faster + I should learn it).
2
+ The rust program should have a few modes:
3
+ - [ ] CPU vs CPU
4
+ - [ ] Human vs CPU
5
+ - [ ] Human vs human
6
+ - [ ] Find optimal move (this will take in a game state, then calculate the most optimal move for the next player)
@@ -1,4 +1,5 @@
1
1
  import type BitBoard from "../bitBoard/bitBoard.js";
2
+ import LongInt from "../bitBoard/longInt.js";
2
3
  export type Position = {
3
4
  y: number;
4
5
  x: number;
@@ -17,16 +18,17 @@ export declare const enum GridLines {
17
18
  TBottom = "\u2534",
18
19
  Cross = "\u253C"
19
20
  }
20
- /** Represents a game board. */
21
- export default abstract class Board<T extends BitBoard = BitBoard> {
22
- /** Contains the ID of the game. */
23
- gameID: unknown;
21
+ /**
22
+ * Represents a game board.
23
+ * @template T - The type of the numeric data.
24
+ */
25
+ export default abstract class Board<T extends LongInt | number> {
24
26
  /** Contains the data stored in a BitBoard. */
25
- protected readonly bitBoard: T;
26
- /** The width of the board. */
27
- protected readonly boardWidth: number;
28
- /** The height of the board. */
29
- protected readonly boardHeight: number;
27
+ protected readonly bitBoard: BitBoard<T>;
28
+ /** The width of the game board. */
29
+ protected readonly width: number;
30
+ /** The height of the game board. */
31
+ protected readonly height: number;
30
32
  /** A stack of moves. */
31
33
  protected readonly moves: number[];
32
34
  /** How many boards there are representing player positions (most likely 2). */
@@ -34,96 +36,102 @@ export default abstract class Board<T extends BitBoard = BitBoard> {
34
36
  /** Number of boards in total (most likely also 2). */
35
37
  private readonly numberOfBoards;
36
38
  /** The board states which represent a winning state. */
37
- protected abstract readonly winningStates: T[];
39
+ protected abstract readonly winningStates: Array<BitBoard<T>>;
38
40
  /**
39
41
  * Creates an instance of Board.
40
- * @param width The width of the board.
41
- * @param height The height of the board.
42
- * @param playerBoardCount How many boards there are representing player positions (most likely 2).
43
- * @param extraBoardCount Number of extra boards (most likely 0).
42
+ * @param width - The width of the game board.
43
+ * @param height - The height of the game board.
44
+ * @param playerBoardCount - How many boards there are representing player positions (most likely 2).
45
+ * @param extraBoardCount - Number of extra boards (most likely 0).
44
46
  */
45
47
  protected constructor(width: number, height: number, playerBoardCount?: number, extraBoardCount?: number);
46
- /** Calculates whether or not the board is full. */
48
+ /**
49
+ * Calculates whether or not the board is full.
50
+ * @returns Whether or not the board is full.
51
+ */
47
52
  get isFull(): boolean;
48
- /** Calculates whether or not the board is empty. */
53
+ /**
54
+ * Calculates whether or not the board is empty.
55
+ * @returns Whether or not the board is empty.
56
+ */
49
57
  get isEmpty(): boolean;
50
58
  /**
51
59
  * Calculates who the winner is.
52
- * If the game is not over, the output is false.
53
- * If there is a winner, the output is their ID.
54
- * If there is a draw, the output is null.
60
+ * @returns `false` if the game is not over, the player ID if there is a winner, and `null` if there is a draw.
55
61
  */
56
62
  get winner(): 0 | 1 | false | null;
57
- /** Calculates which cells are empty. */
63
+ /**
64
+ * Calculates which cells are empty.
65
+ * @returns The empty cells on the board.
66
+ */
58
67
  get emptyCells(): Position[];
59
68
  /** Calculates the heuristic score for a given board state. */
60
69
  abstract get heuristic(): number;
61
- /**
62
- * Sets the game ID.
63
- * @param id The ID of the game.
64
- */
65
- setGameID(id: unknown): void;
66
70
  /**
67
71
  * Makes a move on the board.
68
- * @param move The position of the move.
69
- * @param playerId The player who's making the move.
72
+ * @param move - The position of the move.
73
+ * @param playerId - The player who's making the move.
70
74
  */
71
75
  makeMove(move: Position, playerId: number): void;
72
- /** Reverses the last move. */
76
+ /**
77
+ * Reverses the last move.
78
+ * @throws {Error} - If there is no move to undo.
79
+ */
73
80
  undoLastMove(): void;
74
81
  /**
75
82
  * Checks if a move is valid.
76
- * @param move The move.
83
+ * @param move - The position of the move.
77
84
  * @returns Whether or not it's valid.
78
85
  */
79
86
  moveIsValid(move: Position): boolean;
80
87
  /**
81
88
  * Checks which player is occupying a given cell.
82
- * @param cell The cell to check.
89
+ * @param cell - The cell to check.
83
90
  * @returns If the cell is empty, the output is null, otherwise the output is the player's ID.
84
91
  */
85
92
  cellOccupier(cell: Position): number | null;
86
93
  /**
87
94
  * Creates a string representation of the board.
88
- * @param wrap Whether or not to provide a border for the board.
89
- * @param labelX Whether or not to label x.
90
- * @param labelY Whether or not to label y.
91
- * @param symbols The symbols to use as board pieces.
92
- * @param colour Whether or not to colour the pieces.
95
+ * @param wrap - Whether or not to provide a border for the board.
96
+ * @param labelX - Whether or not to label x.
97
+ * @param labelY - Whether or not to label y.
98
+ * @param symbols - The symbols to use as board pieces.
99
+ * @param colour - Whether or not to colour the pieces.
93
100
  * @returns The string representation.
101
+ * @throws {Error} - If the symbols are not the same length.
94
102
  */
95
103
  toString(wrap?: boolean, labelX?: boolean, labelY?: boolean, symbols?: string[], colour?: boolean): string;
96
104
  /**
97
105
  * Determines if a given player has a line of pieces on the board.
98
- * @param playerId The ID of the player.
99
- * @param length The number of pieces needed.
100
- * @param maxGaps The number of gaps allowed for a line to be valid. Defaults to 0.
106
+ * @param playerId - The ID of the player to check.
107
+ * @param length - The number of pieces needed.
108
+ * @param maxGaps - The number of gaps allowed for a line to be valid. Defaults to 0.
101
109
  * @returns How many lines exist.
102
110
  */
103
111
  hasLine(playerId: number, length: number, maxGaps?: number): number;
104
112
  /**
105
113
  * Gets a bit index from its coordinates and player ID.
106
- * @param move The coordinates.
107
- * @param playerId The player ID.
108
- * @returns The bit index.
114
+ * @param move - The coordinates.
115
+ * @param playerId - The player ID to use.
116
+ * @returns The bit index of the move.
109
117
  */
110
118
  protected getBitIndex(move: Position, playerId: number): number;
111
119
  /**
112
120
  * A BitBoard containing only the player's bits.
113
- * @param playerId The player's ID.
121
+ * @param playerId - The player's ID to get the board for.
114
122
  * @returns The player's bits.
115
123
  */
116
- protected getPlayerBoard(playerId: number): T;
124
+ protected getPlayerBoard(playerId: number): BitBoard<T>;
117
125
  /**
118
126
  * Gets a bit index from its coordinates.
119
- * @param move The coordinates.
127
+ * @param move - The coordinates.
120
128
  * @returns The bit index.
121
129
  */
122
130
  private getIndex;
123
131
  /**
124
132
  * Checks if a move is valid for the given board.
125
133
  * Does not check if that cell is already occupied.
126
- * @param position The position to check.
134
+ * @param position - The position to check.
127
135
  * @returns Whether or not that cell exists on the board.
128
136
  */
129
137
  private isValidPosition;
@@ -1,16 +1,17 @@
1
1
  import IntBitBoard from "../bitBoard/intBitBoard.js";
2
2
  import LongInt from "../bitBoard/longInt.js";
3
3
  import LongIntBitBoard from "../bitBoard/longIntBitBoard.js";
4
- /** Represents a game board. */
4
+ /**
5
+ * Represents a game board.
6
+ * @template T - The type of the numeric data.
7
+ */
5
8
  export default class Board {
6
- /** Contains the ID of the game. */
7
- gameID = "";
8
9
  /** Contains the data stored in a BitBoard. */
9
10
  bitBoard;
10
- /** The width of the board. */
11
- boardWidth;
12
- /** The height of the board. */
13
- boardHeight;
11
+ /** The width of the game board. */
12
+ width;
13
+ /** The height of the game board. */
14
+ height;
14
15
  /** A stack of moves. */
15
16
  moves = [];
16
17
  /** How many boards there are representing player positions (most likely 2). */
@@ -19,38 +20,46 @@ export default class Board {
19
20
  numberOfBoards;
20
21
  /**
21
22
  * Creates an instance of Board.
22
- * @param width The width of the board.
23
- * @param height The height of the board.
24
- * @param playerBoardCount How many boards there are representing player positions (most likely 2).
25
- * @param extraBoardCount Number of extra boards (most likely 0).
23
+ * @param width - The width of the game board.
24
+ * @param height - The height of the game board.
25
+ * @param playerBoardCount - How many boards there are representing player positions (most likely 2).
26
+ * @param extraBoardCount - Number of extra boards (most likely 0).
26
27
  */
27
28
  constructor(width, height, playerBoardCount = 2, extraBoardCount = 0) {
28
- this.boardWidth = width;
29
- this.boardHeight = height;
29
+ this.width = width;
30
+ this.height = height;
30
31
  this.numberOfPlayerBoards = playerBoardCount;
31
32
  this.numberOfBoards = this.numberOfPlayerBoards + extraBoardCount;
32
- const totalBits = this.boardWidth * this.boardHeight * this.numberOfBoards;
33
- this.bitBoard = (totalBits > 32 ? new LongIntBitBoard(Math.ceil(totalBits / 32)) : new IntBitBoard());
33
+ const totalBits = this.width * this.height * this.numberOfBoards;
34
+ this.bitBoard = (totalBits > 32
35
+ ? new LongIntBitBoard(Math.ceil(totalBits / 32))
36
+ : new IntBitBoard());
34
37
  }
35
- /** Calculates whether or not the board is full. */
38
+ /**
39
+ * Calculates whether or not the board is full.
40
+ * @returns Whether or not the board is full.
41
+ */
36
42
  get isFull() {
37
43
  let isFull = (this.bitBoard instanceof LongIntBitBoard
38
- ? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))
44
+ ? new LongIntBitBoard(Math.ceil(this.width * this.height / 32))
39
45
  : new IntBitBoard());
40
46
  for (let i = 0; i < this.numberOfPlayerBoards; i++)
41
47
  isFull = isFull.or(this.getPlayerBoard(i));
42
- const fullValue = this.bitBoard instanceof LongIntBitBoard
48
+ const fullValue = (this.bitBoard instanceof LongIntBitBoard
43
49
  ? new LongInt([
44
- ...Array(Math.ceil(this.boardWidth * this.boardHeight / 32) - 1).fill(~0 >>> 0),
45
- 2 ** (this.boardWidth * this.boardHeight - (Math.ceil(this.boardWidth * this.boardHeight / 32) - 1) * 32) - 1,
50
+ ...Array(Math.ceil(this.width * this.height / 32) - 1).fill(~0 >>> 0),
51
+ 2 ** (this.width * this.height - (Math.ceil(this.width * this.height / 32) - 1) * 32) - 1,
46
52
  ])
47
- : 2 ** (this.boardWidth * this.boardHeight) - 1;
53
+ : 2 ** (this.width * this.height) - 1);
48
54
  return isFull.equals(fullValue);
49
55
  }
50
- /** Calculates whether or not the board is empty. */
56
+ /**
57
+ * Calculates whether or not the board is empty.
58
+ * @returns Whether or not the board is empty.
59
+ */
51
60
  get isEmpty() {
52
61
  let isEmpty = (this.bitBoard instanceof LongIntBitBoard
53
- ? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))
62
+ ? new LongIntBitBoard(Math.ceil(this.width * this.height / 32))
54
63
  : new IntBitBoard());
55
64
  for (let i = 0; i < this.numberOfPlayerBoards; i++)
56
65
  isEmpty = isEmpty.or(this.getPlayerBoard(i));
@@ -58,9 +67,7 @@ export default class Board {
58
67
  }
59
68
  /**
60
69
  * Calculates who the winner is.
61
- * If the game is not over, the output is false.
62
- * If there is a winner, the output is their ID.
63
- * If there is a draw, the output is null.
70
+ * @returns `false` if the game is not over, the player ID if there is a winner, and `null` if there is a draw.
64
71
  */
65
72
  get winner() {
66
73
  const playerOneBoard = this.getPlayerBoard(0);
@@ -75,35 +82,34 @@ export default class Board {
75
82
  return null;
76
83
  return false;
77
84
  }
78
- /** Calculates which cells are empty. */
85
+ /**
86
+ * Calculates which cells are empty.
87
+ * @returns The empty cells on the board.
88
+ */
79
89
  get emptyCells() {
80
90
  const emptyCells = [];
81
- for (let y = 0; y < this.boardHeight; y++) {
82
- for (let x = 0; x < this.boardWidth; x++) {
91
+ for (let y = 0; y < this.height; y++) {
92
+ for (let x = 0; x < this.width; x++) {
83
93
  if (this.cellOccupier({ x, y }) === null)
84
94
  emptyCells.push({ x, y });
85
95
  }
86
96
  }
87
97
  return emptyCells;
88
98
  }
89
- /**
90
- * Sets the game ID.
91
- * @param id The ID of the game.
92
- */
93
- setGameID(id) {
94
- this.gameID = id;
95
- }
96
99
  /**
97
100
  * Makes a move on the board.
98
- * @param move The position of the move.
99
- * @param playerId The player who's making the move.
101
+ * @param move - The position of the move.
102
+ * @param playerId - The player who's making the move.
100
103
  */
101
104
  makeMove(move, playerId) {
102
105
  const bit = this.getBitIndex(move, playerId);
103
106
  this.moves.push(bit);
104
107
  this.bitBoard.setBit(bit);
105
108
  }
106
- /** Reverses the last move. */
109
+ /**
110
+ * Reverses the last move.
111
+ * @throws {Error} - If there is no move to undo.
112
+ */
107
113
  undoLastMove() {
108
114
  const lastMove = this.moves.pop();
109
115
  if (lastMove === undefined)
@@ -112,7 +118,7 @@ export default class Board {
112
118
  }
113
119
  /**
114
120
  * Checks if a move is valid.
115
- * @param move The move.
121
+ * @param move - The position of the move.
116
122
  * @returns Whether or not it's valid.
117
123
  */
118
124
  moveIsValid(move) {
@@ -120,7 +126,7 @@ export default class Board {
120
126
  }
121
127
  /**
122
128
  * Checks which player is occupying a given cell.
123
- * @param cell The cell to check.
129
+ * @param cell - The cell to check.
124
130
  * @returns If the cell is empty, the output is null, otherwise the output is the player's ID.
125
131
  */
126
132
  cellOccupier(cell) {
@@ -132,12 +138,13 @@ export default class Board {
132
138
  }
133
139
  /**
134
140
  * Creates a string representation of the board.
135
- * @param wrap Whether or not to provide a border for the board.
136
- * @param labelX Whether or not to label x.
137
- * @param labelY Whether or not to label y.
138
- * @param symbols The symbols to use as board pieces.
139
- * @param colour Whether or not to colour the pieces.
141
+ * @param wrap - Whether or not to provide a border for the board.
142
+ * @param labelX - Whether or not to label x.
143
+ * @param labelY - Whether or not to label y.
144
+ * @param symbols - The symbols to use as board pieces.
145
+ * @param colour - Whether or not to colour the pieces.
140
146
  * @returns The string representation.
147
+ * @throws {Error} - If the symbols are not the same length.
141
148
  */
142
149
  toString(wrap = true, labelX = true, labelY = true, symbols = ["X", "O"], colour = true) {
143
150
  if (symbols.length !== this.numberOfPlayerBoards)
@@ -148,39 +155,39 @@ export default class Board {
148
155
  const matchCellSpace = new RegExp(`.{${symbolLength + 2}}`, "gu");
149
156
  const alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz";
150
157
  const xLabels = labelX
151
- ? `${alphabet.slice(0, this.boardWidth)
158
+ ? `${alphabet.slice(0, this.width)
152
159
  .split("")
153
160
  .map((letter) => ` ${letter.padStart(symbolLength)} `)
154
161
  .join("")
155
162
  .match(matchCellSpace)
156
163
  .join(" ")
157
- .padStart(4 * this.boardWidth - 1 + Number(wrap) + Number(labelY))}\n`
164
+ .padStart(4 * this.width - 1 + Number(wrap) + Number(labelY))}\n`
158
165
  : "";
159
166
  const topBorder = wrap
160
167
  ? `${labelY ? " " : ""}${"\u250C" /* GridLines.TopLeft */}${"\u2500" /* GridLines.Horizontal */
161
- .repeat(this.boardWidth * (symbolLength + 2))
168
+ .repeat(this.width * (symbolLength + 2))
162
169
  .match(matchCellSpace)
163
170
  .join("\u252C" /* GridLines.TTop */)}${"\u2510" /* GridLines.TopRight */}\n`
164
171
  : "";
165
172
  const bottomBorder = wrap
166
173
  ? `${labelY ? " " : ""}${"\u2514" /* GridLines.BottomLeft */}${"\u2500" /* GridLines.Horizontal */
167
- .repeat(this.boardWidth * (symbolLength + 2))
174
+ .repeat(this.width * (symbolLength + 2))
168
175
  .match(matchCellSpace)
169
176
  .join("\u2534" /* GridLines.TBottom */)}${"\u2518" /* GridLines.BottomRight */}`
170
177
  : "";
171
178
  const rowSeparator = `${labelY ? " " : ""}${wrap ? "\u251C" /* GridLines.TLeft */ : ""}${"\u2500" /* GridLines.Horizontal */
172
- .repeat(this.boardWidth * (symbolLength + 2))
179
+ .repeat(this.width * (symbolLength + 2))
173
180
  .match(matchCellSpace)
174
181
  .join("\u253C" /* GridLines.Cross */)}${wrap ? "\u2524" /* GridLines.TRight */ : ""}\n`;
175
182
  const rows = [];
176
- for (let y = 0; y < this.boardHeight; y++) {
183
+ for (let y = 0; y < this.height; y++) {
177
184
  const yLabel = labelY ? `${y + 1}` : "";
178
185
  const leftBorder = wrap ? "\u2502" /* GridLines.Vertical */ : "";
179
186
  const rightBorder = wrap ? "\u2502" /* GridLines.Vertical */ : "";
180
187
  let row = `${yLabel}${leftBorder}`;
181
- for (let x = 0; x < this.boardWidth; x++) {
188
+ for (let x = 0; x < this.width; x++) {
182
189
  const cell = { x, y };
183
- const bar = x === this.boardWidth - 1 ? "" : "\u2502" /* GridLines.Vertical */;
190
+ const bar = x === this.width - 1 ? "" : "\u2502" /* GridLines.Vertical */;
184
191
  const cellOccupier = this.cellOccupier(cell);
185
192
  if (cellOccupier === null) {
186
193
  row += ` ${" ".repeat(symbolLength)} ${bar}`;
@@ -197,15 +204,20 @@ export default class Board {
197
204
  }
198
205
  /**
199
206
  * Determines if a given player has a line of pieces on the board.
200
- * @param playerId The ID of the player.
201
- * @param length The number of pieces needed.
202
- * @param maxGaps The number of gaps allowed for a line to be valid. Defaults to 0.
207
+ * @param playerId - The ID of the player to check.
208
+ * @param length - The number of pieces needed.
209
+ * @param maxGaps - The number of gaps allowed for a line to be valid. Defaults to 0.
203
210
  * @returns How many lines exist.
204
211
  */
205
212
  hasLine(playerId, length, maxGaps = 0) {
206
- if (length > Math.max(this.boardWidth, this.boardHeight))
213
+ if (length > Math.max(this.width, this.height))
207
214
  return 0;
208
- const DIRECTIONS = [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: 1, y: 1 }, { x: 1, y: -1 }];
215
+ const DIRECTIONS = [
216
+ { x: 1, y: 0 },
217
+ { x: 0, y: 1 },
218
+ { x: 1, y: 1 },
219
+ { x: 1, y: -1 },
220
+ ];
209
221
  let lineCount = 0;
210
222
  let gaps = [0, 0, 0, 0];
211
223
  let lengths = [0, 0, 0, 0];
@@ -219,8 +231,8 @@ export default class Board {
219
231
  lengths[direction]++;
220
232
  }
221
233
  };
222
- for (let x = 0; x < this.boardWidth; x++) {
223
- for (let y = 0; y < this.boardHeight; y++) {
234
+ for (let x = 0; x < this.width; x++) {
235
+ for (let y = 0; y < this.height; y++) {
224
236
  gaps = [0, 0, 0, 0];
225
237
  lengths = [0, 0, 0, 0];
226
238
  for (let i = 0; i < length; i++) {
@@ -238,41 +250,41 @@ export default class Board {
238
250
  }
239
251
  /**
240
252
  * Gets a bit index from its coordinates and player ID.
241
- * @param move The coordinates.
242
- * @param playerId The player ID.
243
- * @returns The bit index.
253
+ * @param move - The coordinates.
254
+ * @param playerId - The player ID to use.
255
+ * @returns The bit index of the move.
244
256
  */
245
257
  getBitIndex(move, playerId) {
246
258
  const moveIndex = this.getIndex(move);
247
- const bitBoardMoveIndex = moveIndex + this.boardWidth * this.boardHeight * playerId;
259
+ const bitBoardMoveIndex = moveIndex + this.width * this.height * playerId;
248
260
  return bitBoardMoveIndex;
249
261
  }
250
262
  /**
251
263
  * A BitBoard containing only the player's bits.
252
- * @param playerId The player's ID.
264
+ * @param playerId - The player's ID to get the board for.
253
265
  * @returns The player's bits.
254
266
  */
255
267
  getPlayerBoard(playerId) {
256
268
  const totalBits = (this.bitBoard instanceof LongIntBitBoard ? this.bitBoard.data.wordCount : 1) * 32;
257
- const boardSize = this.boardWidth * this.boardHeight;
269
+ const boardSize = this.width * this.height;
258
270
  return this.bitBoard.leftShift(totalBits - boardSize * (playerId + 1)).rightShift(totalBits - boardSize);
259
271
  }
260
272
  /**
261
273
  * Gets a bit index from its coordinates.
262
- * @param move The coordinates.
274
+ * @param move - The coordinates.
263
275
  * @returns The bit index.
264
276
  */
265
277
  getIndex(move) {
266
- return this.boardWidth * move.y + move.x;
278
+ return this.width * move.y + move.x;
267
279
  }
268
280
  /**
269
281
  * Checks if a move is valid for the given board.
270
282
  * Does not check if that cell is already occupied.
271
- * @param position The position to check.
283
+ * @param position - The position to check.
272
284
  * @returns Whether or not that cell exists on the board.
273
285
  */
274
286
  isValidPosition(position) {
275
- return position.x >= 0 && position.x < this.boardWidth && position.y >= 0 && position.y < this.boardHeight;
287
+ return position.x >= 0 && position.x < this.width && position.y >= 0 && position.y < this.height;
276
288
  }
277
289
  }
278
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.js","sourceRoot":"","sources":["../../src/base/board.ts"],"names":[],"mappings":"AACA,OAAO,WAAW,MAAM,4BAA4B,CAAC;AACrD,OAAO,OAAO,MAAM,wBAAwB,CAAC;AAC7C,OAAO,eAAe,MAAM,gCAAgC,CAAC;AAsB7D,+BAA+B;AAC/B,MAAM,CAAC,OAAO,OAAgB,KAAK;IAC/B,mCAAmC;IAC5B,MAAM,GAAY,EAAE,CAAC;IAE5B,8CAA8C;IAC3B,QAAQ,CAAI;IAE/B,8BAA8B;IACX,UAAU,CAAS;IAEtC,+BAA+B;IACZ,WAAW,CAAS;IAEvC,wBAAwB;IACL,KAAK,GAAa,EAAE,CAAC;IAExC,+EAA+E;IAC9D,oBAAoB,CAAS;IAE9C,sDAAsD;IACrC,cAAc,CAAS;IAKxC;;;;;;OAMG;IACH,YAAsB,KAAa,EAAE,MAAc,EAAE,mBAA2B,CAAC,EAAE,kBAA0B,CAAC;QAC1G,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;QACxB,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC;QAC1B,IAAI,CAAC,oBAAoB,GAAG,gBAAgB,CAAC;QAC7C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,oBAAoB,GAAG,eAAe,CAAC;QAClE,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,cAAc,CAAC;QAE3E,IAAI,CAAC,QAAQ,GAAG,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC,CAAC,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,WAAW,EAAE,CAAM,CAAC;IAC/G,CAAC;IAED,mDAAmD;IACnD,IAAW,MAAM;QACb,IAAI,MAAM,GAAG,CAAC,IAAI,CAAC,QAAQ,YAAY,eAAe;YAClD,CAAC,CAAC,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC,CAAC;YACzE,CAAC,CAAC,IAAI,WAAW,EAAE,CAAM,CAAC;QAE9B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC,EAAE;YAC9C,MAAM,GAAG,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/C,MAAM,SAAS,GAAG,IAAI,CAAC,QAAQ,YAAY,eAAe;YACtD,CAAC,CAAC,IAAI,OAAO,CAAC;gBACV,GAAG,KAAK,CAAS,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;gBACvF,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC;aAChH,CAAC;YACF,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;QAEpD,OAAO,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACpC,CAAC;IAED,oDAAoD;IACpD,IAAW,OAAO;QACd,IAAI,OAAO,GAAG,CAAC,IAAI,CAAC,QAAQ,YAAY,eAAe;YACnD,CAAC,CAAC,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC,CAAC;YACzE,CAAC,CAAC,IAAI,WAAW,EAAE,CAAM,CAAC;QAE9B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC,EAAE;YAC9C,OAAO,GAAG,OAAO,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;QAEjD,OAAO,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;IAC7B,CAAC;IAED;;;;;OAKG;IACH,IAAW,MAAM;QACb,MAAM,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC;QAC9C,MAAM,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC;QAE9C,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACrC,IAAI,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;gBACvC,OAAO,CAAC,CAAC;YAEb,IAAI,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;gBACvC,OAAO,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,IAAI,CAAC,MAAM;YACX,OAAO,IAAI,CAAC;QAEhB,OAAO,KAAK,CAAC;IACjB,CAAC;IAED,wCAAwC;IACxC,IAAW,UAAU;QACjB,MAAM,UAAU,GAAe,EAAE,CAAC;QAElC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,EAAE,EAAE,CAAC;YACxC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;gBACvC,IAAI,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,KAAK,IAAI;oBACpC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YAClC,CAAC;QACL,CAAC;QAED,OAAO,UAAU,CAAC;IACtB,CAAC;IAKD;;;OAGG;IACI,SAAS,CAAC,EAAW;QACxB,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC;IACrB,CAAC;IAED;;;;OAIG;IACI,QAAQ,CAAC,IAAc,EAAE,QAAgB;QAC5C,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAE7C,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IAC9B,CAAC;IAED,8BAA8B;IACvB,YAAY;QACf,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC;QAElC,IAAI,QAAQ,KAAK,SAAS;YACtB,MAAM,IAAI,KAAK,CAAC,kBAAkB,CAAC,CAAC;QAExC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IACrC,CAAC;IAED;;;;OAIG;IACI,WAAW,CAAC,IAAc;QAC7B,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC;IAC1E,CAAC;IAED;;;;OAIG;IACI,YAAY,CAAC,IAAc;QAC9B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC,EAAE,EAAE,CAAC;YACjD,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC;gBACrD,OAAO,CAAC,CAAC;QACjB,CAAC;QAED,OAAO,IAAI,CAAC;IAChB,CAAC;IAED;;;;;;;;OAQG;IACI,QAAQ,CACX,OAAgB,IAAI,EACpB,SAAkB,IAAI,EACtB,SAAkB,IAAI,EACtB,UAAoB,CAAC,GAAG,EAAE,GAAG,CAAC,EAC9B,SAAkB,IAAI;QAEtB,IAAI,OAAO,CAAC,MAAM,KAAK,IAAI,CAAC,oBAAoB;YAC5C,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;QAEzC,MAAM,YAAY,GAAG,OAAO,CAAC,CAAC,CAAE,CAAC,MAAM,CAAC;QAExC,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,YAAY,CAAC;YAC9C,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,CAAC;QAExD,MAAM,cAAc,GAAG,IAAI,MAAM,CAAC,KAAK,YAAY,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAClE,MAAM,QAAQ,GAAG,sDAAsD,CAAC;QACxE,MAAM,OAAO,GAAG,MAAM;YAClB,CAAC,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,UAAU,CAAC;iBAClC,KAAK,CAAC,EAAE,CAAC;iBACT,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,IAAI,MAAM,CAAC,QAAQ,CAAC,YAAY,CAAC,GAAG,CAAC;iBACrD,IAAI,CAAC,EAAE,CAAC;iBACR,KAAK,CAAC,cAAc,CAAE;iBACtB,IAAI,CAAC,GAAG,CAAC;iBACT,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,UAAU,GAAG,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC,CACrE,IAAI;YACJ,CAAC,CAAC,EAAE,CAAC;QACT,MAAM,SAAS,GAAG,IAAI;YAClB,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,gCAAiB,GAAG;iBACxC,MAAM,CAAC,IAAI,CAAC,UAAU,GAAG,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;iBAC5C,KAAK,CAAC,cAAc,CAAE;iBACtB,IAAI,+BAAgB,GAAG,iCAAkB,IAAI;YAClD,CAAC,CAAC,EAAE,CAAC;QACT,MAAM,YAAY,GAAG,IAAI;YACrB,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,mCAAoB,GAAG;iBAC3C,MAAM,CAAC,IAAI,CAAC,UAAU,GAAG,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;iBAC5C,KAAK,CAAC,cAAc,CAAE;iBACtB,IAAI,kCAAmB,GAAG,oCAAqB,EAAE;YACtD,CAAC,CAAC,EAAE,CAAC;QACT,MAAM,YAAY,GAAG,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC,gCAAiB,CAAC,CAAC,EAAE,GAAG;aACrE,MAAM,CAAC,IAAI,CAAC,UAAU,GAAG,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;aAC5C,KAAK,CAAC,cAAc,CAAE;aACtB,IAAI,gCAAiB,GAAG,IAAI,CAAC,CAAC,iCAAkB,CAAC,CAAC,EAAE,IAAI,CAAC;QAC9D,MAAM,IAAI,GAAa,EAAE,CAAC;QAE1B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,EAAE,EAAE,CAAC;YACxC,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACxC,MAAM,UAAU,GAAG,IAAI,CAAC,CAAC,mCAAoB,CAAC,CAAC,EAAE,CAAC;YAClD,MAAM,WAAW,GAAG,IAAI,CAAC,CAAC,mCAAoB,CAAC,CAAC,EAAE,CAAC;YACnD,IAAI,GAAG,GAAG,GAAG,MAAM,GAAG,UAAU,EAAE,CAAC;YAEnC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;gBACvC,MAAM,IAAI,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC;gBACtB,MAAM,GAAG,GAAG,CAAC,KAAK,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,kCAAmB,CAAC;gBAChE,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;gBAE7C,IAAI,YAAY,KAAK,IAAI,EAAE,CAAC;oBACxB,GAAG,IAAI,IAAI,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,GAAG,EAAE,CAAC;gBACjD,CAAC;qBAAM,CAAC;oBACJ,GAAG,IAAI,IAAI,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,YAAY,CAAE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;wBACzD,GAAG,OAAO,CAAC,YAAY,CAAC,EAAE;wBAC1B,GAAG,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,IAAI,GAAG,EAAE,CAAC;gBAC5C,CAAC;YACL,CAAC;YACD,IAAI,CAAC,IAAI,CAAC,GAAG,GAAG,GAAG,WAAW,IAAI,CAAC,CAAC;QACxC,CAAC;QAED,OAAO,GAAG,OAAO,GAAG,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,YAAY,EAAE,CAAC;IAC7E,CAAC;IAED;;;;;;OAMG;IACI,OAAO,CAAC,QAAgB,EAAE,MAAc,EAAE,UAAkB,CAAC;QAChE,IAAI,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC;YACpD,OAAO,CAAC,CAAC;QAEb,MAAM,UAAU,GAA6C,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;QAC/H,IAAI,SAAS,GAAG,CAAC,CAAC;QAClB,IAAI,IAAI,GAAqC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAC1D,IAAI,OAAO,GAAqC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAC7D,MAAM,SAAS,GAAG,CAAC,CAAS,EAAE,CAAS,EAAE,SAAwB,EAAQ,EAAE;YACvE,MAAM,IAAI,GAAG,EAAE,CAAC,EAAE,CAAC,EAAc,CAAC;YAElC,IAAI,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC7B,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;gBAE7C,IAAI,YAAY,KAAK,IAAI;oBACrB,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC;qBACjB,IAAI,YAAY,KAAK,QAAQ;oBAC9B,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;YAC7B,CAAC;QACL,CAAC,CAAC;QAEF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;YACvC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,EAAE,EAAE,CAAC;gBACxC,IAAI,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;gBACpB,OAAO,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;gBACvB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC9B,KAAK,IAAI,CAAC,GAAG,CAAkB,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;wBAC1C,IAAI,IAAI,CAAC,CAAC,CAAC,GAAG,OAAO;4BACjB,SAAS;wBAEb,SAAS,CAAC,CAAC,GAAG,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;wBAE/D,IAAI,OAAO,CAAC,CAAC,CAAC,KAAK,MAAM;4BACrB,SAAS,EAAE,CAAC;oBACpB,CAAC;gBACL,CAAC;YACL,CAAC;QACL,CAAC;QAED,OAAO,SAAS,CAAC;IACrB,CAAC;IAED;;;;;OAKG;IACO,WAAW,CAAC,IAAc,EAAE,QAAgB;QAClD,MAAM,SAAS,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,iBAAiB,GAAG,SAAS,GAAG,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,GAAG,QAAQ,CAAC;QAEpF,OAAO,iBAAiB,CAAC;IAC7B,CAAC;IAED;;;;OAIG;IACO,cAAc,CAAC,QAAgB;QACrC,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,QAAQ,YAAY,eAAe,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC;QACrG,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC;QAErD,OAAO,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,SAAS,GAAG,SAAS,GAAG,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,SAAS,GAAG,SAAS,CAAC,CAAC;IAC7G,CAAC;IAED;;;;OAIG;IACK,QAAQ,CAAC,IAAc;QAC3B,OAAO,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC;IAC7C,CAAC;IAED;;;;;OAKG;IACK,eAAe,CAAC,QAAkB;QACtC,OAAO,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,UAAU,IAAI,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,WAAW,CAAC;IAC/G,CAAC;CACJ","sourcesContent":["import type BitBoard from \"../bitBoard/bitBoard.js\";\nimport IntBitBoard from \"../bitBoard/intBitBoard.js\";\nimport LongInt from \"../bitBoard/longInt.js\";\nimport LongIntBitBoard from \"../bitBoard/longIntBitBoard.js\";\n\nexport type Position = {\n    y: number;\n    x: number;\n};\n\n/** Defines the characters used to draw a grid. */\nexport const enum GridLines {\n    Horizontal = \"\\u2500\",\n    Vertical = \"\\u2502\",\n    TopLeft = \"\\u250C\",\n    TopRight = \"\\u2510\",\n    BottomLeft = \"\\u2514\",\n    BottomRight = \"\\u2518\",\n    TLeft = \"\\u251C\",\n    TRight = \"\\u2524\",\n    TTop = \"\\u252C\",\n    TBottom = \"\\u2534\",\n    Cross = \"\\u253C\",\n}\n\n/** Represents a game board. */\nexport default abstract class Board<T extends BitBoard = BitBoard> {\n    /** Contains the ID of the game. */\n    public gameID: unknown = \"\";\n\n    /** Contains the data stored in a BitBoard. */\n    protected readonly bitBoard: T;\n\n    /** The width of the board. */\n    protected readonly boardWidth: number;\n\n    /** The height of the board. */\n    protected readonly boardHeight: number;\n\n    /** A stack of moves. */\n    protected readonly moves: number[] = [];\n\n    /** How many boards there are representing player positions (most likely 2). */\n    private readonly numberOfPlayerBoards: number;\n\n    /** Number of boards in total (most likely also 2). */\n    private readonly numberOfBoards: number;\n\n    /** The board states which represent a winning state. */\n    protected abstract readonly winningStates: T[];\n\n    /**\n     * Creates an instance of Board.\n     * @param width The width of the board.\n     * @param height The height of the board.\n     * @param playerBoardCount How many boards there are representing player positions (most likely 2).\n     * @param extraBoardCount Number of extra boards (most likely 0).\n     */\n    protected constructor(width: number, height: number, playerBoardCount: number = 2, extraBoardCount: number = 0) {\n        this.boardWidth = width;\n        this.boardHeight = height;\n        this.numberOfPlayerBoards = playerBoardCount;\n        this.numberOfBoards = this.numberOfPlayerBoards + extraBoardCount;\n        const totalBits = this.boardWidth * this.boardHeight * this.numberOfBoards;\n\n        this.bitBoard = (totalBits > 32 ? new LongIntBitBoard(Math.ceil(totalBits / 32)) : new IntBitBoard()) as T;\n    }\n\n    /** Calculates whether or not the board is full. */\n    public get isFull(): boolean {\n        let isFull = (this.bitBoard instanceof LongIntBitBoard\n            ? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))\n            : new IntBitBoard()) as T;\n\n        for (let i = 0; i < this.numberOfPlayerBoards; i++)\n            isFull = isFull.or(this.getPlayerBoard(i));\n        const fullValue = this.bitBoard instanceof LongIntBitBoard\n            ? new LongInt([\n                ...Array<number>(Math.ceil(this.boardWidth * this.boardHeight / 32) - 1).fill(~0 >>> 0),\n                2 ** (this.boardWidth * this.boardHeight - (Math.ceil(this.boardWidth * this.boardHeight / 32) - 1) * 32) - 1,\n            ])\n            : 2 ** (this.boardWidth * this.boardHeight) - 1;\n\n        return isFull.equals(fullValue);\n    }\n\n    /** Calculates whether or not the board is empty. */\n    public get isEmpty(): boolean {\n        let isEmpty = (this.bitBoard instanceof LongIntBitBoard\n            ? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))\n            : new IntBitBoard()) as T;\n\n        for (let i = 0; i < this.numberOfPlayerBoards; i++)\n            isEmpty = isEmpty.or(this.getPlayerBoard(i));\n\n        return isEmpty.equals(0);\n    }\n\n    /**\n     * Calculates who the winner is.\n     * If the game is not over, the output is false.\n     * If there is a winner, the output is their ID.\n     * If there is a draw, the output is null.\n     */\n    public get winner(): 0 | 1 | false | null {\n        const playerOneBoard = this.getPlayerBoard(0);\n        const playerTwoBoard = this.getPlayerBoard(1);\n\n        for (const state of this.winningStates) {\n            if (playerOneBoard.and(state).equals(state))\n                return 0;\n\n            if (playerTwoBoard.and(state).equals(state))\n                return 1;\n        }\n\n        if (this.isFull)\n            return null;\n\n        return false;\n    }\n\n    /** Calculates which cells are empty. */\n    public get emptyCells(): Position[] {\n        const emptyCells: Position[] = [];\n\n        for (let y = 0; y < this.boardHeight; y++) {\n            for (let x = 0; x < this.boardWidth; x++) {\n                if (this.cellOccupier({ x, y }) === null)\n                    emptyCells.push({ x, y });\n            }\n        }\n\n        return emptyCells;\n    }\n\n    /** Calculates the heuristic score for a given board state. */\n    public abstract get heuristic(): number;\n\n    /**\n     * Sets the game ID.\n     * @param id The ID of the game.\n     */\n    public setGameID(id: unknown): void {\n        this.gameID = id;\n    }\n\n    /**\n     * Makes a move on the board.\n     * @param move The position of the move.\n     * @param playerId The player who's making the move.\n     */\n    public makeMove(move: Position, playerId: number): void {\n        const bit = this.getBitIndex(move, playerId);\n\n        this.moves.push(bit);\n        this.bitBoard.setBit(bit);\n    }\n\n    /** Reverses the last move. */\n    public undoLastMove(): void {\n        const lastMove = this.moves.pop();\n\n        if (lastMove === undefined)\n            throw new Error(\"No move to undo.\");\n\n        this.bitBoard.clearBit(lastMove);\n    }\n\n    /**\n     * Checks if a move is valid.\n     * @param move The move.\n     * @returns Whether or not it's valid.\n     */\n    public moveIsValid(move: Position): boolean {\n        return this.isValidPosition(move) && this.cellOccupier(move) === null;\n    }\n\n    /**\n     * Checks which player is occupying a given cell.\n     * @param cell The cell to check.\n     * @returns If the cell is empty, the output is null, otherwise the output is the player's ID.\n     */\n    public cellOccupier(cell: Position): number | null {\n        for (let i = 0; i < this.numberOfPlayerBoards; i++) {\n            if (this.bitBoard.getBit(this.getBitIndex(cell, i)) === 1)\n                return i;\n        }\n\n        return null;\n    }\n\n    /**\n     * Creates a string representation of the board.\n     * @param wrap Whether or not to provide a border for the board.\n     * @param labelX Whether or not to label x.\n     * @param labelY Whether or not to label y.\n     * @param symbols The symbols to use as board pieces.\n     * @param colour Whether or not to colour the pieces.\n     * @returns The string representation.\n     */\n    public toString(\n        wrap: boolean = true,\n        labelX: boolean = true,\n        labelY: boolean = true,\n        symbols: string[] = [\"X\", \"O\"],\n        colour: boolean = true,\n    ): string {\n        if (symbols.length !== this.numberOfPlayerBoards)\n            throw new Error(\"Too many symbols.\");\n\n        const symbolLength = symbols[0]!.length;\n\n        if (symbols.some((s) => s.length !== symbolLength))\n            throw new Error(\"Symbols must be the same length.\");\n\n        const matchCellSpace = new RegExp(`.{${symbolLength + 2}}`, \"gu\");\n        const alphabet = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\";\n        const xLabels = labelX\n            ? `${alphabet.slice(0, this.boardWidth)\n                .split(\"\")\n                .map((letter) => ` ${letter.padStart(symbolLength)} `)\n                .join(\"\")\n                .match(matchCellSpace)!\n                .join(\" \")\n                .padStart(4 * this.boardWidth - 1 + Number(wrap) + Number(labelY))\n            }\\n`\n            : \"\";\n        const topBorder = wrap\n            ? `${labelY ? \" \" : \"\"}${GridLines.TopLeft}${GridLines.Horizontal\n                .repeat(this.boardWidth * (symbolLength + 2))\n                .match(matchCellSpace)!\n                .join(GridLines.TTop)}${GridLines.TopRight}\\n`\n            : \"\";\n        const bottomBorder = wrap\n            ? `${labelY ? \" \" : \"\"}${GridLines.BottomLeft}${GridLines.Horizontal\n                .repeat(this.boardWidth * (symbolLength + 2))\n                .match(matchCellSpace)!\n                .join(GridLines.TBottom)}${GridLines.BottomRight}`\n            : \"\";\n        const rowSeparator = `${labelY ? \" \" : \"\"}${wrap ? GridLines.TLeft : \"\"}${GridLines.Horizontal\n            .repeat(this.boardWidth * (symbolLength + 2))\n            .match(matchCellSpace)!\n            .join(GridLines.Cross)}${wrap ? GridLines.TRight : \"\"}\\n`;\n        const rows: string[] = [];\n\n        for (let y = 0; y < this.boardHeight; y++) {\n            const yLabel = labelY ? `${y + 1}` : \"\";\n            const leftBorder = wrap ? GridLines.Vertical : \"\";\n            const rightBorder = wrap ? GridLines.Vertical : \"\";\n            let row = `${yLabel}${leftBorder}`;\n\n            for (let x = 0; x < this.boardWidth; x++) {\n                const cell = { x, y };\n                const bar = x === this.boardWidth - 1 ? \"\" : GridLines.Vertical;\n                const cellOccupier = this.cellOccupier(cell);\n\n                if (cellOccupier === null) {\n                    row += ` ${\" \".repeat(symbolLength)} ${bar}`;\n                } else {\n                    row += ` ${colour ? `\\x1b[${[91, 93][cellOccupier]!}m` : \"\"}` +\n                        `${symbols[cellOccupier]}` +\n                        `${colour ? \"\\x1b[0m\" : \"\"} ${bar}`;\n                }\n            }\n            rows.push(`${row}${rightBorder}\\n`);\n        }\n\n        return `${xLabels}${topBorder}${rows.join(rowSeparator)}${bottomBorder}`;\n    }\n\n    /**\n     * Determines if a given player has a line of pieces on the board.\n     * @param playerId The ID of the player.\n     * @param length The number of pieces needed.\n     * @param maxGaps The number of gaps allowed for a line to be valid. Defaults to 0.\n     * @returns How many lines exist.\n     */\n    public hasLine(playerId: number, length: number, maxGaps: number = 0): number {\n        if (length > Math.max(this.boardWidth, this.boardHeight))\n            return 0;\n\n        const DIRECTIONS: [Position, Position, Position, Position] = [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: 1, y: 1 }, { x: 1, y: -1 }];\n        let lineCount = 0;\n        let gaps: [number, number, number, number] = [0, 0, 0, 0];\n        let lengths: [number, number, number, number] = [0, 0, 0, 0];\n        const checkCell = (x: number, y: number, direction: 0 | 1 | 2 | 3): void => {\n            const cell = { x, y } as Position;\n\n            if (this.isValidPosition(cell)) {\n                const cellOccupier = this.cellOccupier(cell);\n\n                if (cellOccupier === null)\n                    gaps[direction]++;\n                else if (cellOccupier === playerId)\n                    lengths[direction]++;\n            }\n        };\n\n        for (let x = 0; x < this.boardWidth; x++) {\n            for (let y = 0; y < this.boardHeight; y++) {\n                gaps = [0, 0, 0, 0];\n                lengths = [0, 0, 0, 0];\n                for (let i = 0; i < length; i++) {\n                    for (let j = 0 as 0 | 1 | 2 | 3; j < 4; j++) {\n                        if (gaps[j] > maxGaps)\n                            continue;\n\n                        checkCell(x + i * DIRECTIONS[j].x, y + i * DIRECTIONS[j].y, j);\n\n                        if (lengths[j] === length)\n                            lineCount++;\n                    }\n                }\n            }\n        }\n\n        return lineCount;\n    }\n\n    /**\n     * Gets a bit index from its coordinates and player ID.\n     * @param move The coordinates.\n     * @param playerId The player ID.\n     * @returns The bit index.\n     */\n    protected getBitIndex(move: Position, playerId: number): number {\n        const moveIndex = this.getIndex(move);\n        const bitBoardMoveIndex = moveIndex + this.boardWidth * this.boardHeight * playerId;\n\n        return bitBoardMoveIndex;\n    }\n\n    /**\n     * A BitBoard containing only the player's bits.\n     * @param playerId The player's ID.\n     * @returns The player's bits.\n     */\n    protected getPlayerBoard(playerId: number): T {\n        const totalBits = (this.bitBoard instanceof LongIntBitBoard ? this.bitBoard.data.wordCount : 1) * 32;\n        const boardSize = this.boardWidth * this.boardHeight;\n\n        return this.bitBoard.leftShift(totalBits - boardSize * (playerId + 1)).rightShift(totalBits - boardSize);\n    }\n\n    /**\n     * Gets a bit index from its coordinates.\n     * @param move The coordinates.\n     * @returns The bit index.\n     */\n    private getIndex(move: Position): number {\n        return this.boardWidth * move.y + move.x;\n    }\n\n    /**\n     * Checks if a move is valid for the given board.\n     * Does not check if that cell is already occupied.\n     * @param position The position to check.\n     * @returns Whether or not that cell exists on the board.\n     */\n    private isValidPosition(position: Position): boolean {\n        return position.x >= 0 && position.x < this.boardWidth && position.y >= 0 && position.y < this.boardHeight;\n    }\n}\n"]}
290
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.js","sourceRoot":"","sources":["../../src/base/board.ts"],"names":[],"mappings":"AACA,OAAO,WAAW,MAAM,4BAA4B,CAAC;AACrD,OAAO,OAAO,MAAM,wBAAwB,CAAC;AAC7C,OAAO,eAAe,MAAM,gCAAgC,CAAC;AAsB7D;;;GAGG;AACH,MAAM,CAAC,OAAO,OAAgB,KAAK;IAC/B,8CAA8C;IAC3B,QAAQ,CAAc;IAEzC,mCAAmC;IAChB,KAAK,CAAS;IAEjC,oCAAoC;IACjB,MAAM,CAAS;IAElC,wBAAwB;IACL,KAAK,GAAa,EAAE,CAAC;IAExC,+EAA+E;IAC9D,oBAAoB,CAAS;IAE9C,sDAAsD;IACrC,cAAc,CAAS;IAKxC;;;;;;OAMG;IACH,YAAsB,KAAa,EAAE,MAAc,EAAE,mBAA2B,CAAC,EAAE,kBAA0B,CAAC;QAC1G,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,IAAI,CAAC,oBAAoB,GAAG,gBAAgB,CAAC;QAC7C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,oBAAoB,GAAG,eAAe,CAAC;QAClE,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC;QAEjE,IAAI,CAAC,QAAQ,GAAG,CAAC,SAAS,GAAG,EAAE;YAC3B,CAAC,CAAC,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC;YAChD,CAAC,CAAC,IAAI,WAAW,EAAE,CAAgB,CAAC;IAC5C,CAAC;IAED;;;OAGG;IACH,IAAW,MAAM;QACb,IAAI,MAAM,GAAG,CAAC,IAAI,CAAC,QAAQ,YAAY,eAAe;YAClD,CAAC,CAAC,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,CAAC;YAC/D,CAAC,CAAC,IAAI,WAAW,EAAE,CAAgB,CAAC;QAExC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC,EAAE;YAC9C,MAAM,GAAG,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/C,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,QAAQ,YAAY,eAAe;YACvD,CAAC,CAAC,IAAI,OAAO,CAAC;gBACV,GAAG,KAAK,CAAS,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;gBAC7E,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC;aAC5F,CAAC;YACF,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAM,CAAC;QAEhD,OAAO,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACpC,CAAC;IAED;;;OAGG;IACH,IAAW,OAAO;QACd,IAAI,OAAO,GAAG,CAAC,IAAI,CAAC,QAAQ,YAAY,eAAe;YACnD,CAAC,CAAC,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC,CAAC;YAC/D,CAAC,CAAC,IAAI,WAAW,EAAE,CAAgB,CAAC;QAExC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC,EAAE;YAC9C,OAAO,GAAG,OAAO,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;QAEjD,OAAO,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;IAC7B,CAAC;IAED;;;OAGG;IACH,IAAW,MAAM;QACb,MAAM,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC;QAC9C,MAAM,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC;QAE9C,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACrC,IAAI,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;gBACvC,OAAO,CAAC,CAAC;YAEb,IAAI,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;gBACvC,OAAO,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,IAAI,CAAC,MAAM;YACX,OAAO,IAAI,CAAC;QAEhB,OAAO,KAAK,CAAC;IACjB,CAAC;IAED;;;OAGG;IACH,IAAW,UAAU;QACjB,MAAM,UAAU,GAAe,EAAE,CAAC;QAElC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YACnC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC;gBAClC,IAAI,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,KAAK,IAAI;oBACpC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YAClC,CAAC;QACL,CAAC;QAED,OAAO,UAAU,CAAC;IACtB,CAAC;IAKD;;;;OAIG;IACI,QAAQ,CAAC,IAAc,EAAE,QAAgB;QAC5C,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAE7C,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IAC9B,CAAC;IAED;;;OAGG;IACI,YAAY;QACf,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC;QAElC,IAAI,QAAQ,KAAK,SAAS;YACtB,MAAM,IAAI,KAAK,CAAC,kBAAkB,CAAC,CAAC;QAExC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IACrC,CAAC;IAED;;;;OAIG;IACI,WAAW,CAAC,IAAc;QAC7B,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC;IAC1E,CAAC;IAED;;;;OAIG;IACI,YAAY,CAAC,IAAc;QAC9B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC,EAAE,EAAE,CAAC;YACjD,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC;gBACrD,OAAO,CAAC,CAAC;QACjB,CAAC;QAED,OAAO,IAAI,CAAC;IAChB,CAAC;IAED;;;;;;;;;OASG;IACI,QAAQ,CACX,OAAgB,IAAI,EACpB,SAAkB,IAAI,EACtB,SAAkB,IAAI,EACtB,UAAoB,CAAC,GAAG,EAAE,GAAG,CAAC,EAC9B,SAAkB,IAAI;QAEtB,IAAI,OAAO,CAAC,MAAM,KAAK,IAAI,CAAC,oBAAoB;YAC5C,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;QAEzC,MAAM,YAAY,GAAG,OAAO,CAAC,CAAC,CAAE,CAAC,MAAM,CAAC;QAExC,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,YAAY,CAAC;YAC9C,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,CAAC;QAExD,MAAM,cAAc,GAAG,IAAI,MAAM,CAAC,KAAK,YAAY,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAClE,MAAM,QAAQ,GAAG,sDAAsD,CAAC;QACxE,MAAM,OAAO,GAAG,MAAM;YAClB,CAAC,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC;iBAC7B,KAAK,CAAC,EAAE,CAAC;iBACT,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,IAAI,MAAM,CAAC,QAAQ,CAAC,YAAY,CAAC,GAAG,CAAC;iBACrD,IAAI,CAAC,EAAE,CAAC;iBACR,KAAK,CAAC,cAAc,CAAE;iBACtB,IAAI,CAAC,GAAG,CAAC;iBACT,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,GAAG,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC,CAChE,IAAI;YACJ,CAAC,CAAC,EAAE,CAAC;QACT,MAAM,SAAS,GAAG,IAAI;YAClB,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,gCAAiB,GAAG;iBACxC,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;iBACvC,KAAK,CAAC,cAAc,CAAE;iBACtB,IAAI,+BAAgB,GAAG,iCAAkB,IAAI;YAClD,CAAC,CAAC,EAAE,CAAC;QACT,MAAM,YAAY,GAAG,IAAI;YACrB,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,mCAAoB,GAAG;iBAC3C,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;iBACvC,KAAK,CAAC,cAAc,CAAE;iBACtB,IAAI,kCAAmB,GAAG,oCAAqB,EAAE;YACtD,CAAC,CAAC,EAAE,CAAC;QACT,MAAM,YAAY,GAAG,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,IAAI,CAAC,CAAC,gCAAiB,CAAC,CAAC,EAAE,GAAG;aACrE,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;aACvC,KAAK,CAAC,cAAc,CAAE;aACtB,IAAI,gCAAiB,GAAG,IAAI,CAAC,CAAC,iCAAkB,CAAC,CAAC,EAAE,IAAI,CAAC;QAC9D,MAAM,IAAI,GAAa,EAAE,CAAC;QAE1B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YACnC,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACxC,MAAM,UAAU,GAAG,IAAI,CAAC,CAAC,mCAAoB,CAAC,CAAC,EAAE,CAAC;YAClD,MAAM,WAAW,GAAG,IAAI,CAAC,CAAC,mCAAoB,CAAC,CAAC,EAAE,CAAC;YACnD,IAAI,GAAG,GAAG,GAAG,MAAM,GAAG,UAAU,EAAE,CAAC;YAEnC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC;gBAClC,MAAM,IAAI,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC;gBACtB,MAAM,GAAG,GAAG,CAAC,KAAK,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,kCAAmB,CAAC;gBAC3D,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;gBAE7C,IAAI,YAAY,KAAK,IAAI,EAAE,CAAC;oBACxB,GAAG,IAAI,IAAI,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,IAAI,GAAG,EAAE,CAAC;gBACjD,CAAC;qBAAM,CAAC;oBACJ,GAAG,IAAI,IAAI,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,YAAY,CAAE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;wBACzD,GAAG,OAAO,CAAC,YAAY,CAAC,EAAE;wBAC1B,GAAG,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,IAAI,GAAG,EAAE,CAAC;gBAC5C,CAAC;YACL,CAAC;YACD,IAAI,CAAC,IAAI,CAAC,GAAG,GAAG,GAAG,WAAW,IAAI,CAAC,CAAC;QACxC,CAAC;QAED,OAAO,GAAG,OAAO,GAAG,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,YAAY,EAAE,CAAC;IAC7E,CAAC;IAED;;;;;;OAMG;IACI,OAAO,CAAC,QAAgB,EAAE,MAAc,EAAE,UAAkB,CAAC;QAChE,IAAI,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC;YAC1C,OAAO,CAAC,CAAC;QAEb,MAAM,UAAU,GAA6C;YACzD,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE;YACd,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE;YACd,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE;YACd,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,EAAE;SAClB,CAAC;QACF,IAAI,SAAS,GAAG,CAAC,CAAC;QAClB,IAAI,IAAI,GAAqC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAC1D,IAAI,OAAO,GAAqC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAC7D,MAAM,SAAS,GAAG,CAAC,CAAS,EAAE,CAAS,EAAE,SAAwB,EAAQ,EAAE;YACvE,MAAM,IAAI,GAAG,EAAE,CAAC,EAAE,CAAC,EAAc,CAAC;YAElC,IAAI,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC7B,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;gBAE7C,IAAI,YAAY,KAAK,IAAI;oBACrB,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC;qBACjB,IAAI,YAAY,KAAK,QAAQ;oBAC9B,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;YAC7B,CAAC;QACL,CAAC,CAAC;QAEF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC;YAClC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACnC,IAAI,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;gBACpB,OAAO,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;gBACvB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC9B,KAAK,IAAI,CAAC,GAAG,CAAkB,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;wBAC1C,IAAI,IAAI,CAAC,CAAC,CAAC,GAAG,OAAO;4BACjB,SAAS;wBAEb,SAAS,CAAC,CAAC,GAAG,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;wBAE/D,IAAI,OAAO,CAAC,CAAC,CAAC,KAAK,MAAM;4BACrB,SAAS,EAAE,CAAC;oBACpB,CAAC;gBACL,CAAC;YACL,CAAC;QACL,CAAC;QAED,OAAO,SAAS,CAAC;IACrB,CAAC;IAED;;;;;OAKG;IACO,WAAW,CAAC,IAAc,EAAE,QAAgB;QAClD,MAAM,SAAS,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,iBAAiB,GAAG,SAAS,GAAG,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,GAAG,QAAQ,CAAC;QAE1E,OAAO,iBAAiB,CAAC;IAC7B,CAAC;IAED;;;;OAIG;IACO,cAAc,CAAC,QAAgB;QACrC,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,QAAQ,YAAY,eAAe,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC;QACrG,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC;QAE3C,OAAO,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,SAAS,GAAG,SAAS,GAAG,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,SAAS,GAAG,SAAS,CAAC,CAAC;IAC7G,CAAC;IAED;;;;OAIG;IACK,QAAQ,CAAC,IAAc;QAC3B,OAAO,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC;IACxC,CAAC;IAED;;;;;OAKG;IACK,eAAe,CAAC,QAAkB;QACtC,OAAO,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,IAAI,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC;IACrG,CAAC;CACJ","sourcesContent":["import type BitBoard from \"../bitBoard/bitBoard.js\";\nimport IntBitBoard from \"../bitBoard/intBitBoard.js\";\nimport LongInt from \"../bitBoard/longInt.js\";\nimport LongIntBitBoard from \"../bitBoard/longIntBitBoard.js\";\n\nexport type Position = {\n    y: number;\n    x: number;\n};\n\n/** Defines the characters used to draw a grid. */\nexport const enum GridLines {\n    Horizontal = \"\\u2500\",\n    Vertical = \"\\u2502\",\n    TopLeft = \"\\u250C\",\n    TopRight = \"\\u2510\",\n    BottomLeft = \"\\u2514\",\n    BottomRight = \"\\u2518\",\n    TLeft = \"\\u251C\",\n    TRight = \"\\u2524\",\n    TTop = \"\\u252C\",\n    TBottom = \"\\u2534\",\n    Cross = \"\\u253C\",\n}\n\n/**\n * Represents a game board.\n * @template T - The type of the numeric data.\n */\nexport default abstract class Board<T extends LongInt | number> {\n    /** Contains the data stored in a BitBoard. */\n    protected readonly bitBoard: BitBoard<T>;\n\n    /** The width of the game board. */\n    protected readonly width: number;\n\n    /** The height of the game board. */\n    protected readonly height: number;\n\n    /** A stack of moves. */\n    protected readonly moves: number[] = [];\n\n    /** How many boards there are representing player positions (most likely 2). */\n    private readonly numberOfPlayerBoards: number;\n\n    /** Number of boards in total (most likely also 2). */\n    private readonly numberOfBoards: number;\n\n    /** The board states which represent a winning state. */\n    protected abstract readonly winningStates: Array<BitBoard<T>>;\n\n    /**\n     * Creates an instance of Board.\n     * @param width - The width of the game board.\n     * @param height - The height of the game board.\n     * @param playerBoardCount - How many boards there are representing player positions (most likely 2).\n     * @param extraBoardCount - Number of extra boards (most likely 0).\n     */\n    protected constructor(width: number, height: number, playerBoardCount: number = 2, extraBoardCount: number = 0) {\n        this.width = width;\n        this.height = height;\n        this.numberOfPlayerBoards = playerBoardCount;\n        this.numberOfBoards = this.numberOfPlayerBoards + extraBoardCount;\n        const totalBits = this.width * this.height * this.numberOfBoards;\n\n        this.bitBoard = (totalBits > 32\n            ? new LongIntBitBoard(Math.ceil(totalBits / 32))\n            : new IntBitBoard()) as BitBoard<T>;\n    }\n\n    /**\n     * Calculates whether or not the board is full.\n     * @returns Whether or not the board is full.\n     */\n    public get isFull(): boolean {\n        let isFull = (this.bitBoard instanceof LongIntBitBoard\n            ? new LongIntBitBoard(Math.ceil(this.width * this.height / 32))\n            : new IntBitBoard()) as BitBoard<T>;\n\n        for (let i = 0; i < this.numberOfPlayerBoards; i++)\n            isFull = isFull.or(this.getPlayerBoard(i));\n        const fullValue = (this.bitBoard instanceof LongIntBitBoard\n            ? new LongInt([\n                ...Array<number>(Math.ceil(this.width * this.height / 32) - 1).fill(~0 >>> 0),\n                2 ** (this.width * this.height - (Math.ceil(this.width * this.height / 32) - 1) * 32) - 1,\n            ])\n            : 2 ** (this.width * this.height) - 1) as T;\n\n        return isFull.equals(fullValue);\n    }\n\n    /**\n     * Calculates whether or not the board is empty.\n     * @returns Whether or not the board is empty.\n     */\n    public get isEmpty(): boolean {\n        let isEmpty = (this.bitBoard instanceof LongIntBitBoard\n            ? new LongIntBitBoard(Math.ceil(this.width * this.height / 32))\n            : new IntBitBoard()) as BitBoard<T>;\n\n        for (let i = 0; i < this.numberOfPlayerBoards; i++)\n            isEmpty = isEmpty.or(this.getPlayerBoard(i));\n\n        return isEmpty.equals(0);\n    }\n\n    /**\n     * Calculates who the winner is.\n     * @returns `false` if the game is not over, the player ID if there is a winner, and `null` if there is a draw.\n     */\n    public get winner(): 0 | 1 | false | null {\n        const playerOneBoard = this.getPlayerBoard(0);\n        const playerTwoBoard = this.getPlayerBoard(1);\n\n        for (const state of this.winningStates) {\n            if (playerOneBoard.and(state).equals(state))\n                return 0;\n\n            if (playerTwoBoard.and(state).equals(state))\n                return 1;\n        }\n\n        if (this.isFull)\n            return null;\n\n        return false;\n    }\n\n    /**\n     * Calculates which cells are empty.\n     * @returns The empty cells on the board.\n     */\n    public get emptyCells(): Position[] {\n        const emptyCells: Position[] = [];\n\n        for (let y = 0; y < this.height; y++) {\n            for (let x = 0; x < this.width; x++) {\n                if (this.cellOccupier({ x, y }) === null)\n                    emptyCells.push({ x, y });\n            }\n        }\n\n        return emptyCells;\n    }\n\n    /** Calculates the heuristic score for a given board state. */\n    public abstract get heuristic(): number;\n\n    /**\n     * Makes a move on the board.\n     * @param move - The position of the move.\n     * @param playerId - The player who's making the move.\n     */\n    public makeMove(move: Position, playerId: number): void {\n        const bit = this.getBitIndex(move, playerId);\n\n        this.moves.push(bit);\n        this.bitBoard.setBit(bit);\n    }\n\n    /**\n     * Reverses the last move.\n     * @throws {Error} - If there is no move to undo.\n     */\n    public undoLastMove(): void {\n        const lastMove = this.moves.pop();\n\n        if (lastMove === undefined)\n            throw new Error(\"No move to undo.\");\n\n        this.bitBoard.clearBit(lastMove);\n    }\n\n    /**\n     * Checks if a move is valid.\n     * @param move - The position of the move.\n     * @returns Whether or not it's valid.\n     */\n    public moveIsValid(move: Position): boolean {\n        return this.isValidPosition(move) && this.cellOccupier(move) === null;\n    }\n\n    /**\n     * Checks which player is occupying a given cell.\n     * @param cell - The cell to check.\n     * @returns If the cell is empty, the output is null, otherwise the output is the player's ID.\n     */\n    public cellOccupier(cell: Position): number | null {\n        for (let i = 0; i < this.numberOfPlayerBoards; i++) {\n            if (this.bitBoard.getBit(this.getBitIndex(cell, i)) === 1)\n                return i;\n        }\n\n        return null;\n    }\n\n    /**\n     * Creates a string representation of the board.\n     * @param wrap - Whether or not to provide a border for the board.\n     * @param labelX - Whether or not to label x.\n     * @param labelY - Whether or not to label y.\n     * @param symbols - The symbols to use as board pieces.\n     * @param colour - Whether or not to colour the pieces.\n     * @returns The string representation.\n     * @throws {Error} - If the symbols are not the same length.\n     */\n    public toString(\n        wrap: boolean = true,\n        labelX: boolean = true,\n        labelY: boolean = true,\n        symbols: string[] = [\"X\", \"O\"],\n        colour: boolean = true,\n    ): string {\n        if (symbols.length !== this.numberOfPlayerBoards)\n            throw new Error(\"Too many symbols.\");\n\n        const symbolLength = symbols[0]!.length;\n\n        if (symbols.some((s) => s.length !== symbolLength))\n            throw new Error(\"Symbols must be the same length.\");\n\n        const matchCellSpace = new RegExp(`.{${symbolLength + 2}}`, \"gu\");\n        const alphabet = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\";\n        const xLabels = labelX\n            ? `${alphabet.slice(0, this.width)\n                .split(\"\")\n                .map((letter) => ` ${letter.padStart(symbolLength)} `)\n                .join(\"\")\n                .match(matchCellSpace)!\n                .join(\" \")\n                .padStart(4 * this.width - 1 + Number(wrap) + Number(labelY))\n            }\\n`\n            : \"\";\n        const topBorder = wrap\n            ? `${labelY ? \" \" : \"\"}${GridLines.TopLeft}${GridLines.Horizontal\n                .repeat(this.width * (symbolLength + 2))\n                .match(matchCellSpace)!\n                .join(GridLines.TTop)}${GridLines.TopRight}\\n`\n            : \"\";\n        const bottomBorder = wrap\n            ? `${labelY ? \" \" : \"\"}${GridLines.BottomLeft}${GridLines.Horizontal\n                .repeat(this.width * (symbolLength + 2))\n                .match(matchCellSpace)!\n                .join(GridLines.TBottom)}${GridLines.BottomRight}`\n            : \"\";\n        const rowSeparator = `${labelY ? \" \" : \"\"}${wrap ? GridLines.TLeft : \"\"}${GridLines.Horizontal\n            .repeat(this.width * (symbolLength + 2))\n            .match(matchCellSpace)!\n            .join(GridLines.Cross)}${wrap ? GridLines.TRight : \"\"}\\n`;\n        const rows: string[] = [];\n\n        for (let y = 0; y < this.height; y++) {\n            const yLabel = labelY ? `${y + 1}` : \"\";\n            const leftBorder = wrap ? GridLines.Vertical : \"\";\n            const rightBorder = wrap ? GridLines.Vertical : \"\";\n            let row = `${yLabel}${leftBorder}`;\n\n            for (let x = 0; x < this.width; x++) {\n                const cell = { x, y };\n                const bar = x === this.width - 1 ? \"\" : GridLines.Vertical;\n                const cellOccupier = this.cellOccupier(cell);\n\n                if (cellOccupier === null) {\n                    row += ` ${\" \".repeat(symbolLength)} ${bar}`;\n                } else {\n                    row += ` ${colour ? `\\x1b[${[91, 93][cellOccupier]!}m` : \"\"}` +\n                        `${symbols[cellOccupier]}` +\n                        `${colour ? \"\\x1b[0m\" : \"\"} ${bar}`;\n                }\n            }\n            rows.push(`${row}${rightBorder}\\n`);\n        }\n\n        return `${xLabels}${topBorder}${rows.join(rowSeparator)}${bottomBorder}`;\n    }\n\n    /**\n     * Determines if a given player has a line of pieces on the board.\n     * @param playerId - The ID of the player to check.\n     * @param length - The number of pieces needed.\n     * @param maxGaps - The number of gaps allowed for a line to be valid. Defaults to 0.\n     * @returns How many lines exist.\n     */\n    public hasLine(playerId: number, length: number, maxGaps: number = 0): number {\n        if (length > Math.max(this.width, this.height))\n            return 0;\n\n        const DIRECTIONS: [Position, Position, Position, Position] = [\n            { x: 1, y: 0 },\n            { x: 0, y: 1 },\n            { x: 1, y: 1 },\n            { x: 1, y: -1 },\n        ];\n        let lineCount = 0;\n        let gaps: [number, number, number, number] = [0, 0, 0, 0];\n        let lengths: [number, number, number, number] = [0, 0, 0, 0];\n        const checkCell = (x: number, y: number, direction: 0 | 1 | 2 | 3): void => {\n            const cell = { x, y } as Position;\n\n            if (this.isValidPosition(cell)) {\n                const cellOccupier = this.cellOccupier(cell);\n\n                if (cellOccupier === null)\n                    gaps[direction]++;\n                else if (cellOccupier === playerId)\n                    lengths[direction]++;\n            }\n        };\n\n        for (let x = 0; x < this.width; x++) {\n            for (let y = 0; y < this.height; y++) {\n                gaps = [0, 0, 0, 0];\n                lengths = [0, 0, 0, 0];\n                for (let i = 0; i < length; i++) {\n                    for (let j = 0 as 0 | 1 | 2 | 3; j < 4; j++) {\n                        if (gaps[j] > maxGaps)\n                            continue;\n\n                        checkCell(x + i * DIRECTIONS[j].x, y + i * DIRECTIONS[j].y, j);\n\n                        if (lengths[j] === length)\n                            lineCount++;\n                    }\n                }\n            }\n        }\n\n        return lineCount;\n    }\n\n    /**\n     * Gets a bit index from its coordinates and player ID.\n     * @param move - The coordinates.\n     * @param playerId - The player ID to use.\n     * @returns The bit index of the move.\n     */\n    protected getBitIndex(move: Position, playerId: number): number {\n        const moveIndex = this.getIndex(move);\n        const bitBoardMoveIndex = moveIndex + this.width * this.height * playerId;\n\n        return bitBoardMoveIndex;\n    }\n\n    /**\n     * A BitBoard containing only the player's bits.\n     * @param playerId - The player's ID to get the board for.\n     * @returns The player's bits.\n     */\n    protected getPlayerBoard(playerId: number): BitBoard<T> {\n        const totalBits = (this.bitBoard instanceof LongIntBitBoard ? this.bitBoard.data.wordCount : 1) * 32;\n        const boardSize = this.width * this.height;\n\n        return this.bitBoard.leftShift(totalBits - boardSize * (playerId + 1)).rightShift(totalBits - boardSize);\n    }\n\n    /**\n     * Gets a bit index from its coordinates.\n     * @param move - The coordinates.\n     * @returns The bit index.\n     */\n    private getIndex(move: Position): number {\n        return this.width * move.y + move.x;\n    }\n\n    /**\n     * Checks if a move is valid for the given board.\n     * Does not check if that cell is already occupied.\n     * @param position - The position to check.\n     * @returns Whether or not that cell exists on the board.\n     */\n    private isValidPosition(position: Position): boolean {\n        return position.x >= 0 && position.x < this.width && position.y >= 0 && position.y < this.height;\n    }\n}\n"]}