@oathompsonjones/mini-games 1.0.2 → 1.0.4
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/.github/workflows/lint-and-test.yml +52 -0
- package/.github/workflows/publish.yml +36 -0
- package/build/base/board.d.ts +17 -58
- package/build/base/board.js +23 -75
- package/build/base/controller.d.ts +16 -34
- package/build/base/controller.js +27 -39
- package/build/bitBoard/bitBoard.d.ts +8 -33
- package/build/bitBoard/bitBoard.js +7 -12
- package/build/bitBoard/intBitBoard.d.ts +65 -4
- package/build/bitBoard/intBitBoard.js +66 -5
- package/build/bitBoard/longInt.d.ts +2 -32
- package/build/bitBoard/longInt.js +17 -25
- package/build/bitBoard/longIntBitBoard.d.ts +65 -7
- package/build/bitBoard/longIntBitBoard.js +72 -4
- package/build/console.d.ts +1 -4
- package/build/console.js +4 -7
- package/build/games/connect4/board.d.ts +9 -0
- package/build/games/connect4/board.js +10 -1
- package/build/games/connect4/controller.d.ts +18 -0
- package/build/games/connect4/controller.js +23 -1
- package/build/games/tictactoe/board.d.ts +3 -0
- package/build/games/tictactoe/board.js +4 -1
- package/build/games/tictactoe/controller.d.ts +20 -0
- package/build/games/tictactoe/controller.js +25 -1
- package/build/index.js +1 -1
- package/eslint.config.js +3 -0
- package/package.json +6 -8
- package/patches/{eventemitter3+^5.0.1.patch → eventemitter3@5.0.1.patch} +28 -9
- package/.eslintrc +0 -6
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
name: lint-and-test
|
|
2
|
+
|
|
3
|
+
on:
|
|
4
|
+
push:
|
|
5
|
+
branches: ["*"]
|
|
6
|
+
pull_request:
|
|
7
|
+
branches: ["*"]
|
|
8
|
+
|
|
9
|
+
jobs:
|
|
10
|
+
lint:
|
|
11
|
+
runs-on: ${{ matrix.os }}
|
|
12
|
+
strategy:
|
|
13
|
+
matrix:
|
|
14
|
+
os:
|
|
15
|
+
- ubuntu-latest
|
|
16
|
+
node: [14.x, 18.x, 20.x]
|
|
17
|
+
steps:
|
|
18
|
+
- uses: actions/checkout@v3
|
|
19
|
+
- uses: pnpm/action-setup@v2
|
|
20
|
+
with:
|
|
21
|
+
version: 8
|
|
22
|
+
- name: Use Node.js ${{ matrix.node-version }}
|
|
23
|
+
uses: actions/setup-node@v3
|
|
24
|
+
with:
|
|
25
|
+
node-version: ${{ matrix.node-version }}
|
|
26
|
+
cache: "pnpm"
|
|
27
|
+
- name: Install dependencies
|
|
28
|
+
run: pnpm install
|
|
29
|
+
- name: Run Linter
|
|
30
|
+
run: pnpm lint
|
|
31
|
+
test:
|
|
32
|
+
runs-on: ${{ matrix.os }}
|
|
33
|
+
strategy:
|
|
34
|
+
matrix:
|
|
35
|
+
os:
|
|
36
|
+
- ubuntu-latest
|
|
37
|
+
node: [14.x, 18.x, 20.x]
|
|
38
|
+
steps:
|
|
39
|
+
- uses: actions/checkout@v3
|
|
40
|
+
- uses: pnpm/action-setup@v2
|
|
41
|
+
with:
|
|
42
|
+
version: 8
|
|
43
|
+
- name: Use Node.js ${{ matrix.node-version }}
|
|
44
|
+
uses: actions/setup-node@v3
|
|
45
|
+
with:
|
|
46
|
+
node-version: ${{ matrix.node-version }}
|
|
47
|
+
cache: "pnpm"
|
|
48
|
+
- name: Install dependencies
|
|
49
|
+
run: pnpm install
|
|
50
|
+
- name: Run tests
|
|
51
|
+
run: pnpm test
|
|
52
|
+
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
name: publish
|
|
2
|
+
|
|
3
|
+
on:
|
|
4
|
+
workflow_run:
|
|
5
|
+
workflows: ["lint-and-test"]
|
|
6
|
+
branches: [main, master]
|
|
7
|
+
types:
|
|
8
|
+
- completed
|
|
9
|
+
|
|
10
|
+
jobs:
|
|
11
|
+
publish:
|
|
12
|
+
runs-on: ${{ matrix.os }}
|
|
13
|
+
strategy:
|
|
14
|
+
matrix:
|
|
15
|
+
os:
|
|
16
|
+
- ubuntu-latest
|
|
17
|
+
node: [14.x]
|
|
18
|
+
steps:
|
|
19
|
+
- uses: actions/checkout@v3
|
|
20
|
+
- uses: pnpm/action-setup@v2
|
|
21
|
+
with:
|
|
22
|
+
version: 8
|
|
23
|
+
- name: Use Node.js ${{ matrix.node-version }}
|
|
24
|
+
uses: actions/setup-node@v3
|
|
25
|
+
with:
|
|
26
|
+
node-version: ${{ matrix.node-version }}
|
|
27
|
+
registry-url: 'https://registry.npmjs.org'
|
|
28
|
+
cache: "pnpm"
|
|
29
|
+
- name: Install dependencies
|
|
30
|
+
run: pnpm install
|
|
31
|
+
- name: Build code
|
|
32
|
+
run: pnpm build
|
|
33
|
+
- name: Publish code
|
|
34
|
+
run: pnpm publish
|
|
35
|
+
env:
|
|
36
|
+
NODE_AUTH_TOKEN: ${{ secrets.NPM_REGISTRY_KEY }}
|
package/build/base/board.d.ts
CHANGED
|
@@ -1,11 +1,9 @@
|
|
|
1
1
|
import type BitBoard from "../bitBoard/bitBoard.js";
|
|
2
|
-
export
|
|
2
|
+
export type Position = {
|
|
3
3
|
y: number;
|
|
4
4
|
x: number;
|
|
5
|
-
}
|
|
6
|
-
/**
|
|
7
|
-
* Defines the characters used to draw a grid.
|
|
8
|
-
*/
|
|
5
|
+
};
|
|
6
|
+
/** Defines the characters used to draw a grid. */
|
|
9
7
|
export declare const enum GridLines {
|
|
10
8
|
Horizontal = "\u2500",
|
|
11
9
|
Vertical = "\u2502",
|
|
@@ -19,59 +17,35 @@ export declare const enum GridLines {
|
|
|
19
17
|
TBottom = "\u2534",
|
|
20
18
|
Cross = "\u253C"
|
|
21
19
|
}
|
|
22
|
-
/**
|
|
23
|
-
* Represents a game board.
|
|
24
|
-
*/
|
|
20
|
+
/** Represents a game board. */
|
|
25
21
|
export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
26
|
-
/**
|
|
27
|
-
* Contains the ID of the game.
|
|
28
|
-
*/
|
|
22
|
+
/** Contains the ID of the game. */
|
|
29
23
|
gameID: string;
|
|
30
|
-
/**
|
|
31
|
-
* Contains the data stored in a BitBoard.
|
|
32
|
-
*/
|
|
24
|
+
/** Contains the data stored in a BitBoard. */
|
|
33
25
|
protected readonly bitBoard: T;
|
|
34
|
-
/**
|
|
35
|
-
* The width of the board.
|
|
36
|
-
|
|
37
|
-
*/
|
|
26
|
+
/** The width of the board. */
|
|
38
27
|
protected readonly boardWidth: number;
|
|
39
|
-
/**
|
|
40
|
-
* The height of the board.
|
|
41
|
-
*/
|
|
28
|
+
/** The height of the board. */
|
|
42
29
|
protected readonly boardHeight: number;
|
|
43
|
-
/**
|
|
44
|
-
* A stack of moves.
|
|
45
|
-
*/
|
|
30
|
+
/** A stack of moves. */
|
|
46
31
|
protected readonly moves: number[];
|
|
47
|
-
/**
|
|
48
|
-
* How many boards there are representing player positions (most likely 2).
|
|
49
|
-
*/
|
|
32
|
+
/** How many boards there are representing player positions (most likely 2). */
|
|
50
33
|
private readonly numberOfPlayerBoards;
|
|
51
|
-
/**
|
|
52
|
-
* Number of boards in total (most likely also 2).
|
|
53
|
-
*/
|
|
34
|
+
/** Number of boards in total (most likely also 2). */
|
|
54
35
|
private readonly numberOfBoards;
|
|
55
|
-
/**
|
|
56
|
-
* The board states which represent a winning state.
|
|
57
|
-
*/
|
|
36
|
+
/** The board states which represent a winning state. */
|
|
58
37
|
protected abstract readonly winningStates: T[];
|
|
59
38
|
/**
|
|
60
39
|
* Creates an instance of Board.
|
|
61
|
-
*
|
|
62
40
|
* @param width The width of the board.
|
|
63
41
|
* @param height The height of the board.
|
|
64
42
|
* @param playerBoardCount How many boards there are representing player positions (most likely 2).
|
|
65
43
|
* @param extraBoardCount Number of extra boards (most likely 0).
|
|
66
44
|
*/
|
|
67
45
|
protected constructor(width: number, height: number, playerBoardCount?: number, extraBoardCount?: number);
|
|
68
|
-
/**
|
|
69
|
-
* Calculates whether or not the board is full.
|
|
70
|
-
*/
|
|
46
|
+
/** Calculates whether or not the board is full. */
|
|
71
47
|
get isFull(): boolean;
|
|
72
|
-
/**
|
|
73
|
-
* Calculates whether or not the board is empty.
|
|
74
|
-
*/
|
|
48
|
+
/** Calculates whether or not the board is empty. */
|
|
75
49
|
get isEmpty(): boolean;
|
|
76
50
|
/**
|
|
77
51
|
* Calculates who the winner is.
|
|
@@ -80,13 +54,9 @@ export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
|
80
54
|
* If there is a draw, the output is null.
|
|
81
55
|
*/
|
|
82
56
|
get winner(): 0 | 1 | false | null;
|
|
83
|
-
/**
|
|
84
|
-
* Calculates which cells are empty.
|
|
85
|
-
*/
|
|
57
|
+
/** Calculates which cells are empty. */
|
|
86
58
|
get emptyCells(): Position[];
|
|
87
|
-
/**
|
|
88
|
-
* Calculates the heuristic score for a given board state.
|
|
89
|
-
*/
|
|
59
|
+
/** Calculates the heuristic score for a given board state. */
|
|
90
60
|
abstract get heuristic(): number;
|
|
91
61
|
/**
|
|
92
62
|
* Sets the game ID.
|
|
@@ -95,32 +65,26 @@ export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
|
95
65
|
setGameID(id: string): void;
|
|
96
66
|
/**
|
|
97
67
|
* Makes a move on the board.
|
|
98
|
-
*
|
|
99
68
|
* @param move The position of the move.
|
|
100
69
|
* @param playerId The player who's making the move.
|
|
101
70
|
*/
|
|
102
71
|
makeMove(move: Position, playerId: number): void;
|
|
103
|
-
/**
|
|
104
|
-
* Reverses the last move.
|
|
105
|
-
*/
|
|
72
|
+
/** Reverses the last move. */
|
|
106
73
|
undoLastMove(): void;
|
|
107
74
|
/**
|
|
108
75
|
* Checks if a move is valid.
|
|
109
|
-
*
|
|
110
76
|
* @param move The move.
|
|
111
77
|
* @returns Whether or not it's valid.
|
|
112
78
|
*/
|
|
113
79
|
moveIsValid(move: Position): boolean;
|
|
114
80
|
/**
|
|
115
81
|
* Checks which player is occupying a given cell.
|
|
116
|
-
*
|
|
117
82
|
* @param cell The cell to check.
|
|
118
83
|
* @returns If the cell is empty, the output is null, otherwise the output is the player's ID.
|
|
119
84
|
*/
|
|
120
85
|
cellOccupier(cell: Position): number | null;
|
|
121
86
|
/**
|
|
122
87
|
* Creates a string representation of the board.
|
|
123
|
-
*
|
|
124
88
|
* @param wrap Whether or not to provide a border for the board.
|
|
125
89
|
* @param labelX Whether or not to label x.
|
|
126
90
|
* @param labelY Whether or not to label y.
|
|
@@ -131,7 +95,6 @@ export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
|
131
95
|
toString(wrap?: boolean, labelX?: boolean, labelY?: boolean, symbols?: string[], colour?: boolean): string;
|
|
132
96
|
/**
|
|
133
97
|
* Determines if a given player has a line of pieces on the board.
|
|
134
|
-
*
|
|
135
98
|
* @param playerId The ID of the player.
|
|
136
99
|
* @param length The number of pieces needed.
|
|
137
100
|
* @param maxGaps The number of gaps allowed for a line to be valid. Defaults to 0.
|
|
@@ -140,7 +103,6 @@ export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
|
140
103
|
hasLine(playerId: number, length: number, maxGaps?: number): number;
|
|
141
104
|
/**
|
|
142
105
|
* Gets a bit index from its coordinates and player ID.
|
|
143
|
-
*
|
|
144
106
|
* @param move The coordinates.
|
|
145
107
|
* @param playerId The player ID.
|
|
146
108
|
* @returns The bit index.
|
|
@@ -148,14 +110,12 @@ export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
|
148
110
|
protected getBitIndex(move: Position, playerId: number): number;
|
|
149
111
|
/**
|
|
150
112
|
* A BitBoard containing only the player's bits.
|
|
151
|
-
*
|
|
152
113
|
* @param playerId The player's ID.
|
|
153
114
|
* @returns The player's bits.
|
|
154
115
|
*/
|
|
155
116
|
protected getPlayerBoard(playerId: number): T;
|
|
156
117
|
/**
|
|
157
118
|
* Gets a bit index from its coordinates.
|
|
158
|
-
*
|
|
159
119
|
* @param move The coordinates.
|
|
160
120
|
* @returns The bit index.
|
|
161
121
|
*/
|
|
@@ -163,7 +123,6 @@ export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
|
163
123
|
/**
|
|
164
124
|
* Checks if a move is valid for the given board.
|
|
165
125
|
* Does not check if that cell is already occupied.
|
|
166
|
-
*
|
|
167
126
|
* @param position The position to check.
|
|
168
127
|
* @returns Whether or not that cell exists on the board.
|
|
169
128
|
*/
|
package/build/base/board.js
CHANGED
|
@@ -1,59 +1,24 @@
|
|
|
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
|
-
/**
|
|
5
|
-
* Defines the characters used to draw a grid.
|
|
6
|
-
*/
|
|
7
|
-
export var GridLines;
|
|
8
|
-
(function (GridLines) {
|
|
9
|
-
GridLines["Horizontal"] = "\u2500";
|
|
10
|
-
GridLines["Vertical"] = "\u2502";
|
|
11
|
-
GridLines["TopLeft"] = "\u250C";
|
|
12
|
-
GridLines["TopRight"] = "\u2510";
|
|
13
|
-
GridLines["BottomLeft"] = "\u2514";
|
|
14
|
-
GridLines["BottomRight"] = "\u2518";
|
|
15
|
-
GridLines["TLeft"] = "\u251C";
|
|
16
|
-
GridLines["TRight"] = "\u2524";
|
|
17
|
-
GridLines["TTop"] = "\u252C";
|
|
18
|
-
GridLines["TBottom"] = "\u2534";
|
|
19
|
-
GridLines["Cross"] = "\u253C";
|
|
20
|
-
})(GridLines || (GridLines = {}));
|
|
21
|
-
/**
|
|
22
|
-
* Represents a game board.
|
|
23
|
-
*/
|
|
4
|
+
/** Represents a game board. */
|
|
24
5
|
export default class Board {
|
|
25
|
-
/**
|
|
26
|
-
* Contains the ID of the game.
|
|
27
|
-
*/
|
|
6
|
+
/** Contains the ID of the game. */
|
|
28
7
|
gameID = "";
|
|
29
|
-
/**
|
|
30
|
-
* Contains the data stored in a BitBoard.
|
|
31
|
-
*/
|
|
8
|
+
/** Contains the data stored in a BitBoard. */
|
|
32
9
|
bitBoard;
|
|
33
|
-
/**
|
|
34
|
-
* The width of the board.
|
|
35
|
-
|
|
36
|
-
*/
|
|
10
|
+
/** The width of the board. */
|
|
37
11
|
boardWidth;
|
|
38
|
-
/**
|
|
39
|
-
* The height of the board.
|
|
40
|
-
*/
|
|
12
|
+
/** The height of the board. */
|
|
41
13
|
boardHeight;
|
|
42
|
-
/**
|
|
43
|
-
* A stack of moves.
|
|
44
|
-
*/
|
|
14
|
+
/** A stack of moves. */
|
|
45
15
|
moves = [];
|
|
46
|
-
/**
|
|
47
|
-
* How many boards there are representing player positions (most likely 2).
|
|
48
|
-
*/
|
|
16
|
+
/** How many boards there are representing player positions (most likely 2). */
|
|
49
17
|
numberOfPlayerBoards;
|
|
50
|
-
/**
|
|
51
|
-
* Number of boards in total (most likely also 2).
|
|
52
|
-
*/
|
|
18
|
+
/** Number of boards in total (most likely also 2). */
|
|
53
19
|
numberOfBoards;
|
|
54
20
|
/**
|
|
55
21
|
* Creates an instance of Board.
|
|
56
|
-
*
|
|
57
22
|
* @param width The width of the board.
|
|
58
23
|
* @param height The height of the board.
|
|
59
24
|
* @param playerBoardCount How many boards there are representing player positions (most likely 2).
|
|
@@ -67,9 +32,7 @@ export default class Board {
|
|
|
67
32
|
const totalBits = this.boardWidth * this.boardHeight * this.numberOfBoards;
|
|
68
33
|
this.bitBoard = (totalBits > 32 ? new LongIntBitBoard(Math.ceil(totalBits / 32)) : new IntBitBoard());
|
|
69
34
|
}
|
|
70
|
-
/**
|
|
71
|
-
* Calculates whether or not the board is full.
|
|
72
|
-
*/
|
|
35
|
+
/** Calculates whether or not the board is full. */
|
|
73
36
|
get isFull() {
|
|
74
37
|
let isFull = (this.bitBoard instanceof LongIntBitBoard
|
|
75
38
|
? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))
|
|
@@ -79,14 +42,12 @@ export default class Board {
|
|
|
79
42
|
const fullValue = this.bitBoard instanceof LongIntBitBoard
|
|
80
43
|
? new LongInt([
|
|
81
44
|
...Array(Math.ceil(this.boardWidth * this.boardHeight / 32) - 1).fill(~0 >>> 0),
|
|
82
|
-
2 ** (this.boardWidth * this.boardHeight - (Math.ceil(this.boardWidth * this.boardHeight / 32) - 1) * 32) - 1
|
|
45
|
+
2 ** (this.boardWidth * this.boardHeight - (Math.ceil(this.boardWidth * this.boardHeight / 32) - 1) * 32) - 1,
|
|
83
46
|
])
|
|
84
47
|
: 2 ** (this.boardWidth * this.boardHeight) - 1;
|
|
85
48
|
return isFull.equals(fullValue);
|
|
86
49
|
}
|
|
87
|
-
/**
|
|
88
|
-
* Calculates whether or not the board is empty.
|
|
89
|
-
*/
|
|
50
|
+
/** Calculates whether or not the board is empty. */
|
|
90
51
|
get isEmpty() {
|
|
91
52
|
let isEmpty = (this.bitBoard instanceof LongIntBitBoard
|
|
92
53
|
? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))
|
|
@@ -114,9 +75,7 @@ export default class Board {
|
|
|
114
75
|
return null;
|
|
115
76
|
return false;
|
|
116
77
|
}
|
|
117
|
-
/**
|
|
118
|
-
* Calculates which cells are empty.
|
|
119
|
-
*/
|
|
78
|
+
/** Calculates which cells are empty. */
|
|
120
79
|
get emptyCells() {
|
|
121
80
|
const emptyCells = [];
|
|
122
81
|
for (let y = 0; y < this.boardHeight; y++) {
|
|
@@ -136,7 +95,6 @@ export default class Board {
|
|
|
136
95
|
}
|
|
137
96
|
/**
|
|
138
97
|
* Makes a move on the board.
|
|
139
|
-
*
|
|
140
98
|
* @param move The position of the move.
|
|
141
99
|
* @param playerId The player who's making the move.
|
|
142
100
|
*/
|
|
@@ -145,9 +103,7 @@ export default class Board {
|
|
|
145
103
|
this.moves.push(bit);
|
|
146
104
|
this.bitBoard.setBit(bit);
|
|
147
105
|
}
|
|
148
|
-
/**
|
|
149
|
-
* Reverses the last move.
|
|
150
|
-
*/
|
|
106
|
+
/** Reverses the last move. */
|
|
151
107
|
undoLastMove() {
|
|
152
108
|
const lastMove = this.moves.pop();
|
|
153
109
|
if (lastMove === undefined)
|
|
@@ -156,7 +112,6 @@ export default class Board {
|
|
|
156
112
|
}
|
|
157
113
|
/**
|
|
158
114
|
* Checks if a move is valid.
|
|
159
|
-
*
|
|
160
115
|
* @param move The move.
|
|
161
116
|
* @returns Whether or not it's valid.
|
|
162
117
|
*/
|
|
@@ -165,7 +120,6 @@ export default class Board {
|
|
|
165
120
|
}
|
|
166
121
|
/**
|
|
167
122
|
* Checks which player is occupying a given cell.
|
|
168
|
-
*
|
|
169
123
|
* @param cell The cell to check.
|
|
170
124
|
* @returns If the cell is empty, the output is null, otherwise the output is the player's ID.
|
|
171
125
|
*/
|
|
@@ -178,7 +132,6 @@ export default class Board {
|
|
|
178
132
|
}
|
|
179
133
|
/**
|
|
180
134
|
* Creates a string representation of the board.
|
|
181
|
-
*
|
|
182
135
|
* @param wrap Whether or not to provide a border for the board.
|
|
183
136
|
* @param labelX Whether or not to label x.
|
|
184
137
|
* @param labelY Whether or not to label y.
|
|
@@ -204,30 +157,30 @@ export default class Board {
|
|
|
204
157
|
.padStart(4 * this.boardWidth - 1 + Number(wrap) + Number(labelY))}\n`
|
|
205
158
|
: "";
|
|
206
159
|
const topBorder = wrap
|
|
207
|
-
? `${labelY ? " " : ""}${GridLines.TopLeft}${GridLines.Horizontal
|
|
160
|
+
? `${labelY ? " " : ""}${"\u250C" /* GridLines.TopLeft */}${"\u2500" /* GridLines.Horizontal */
|
|
208
161
|
.repeat(this.boardWidth * (symbolLength + 2))
|
|
209
162
|
.match(matchCellSpace)
|
|
210
|
-
.join(GridLines.TTop)}${GridLines.TopRight}\n`
|
|
163
|
+
.join("\u252C" /* GridLines.TTop */)}${"\u2510" /* GridLines.TopRight */}\n`
|
|
211
164
|
: "";
|
|
212
165
|
const bottomBorder = wrap
|
|
213
|
-
? `${labelY ? " " : ""}${GridLines.BottomLeft}${GridLines.Horizontal
|
|
166
|
+
? `${labelY ? " " : ""}${"\u2514" /* GridLines.BottomLeft */}${"\u2500" /* GridLines.Horizontal */
|
|
214
167
|
.repeat(this.boardWidth * (symbolLength + 2))
|
|
215
168
|
.match(matchCellSpace)
|
|
216
|
-
.join(GridLines.TBottom)}${GridLines.BottomRight}`
|
|
169
|
+
.join("\u2534" /* GridLines.TBottom */)}${"\u2518" /* GridLines.BottomRight */}`
|
|
217
170
|
: "";
|
|
218
|
-
const rowSeparator = `${labelY ? " " : ""}${wrap ? GridLines.TLeft : ""}${GridLines.Horizontal
|
|
171
|
+
const rowSeparator = `${labelY ? " " : ""}${wrap ? "\u251C" /* GridLines.TLeft */ : ""}${"\u2500" /* GridLines.Horizontal */
|
|
219
172
|
.repeat(this.boardWidth * (symbolLength + 2))
|
|
220
173
|
.match(matchCellSpace)
|
|
221
|
-
.join(GridLines.Cross)}${wrap ? GridLines.TRight : ""}\n`;
|
|
174
|
+
.join("\u253C" /* GridLines.Cross */)}${wrap ? "\u2524" /* GridLines.TRight */ : ""}\n`;
|
|
222
175
|
const rows = [];
|
|
223
176
|
for (let y = 0; y < this.boardHeight; y++) {
|
|
224
177
|
const yLabel = labelY ? `${y + 1}` : "";
|
|
225
|
-
const leftBorder = wrap ? GridLines.Vertical : "";
|
|
226
|
-
const rightBorder = wrap ? GridLines.Vertical : "";
|
|
178
|
+
const leftBorder = wrap ? "\u2502" /* GridLines.Vertical */ : "";
|
|
179
|
+
const rightBorder = wrap ? "\u2502" /* GridLines.Vertical */ : "";
|
|
227
180
|
let row = `${yLabel}${leftBorder}`;
|
|
228
181
|
for (let x = 0; x < this.boardWidth; x++) {
|
|
229
182
|
const cell = { x, y };
|
|
230
|
-
const bar = x === this.boardWidth - 1 ? "" : GridLines.Vertical
|
|
183
|
+
const bar = x === this.boardWidth - 1 ? "" : "\u2502" /* GridLines.Vertical */;
|
|
231
184
|
const cellOccupier = this.cellOccupier(cell);
|
|
232
185
|
if (cellOccupier === null) {
|
|
233
186
|
row += ` ${" ".repeat(symbolLength)} ${bar}`;
|
|
@@ -244,7 +197,6 @@ export default class Board {
|
|
|
244
197
|
}
|
|
245
198
|
/**
|
|
246
199
|
* Determines if a given player has a line of pieces on the board.
|
|
247
|
-
*
|
|
248
200
|
* @param playerId The ID of the player.
|
|
249
201
|
* @param length The number of pieces needed.
|
|
250
202
|
* @param maxGaps The number of gaps allowed for a line to be valid. Defaults to 0.
|
|
@@ -286,7 +238,6 @@ export default class Board {
|
|
|
286
238
|
}
|
|
287
239
|
/**
|
|
288
240
|
* Gets a bit index from its coordinates and player ID.
|
|
289
|
-
*
|
|
290
241
|
* @param move The coordinates.
|
|
291
242
|
* @param playerId The player ID.
|
|
292
243
|
* @returns The bit index.
|
|
@@ -298,7 +249,6 @@ export default class Board {
|
|
|
298
249
|
}
|
|
299
250
|
/**
|
|
300
251
|
* A BitBoard containing only the player's bits.
|
|
301
|
-
*
|
|
302
252
|
* @param playerId The player's ID.
|
|
303
253
|
* @returns The player's bits.
|
|
304
254
|
*/
|
|
@@ -309,7 +259,6 @@ export default class Board {
|
|
|
309
259
|
}
|
|
310
260
|
/**
|
|
311
261
|
* Gets a bit index from its coordinates.
|
|
312
|
-
*
|
|
313
262
|
* @param move The coordinates.
|
|
314
263
|
* @returns The bit index.
|
|
315
264
|
*/
|
|
@@ -319,7 +268,6 @@ export default class Board {
|
|
|
319
268
|
/**
|
|
320
269
|
* Checks if a move is valid for the given board.
|
|
321
270
|
* Does not check if that cell is already occupied.
|
|
322
|
-
*
|
|
323
271
|
* @param position The position to check.
|
|
324
272
|
* @returns Whether or not that cell exists on the board.
|
|
325
273
|
*/
|
|
@@ -327,4 +275,4 @@ export default class Board {
|
|
|
327
275
|
return position.x >= 0 && position.x < this.boardWidth && position.y >= 0 && position.y < this.boardHeight;
|
|
328
276
|
}
|
|
329
277
|
}
|
|
330
|
-
//# 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 interface Position {\n    y: number;\n    x: number;\n}\n\n/**\n * Defines the characters used to draw a grid.\n */\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 */\nexport default abstract class Board<T extends BitBoard = BitBoard> {\n    /**\n     * Contains the ID of the game.\n     */\n    public gameID: string = \"\";\n\n    /**\n     * Contains the data stored in a BitBoard.\n     */\n    protected readonly bitBoard: T;\n\n    /**\n     * The width of the board.\n\n     */\n    protected readonly boardWidth: number;\n\n    /**\n     * The height of the board.\n     */\n    protected readonly boardHeight: number;\n\n    /**\n     * A stack of moves.\n     */\n    protected readonly moves: number[] = [];\n\n    /**\n     * How many boards there are representing player positions (most likely 2).\n     */\n    private readonly numberOfPlayerBoards: number;\n\n    /**\n     * Number of boards in total (most likely also 2).\n     */\n    private readonly numberOfBoards: number;\n\n    /**\n     * The board states which represent a winning state.\n     */\n    protected abstract readonly winningStates: T[];\n\n    /**\n     * Creates an instance of Board.\n     *\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        this.bitBoard = (totalBits > 32 ? new LongIntBitBoard(Math.ceil(totalBits / 32)) : new IntBitBoard()) as T;\n    }\n\n    /**\n     * Calculates 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.boardWidth * this.boardHeight / 32))\n            : new IntBitBoard()) as T;\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        return isFull.equals(fullValue);\n    }\n\n    /**\n     * Calculates 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.boardWidth * this.boardHeight / 32))\n            : new IntBitBoard()) as T;\n        for (let i = 0; i < this.numberOfPlayerBoards; i++)\n            isEmpty = isEmpty.or(this.getPlayerBoard(i));\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        for (const state of this.winningStates) {\n            if (playerOneBoard.and(state).equals(state))\n                return 0;\n            if (playerTwoBoard.and(state).equals(state))\n                return 1;\n        }\n        if (this.isFull)\n            return null;\n        return false;\n    }\n\n    /**\n     * Calculates which cells are empty.\n     */\n    public get emptyCells(): Position[] {\n        const emptyCells: Position[] = [];\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        return emptyCells;\n    }\n\n    /**\n     * Calculates the heuristic score for a given board state.\n     */\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: string): void {\n        this.gameID = id;\n    }\n\n    /**\n     * Makes a move on the board.\n     *\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        this.moves.push(bit);\n        this.bitBoard.setBit(bit);\n    }\n\n    /**\n     * Reverses the last move.\n     */\n    public undoLastMove(): void {\n        const lastMove = this.moves.pop();\n        if (lastMove === undefined)\n            throw new Error(\"No move to undo.\");\n        this.bitBoard.clearBit(lastMove);\n    }\n\n    /**\n     * Checks if a move is valid.\n     *\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     *\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        return null;\n    }\n\n    /**\n     * Creates a string representation of the board.\n     *\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        const symbolLength = symbols[0]!.length;\n        if (symbols.some((s) => s.length !== symbolLength))\n            throw new Error(\"Symbols must be the same length.\");\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        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            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                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        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     *\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        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            if (this.isValidPosition(cell)) {\n                const cellOccupier = this.cellOccupier(cell);\n                if (cellOccupier === null)\n                    gaps[direction]++;\n                else if (cellOccupier === playerId)\n                    lengths[direction]++;\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                        checkCell(x + i * DIRECTIONS[j].x, y + i * DIRECTIONS[j].y, j);\n                        if (lengths[j] === length)\n                            lineCount++;\n                    }\n                }\n            }\n        }\n        return lineCount;\n    }\n\n    /**\n     * Gets a bit index from its coordinates and player ID.\n     *\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        return bitBoardMoveIndex;\n    }\n\n    /**\n     * A BitBoard containing only the player's bits.\n     *\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        return this.bitBoard.leftShift(totalBits - boardSize * (playerId + 1)).rightShift(totalBits - boardSize);\n    }\n\n    /**\n     * Gets a bit index from its coordinates.\n     *\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     *\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"]}
|
|
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: string = \"\";\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: string): 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"]}
|