@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 +6 -0
- package/build/base/board.d.ts +53 -45
- package/build/base/board.js +84 -72
- package/build/base/controller.d.ts +48 -37
- package/build/base/controller.js +32 -26
- package/build/bitBoard/bitBoard.d.ts +37 -34
- package/build/bitBoard/bitBoard.js +15 -12
- package/build/bitBoard/intBitBoard.d.ts +22 -22
- package/build/bitBoard/intBitBoard.js +16 -16
- package/build/bitBoard/longInt.d.ts +47 -40
- package/build/bitBoard/longInt.js +40 -37
- package/build/bitBoard/longIntBitBoard.d.ts +26 -26
- package/build/bitBoard/longIntBitBoard.js +16 -16
- package/build/console.d.ts +2 -3
- package/build/console.js +3 -4
- package/build/games/connect4/board.d.ts +12 -5
- package/build/games/connect4/board.js +12 -6
- package/build/games/connect4/controller.d.ts +18 -11
- package/build/games/connect4/controller.js +20 -12
- package/build/games/tictactoe/board.d.ts +5 -2
- package/build/games/tictactoe/board.js +5 -2
- package/build/games/tictactoe/controller.d.ts +17 -12
- package/build/games/tictactoe/controller.js +19 -13
- package/eslint.config.js +1 -1
- package/package.json +6 -7
- package/patches/eventemitter3@5.0.1.patch +0 -68
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)
|
package/build/base/board.d.ts
CHANGED
|
@@ -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
|
-
/**
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
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
|
|
28
|
-
/** The height of the board. */
|
|
29
|
-
protected readonly
|
|
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
|
-
/**
|
|
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
|
-
/**
|
|
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
|
-
*
|
|
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
|
-
/**
|
|
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
|
-
/**
|
|
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;
|
package/build/base/board.js
CHANGED
|
@@ -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
|
-
/**
|
|
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
|
-
|
|
12
|
-
/** The height of the board. */
|
|
13
|
-
|
|
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.
|
|
29
|
-
this.
|
|
29
|
+
this.width = width;
|
|
30
|
+
this.height = height;
|
|
30
31
|
this.numberOfPlayerBoards = playerBoardCount;
|
|
31
32
|
this.numberOfBoards = this.numberOfPlayerBoards + extraBoardCount;
|
|
32
|
-
const totalBits = this.
|
|
33
|
-
this.bitBoard = (totalBits > 32
|
|
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
|
-
/**
|
|
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.
|
|
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.
|
|
45
|
-
2 ** (this.
|
|
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.
|
|
53
|
+
: 2 ** (this.width * this.height) - 1);
|
|
48
54
|
return isFull.equals(fullValue);
|
|
49
55
|
}
|
|
50
|
-
/**
|
|
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.
|
|
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
|
-
*
|
|
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
|
-
/**
|
|
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.
|
|
82
|
-
for (let x = 0; x < this.
|
|
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
|
-
/**
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
188
|
+
for (let x = 0; x < this.width; x++) {
|
|
182
189
|
const cell = { x, y };
|
|
183
|
-
const bar = x === this.
|
|
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.
|
|
213
|
+
if (length > Math.max(this.width, this.height))
|
|
207
214
|
return 0;
|
|
208
|
-
const DIRECTIONS = [
|
|
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.
|
|
223
|
-
for (let y = 0; y < this.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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":"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","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":"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","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"]}
|