@oathompsonjones/mini-games 1.0.0
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/.eslintrc +6 -0
- package/.gitattributes +1 -0
- package/LICENSE +20 -0
- package/README.md +6 -0
- package/build/base/board.d.ts +171 -0
- package/build/base/board.js +330 -0
- package/build/base/controller.d.ts +124 -0
- package/build/base/controller.js +185 -0
- package/build/bitBoard/bitBoard.d.ts +133 -0
- package/build/bitBoard/bitBoard.js +56 -0
- package/build/bitBoard/intBitBoard.d.ts +27 -0
- package/build/bitBoard/intBitBoard.js +65 -0
- package/build/bitBoard/longInt.d.ts +206 -0
- package/build/bitBoard/longInt.js +310 -0
- package/build/bitBoard/longIntBitBoard.d.ts +46 -0
- package/build/bitBoard/longIntBitBoard.js +85 -0
- package/build/console.d.ts +23 -0
- package/build/console.js +30 -0
- package/build/games/connect4/board.d.ts +15 -0
- package/build/games/connect4/board.js +88 -0
- package/build/games/connect4/controller.d.ts +11 -0
- package/build/games/connect4/controller.js +49 -0
- package/build/games/tictactoe/board.d.ts +7 -0
- package/build/games/tictactoe/board.js +18 -0
- package/build/games/tictactoe/controller.d.ts +11 -0
- package/build/games/tictactoe/controller.js +49 -0
- package/build/index.d.ts +2 -0
- package/build/index.js +3 -0
- package/package.json +32 -0
- package/patches/eventemitter3+^5.0.1.patch +49 -0
- package/tsconfig.build.json +8 -0
package/.eslintrc
ADDED
package/.gitattributes
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
*.js linguist-language=TypeScript
|
package/LICENSE
ADDED
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
The MIT License (MIT)
|
|
2
|
+
|
|
3
|
+
Copyright © 2023 Oliver Jones
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
6
|
+
this software and associated documentation files (the "Software"), to deal in
|
|
7
|
+
the Software without restriction, including without limitation the rights to
|
|
8
|
+
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
9
|
+
the Software, and to permit persons to whom the Software is furnished to do so,
|
|
10
|
+
subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
17
|
+
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
18
|
+
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
19
|
+
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
20
|
+
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
package/README.md
ADDED
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
# MiniGames
|
|
2
|
+
This repository contains a selection of MiniGames.
|
|
3
|
+
You will be able to play these games on [my website](https://oathompsonjones.co.uk/arcade) in the future.
|
|
4
|
+
|
|
5
|
+
For now, enjoy this clip of what happens when a computer plays TicTacToe against itself 1000 times.
|
|
6
|
+

|
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
import type BitBoard from "../bitBoard/bitBoard.js";
|
|
2
|
+
export interface Position {
|
|
3
|
+
y: number;
|
|
4
|
+
x: number;
|
|
5
|
+
}
|
|
6
|
+
/**
|
|
7
|
+
* Defines the characters used to draw a grid.
|
|
8
|
+
*/
|
|
9
|
+
export declare const enum GridLines {
|
|
10
|
+
Horizontal = "\u2500",
|
|
11
|
+
Vertical = "\u2502",
|
|
12
|
+
TopLeft = "\u250C",
|
|
13
|
+
TopRight = "\u2510",
|
|
14
|
+
BottomLeft = "\u2514",
|
|
15
|
+
BottomRight = "\u2518",
|
|
16
|
+
TLeft = "\u251C",
|
|
17
|
+
TRight = "\u2524",
|
|
18
|
+
TTop = "\u252C",
|
|
19
|
+
TBottom = "\u2534",
|
|
20
|
+
Cross = "\u253C"
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Represents a game board.
|
|
24
|
+
*/
|
|
25
|
+
export default abstract class Board<T extends BitBoard = BitBoard> {
|
|
26
|
+
/**
|
|
27
|
+
* Contains the ID of the game.
|
|
28
|
+
*/
|
|
29
|
+
gameID: string;
|
|
30
|
+
/**
|
|
31
|
+
* Contains the data stored in a BitBoard.
|
|
32
|
+
*/
|
|
33
|
+
protected readonly bitBoard: T;
|
|
34
|
+
/**
|
|
35
|
+
* The width of the board.
|
|
36
|
+
|
|
37
|
+
*/
|
|
38
|
+
protected readonly boardWidth: number;
|
|
39
|
+
/**
|
|
40
|
+
* The height of the board.
|
|
41
|
+
*/
|
|
42
|
+
protected readonly boardHeight: number;
|
|
43
|
+
/**
|
|
44
|
+
* A stack of moves.
|
|
45
|
+
*/
|
|
46
|
+
protected readonly moves: number[];
|
|
47
|
+
/**
|
|
48
|
+
* How many boards there are representing player positions (most likely 2).
|
|
49
|
+
*/
|
|
50
|
+
private readonly numberOfPlayerBoards;
|
|
51
|
+
/**
|
|
52
|
+
* Number of boards in total (most likely also 2).
|
|
53
|
+
*/
|
|
54
|
+
private readonly numberOfBoards;
|
|
55
|
+
/**
|
|
56
|
+
* The board states which represent a winning state.
|
|
57
|
+
*/
|
|
58
|
+
protected abstract readonly winningStates: T[];
|
|
59
|
+
/**
|
|
60
|
+
* Creates an instance of Board.
|
|
61
|
+
*
|
|
62
|
+
* @param width The width of the board.
|
|
63
|
+
* @param height The height of the board.
|
|
64
|
+
* @param playerBoardCount How many boards there are representing player positions (most likely 2).
|
|
65
|
+
* @param extraBoardCount Number of extra boards (most likely 0).
|
|
66
|
+
*/
|
|
67
|
+
protected constructor(width: number, height: number, playerBoardCount?: number, extraBoardCount?: number);
|
|
68
|
+
/**
|
|
69
|
+
* Calculates whether or not the board is full.
|
|
70
|
+
*/
|
|
71
|
+
get isFull(): boolean;
|
|
72
|
+
/**
|
|
73
|
+
* Calculates whether or not the board is empty.
|
|
74
|
+
*/
|
|
75
|
+
get isEmpty(): boolean;
|
|
76
|
+
/**
|
|
77
|
+
* Calculates who the winner is.
|
|
78
|
+
* If the game is not over, the output is false.
|
|
79
|
+
* If there is a winner, the output is their ID.
|
|
80
|
+
* If there is a draw, the output is null.
|
|
81
|
+
*/
|
|
82
|
+
get winner(): 0 | 1 | false | null;
|
|
83
|
+
/**
|
|
84
|
+
* Calculates which cells are empty.
|
|
85
|
+
*/
|
|
86
|
+
get emptyCells(): Position[];
|
|
87
|
+
/**
|
|
88
|
+
* Calculates the heuristic score for a given board state.
|
|
89
|
+
*/
|
|
90
|
+
abstract get heuristic(): number;
|
|
91
|
+
/**
|
|
92
|
+
* Sets the game ID.
|
|
93
|
+
* @param id The ID of the game.
|
|
94
|
+
*/
|
|
95
|
+
setGameID(id: string): void;
|
|
96
|
+
/**
|
|
97
|
+
* Makes a move on the board.
|
|
98
|
+
*
|
|
99
|
+
* @param move The position of the move.
|
|
100
|
+
* @param playerId The player who's making the move.
|
|
101
|
+
*/
|
|
102
|
+
makeMove(move: Position, playerId: number): void;
|
|
103
|
+
/**
|
|
104
|
+
* Reverses the last move.
|
|
105
|
+
*/
|
|
106
|
+
undoLastMove(): void;
|
|
107
|
+
/**
|
|
108
|
+
* Checks if a move is valid.
|
|
109
|
+
*
|
|
110
|
+
* @param move The move.
|
|
111
|
+
* @returns Whether or not it's valid.
|
|
112
|
+
*/
|
|
113
|
+
moveIsValid(move: Position): boolean;
|
|
114
|
+
/**
|
|
115
|
+
* Checks which player is occupying a given cell.
|
|
116
|
+
*
|
|
117
|
+
* @param cell The cell to check.
|
|
118
|
+
* @returns If the cell is empty, the output is null, otherwise the output is the player's ID.
|
|
119
|
+
*/
|
|
120
|
+
cellOccupier(cell: Position): number | null;
|
|
121
|
+
/**
|
|
122
|
+
* Creates a string representation of the board.
|
|
123
|
+
*
|
|
124
|
+
* @param wrap Whether or not to provide a border for the board.
|
|
125
|
+
* @param labelX Whether or not to label x.
|
|
126
|
+
* @param labelY Whether or not to label y.
|
|
127
|
+
* @param symbols The symbols to use as board pieces.
|
|
128
|
+
* @param colour Whether or not to colour the pieces.
|
|
129
|
+
* @returns The string representation.
|
|
130
|
+
*/
|
|
131
|
+
toString(wrap?: boolean, labelX?: boolean, labelY?: boolean, symbols?: string[], colour?: boolean): string;
|
|
132
|
+
/**
|
|
133
|
+
* Determines if a given player has a line of pieces on the board.
|
|
134
|
+
*
|
|
135
|
+
* @param playerId The ID of the player.
|
|
136
|
+
* @param length The number of pieces needed.
|
|
137
|
+
* @param maxGaps The number of gaps allowed for a line to be valid. Defaults to 0.
|
|
138
|
+
* @returns How many lines exist.
|
|
139
|
+
*/
|
|
140
|
+
hasLine(playerId: number, length: number, maxGaps?: number): number;
|
|
141
|
+
/**
|
|
142
|
+
* Gets a bit index from its coordinates and player ID.
|
|
143
|
+
*
|
|
144
|
+
* @param move The coordinates.
|
|
145
|
+
* @param playerId The player ID.
|
|
146
|
+
* @returns The bit index.
|
|
147
|
+
*/
|
|
148
|
+
protected getBitIndex(move: Position, playerId: number): number;
|
|
149
|
+
/**
|
|
150
|
+
* A BitBoard containing only the player's bits.
|
|
151
|
+
*
|
|
152
|
+
* @param playerId The player's ID.
|
|
153
|
+
* @returns The player's bits.
|
|
154
|
+
*/
|
|
155
|
+
protected getPlayerBoard(playerId: number): T;
|
|
156
|
+
/**
|
|
157
|
+
* Gets a bit index from its coordinates.
|
|
158
|
+
*
|
|
159
|
+
* @param move The coordinates.
|
|
160
|
+
* @returns The bit index.
|
|
161
|
+
*/
|
|
162
|
+
private getIndex;
|
|
163
|
+
/**
|
|
164
|
+
* Checks if a move is valid for the given board.
|
|
165
|
+
* Does not check if that cell is already occupied.
|
|
166
|
+
*
|
|
167
|
+
* @param position The position to check.
|
|
168
|
+
* @returns Whether or not that cell exists on the board.
|
|
169
|
+
*/
|
|
170
|
+
private isValidPosition;
|
|
171
|
+
}
|
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
import IntBitBoard from "../bitBoard/intBitBoard.js";
|
|
2
|
+
import LongInt from "../bitBoard/longInt.js";
|
|
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
|
+
*/
|
|
24
|
+
export default class Board {
|
|
25
|
+
/**
|
|
26
|
+
* Contains the ID of the game.
|
|
27
|
+
*/
|
|
28
|
+
gameID = "";
|
|
29
|
+
/**
|
|
30
|
+
* Contains the data stored in a BitBoard.
|
|
31
|
+
*/
|
|
32
|
+
bitBoard;
|
|
33
|
+
/**
|
|
34
|
+
* The width of the board.
|
|
35
|
+
|
|
36
|
+
*/
|
|
37
|
+
boardWidth;
|
|
38
|
+
/**
|
|
39
|
+
* The height of the board.
|
|
40
|
+
*/
|
|
41
|
+
boardHeight;
|
|
42
|
+
/**
|
|
43
|
+
* A stack of moves.
|
|
44
|
+
*/
|
|
45
|
+
moves = [];
|
|
46
|
+
/**
|
|
47
|
+
* How many boards there are representing player positions (most likely 2).
|
|
48
|
+
*/
|
|
49
|
+
numberOfPlayerBoards;
|
|
50
|
+
/**
|
|
51
|
+
* Number of boards in total (most likely also 2).
|
|
52
|
+
*/
|
|
53
|
+
numberOfBoards;
|
|
54
|
+
/**
|
|
55
|
+
* Creates an instance of Board.
|
|
56
|
+
*
|
|
57
|
+
* @param width The width of the board.
|
|
58
|
+
* @param height The height of the board.
|
|
59
|
+
* @param playerBoardCount How many boards there are representing player positions (most likely 2).
|
|
60
|
+
* @param extraBoardCount Number of extra boards (most likely 0).
|
|
61
|
+
*/
|
|
62
|
+
constructor(width, height, playerBoardCount = 2, extraBoardCount = 0) {
|
|
63
|
+
this.boardWidth = width;
|
|
64
|
+
this.boardHeight = height;
|
|
65
|
+
this.numberOfPlayerBoards = playerBoardCount;
|
|
66
|
+
this.numberOfBoards = this.numberOfPlayerBoards + extraBoardCount;
|
|
67
|
+
const totalBits = this.boardWidth * this.boardHeight * this.numberOfBoards;
|
|
68
|
+
this.bitBoard = (totalBits > 32 ? new LongIntBitBoard(Math.ceil(totalBits / 32)) : new IntBitBoard());
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Calculates whether or not the board is full.
|
|
72
|
+
*/
|
|
73
|
+
get isFull() {
|
|
74
|
+
let isFull = (this.bitBoard instanceof LongIntBitBoard
|
|
75
|
+
? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))
|
|
76
|
+
: new IntBitBoard());
|
|
77
|
+
for (let i = 0; i < this.numberOfPlayerBoards; i++)
|
|
78
|
+
isFull = isFull.or(this.getPlayerBoard(i));
|
|
79
|
+
const fullValue = this.bitBoard instanceof LongIntBitBoard
|
|
80
|
+
? new LongInt([
|
|
81
|
+
...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
|
|
83
|
+
])
|
|
84
|
+
: 2 ** (this.boardWidth * this.boardHeight) - 1;
|
|
85
|
+
return isFull.equals(fullValue);
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Calculates whether or not the board is empty.
|
|
89
|
+
*/
|
|
90
|
+
get isEmpty() {
|
|
91
|
+
let isEmpty = (this.bitBoard instanceof LongIntBitBoard
|
|
92
|
+
? new LongIntBitBoard(Math.ceil(this.boardWidth * this.boardHeight / 32))
|
|
93
|
+
: new IntBitBoard());
|
|
94
|
+
for (let i = 0; i < this.numberOfPlayerBoards; i++)
|
|
95
|
+
isEmpty = isEmpty.or(this.getPlayerBoard(i));
|
|
96
|
+
return isEmpty.equals(0);
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* Calculates who the winner is.
|
|
100
|
+
* If the game is not over, the output is false.
|
|
101
|
+
* If there is a winner, the output is their ID.
|
|
102
|
+
* If there is a draw, the output is null.
|
|
103
|
+
*/
|
|
104
|
+
get winner() {
|
|
105
|
+
const playerOneBoard = this.getPlayerBoard(0);
|
|
106
|
+
const playerTwoBoard = this.getPlayerBoard(1);
|
|
107
|
+
for (const state of this.winningStates) {
|
|
108
|
+
if (playerOneBoard.and(state).equals(state))
|
|
109
|
+
return 0;
|
|
110
|
+
if (playerTwoBoard.and(state).equals(state))
|
|
111
|
+
return 1;
|
|
112
|
+
}
|
|
113
|
+
if (this.isFull)
|
|
114
|
+
return null;
|
|
115
|
+
return false;
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Calculates which cells are empty.
|
|
119
|
+
*/
|
|
120
|
+
get emptyCells() {
|
|
121
|
+
const emptyCells = [];
|
|
122
|
+
for (let y = 0; y < this.boardHeight; y++) {
|
|
123
|
+
for (let x = 0; x < this.boardWidth; x++) {
|
|
124
|
+
if (this.cellOccupier({ x, y }) === null)
|
|
125
|
+
emptyCells.push({ x, y });
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
return emptyCells;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Sets the game ID.
|
|
132
|
+
* @param id The ID of the game.
|
|
133
|
+
*/
|
|
134
|
+
setGameID(id) {
|
|
135
|
+
this.gameID = id;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Makes a move on the board.
|
|
139
|
+
*
|
|
140
|
+
* @param move The position of the move.
|
|
141
|
+
* @param playerId The player who's making the move.
|
|
142
|
+
*/
|
|
143
|
+
makeMove(move, playerId) {
|
|
144
|
+
const bit = this.getBitIndex(move, playerId);
|
|
145
|
+
this.moves.push(bit);
|
|
146
|
+
this.bitBoard.setBit(bit);
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Reverses the last move.
|
|
150
|
+
*/
|
|
151
|
+
undoLastMove() {
|
|
152
|
+
const lastMove = this.moves.pop();
|
|
153
|
+
if (lastMove === undefined)
|
|
154
|
+
throw new Error("No move to undo.");
|
|
155
|
+
this.bitBoard.clearBit(lastMove);
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
158
|
+
* Checks if a move is valid.
|
|
159
|
+
*
|
|
160
|
+
* @param move The move.
|
|
161
|
+
* @returns Whether or not it's valid.
|
|
162
|
+
*/
|
|
163
|
+
moveIsValid(move) {
|
|
164
|
+
return this.isValidPosition(move) && this.cellOccupier(move) === null;
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Checks which player is occupying a given cell.
|
|
168
|
+
*
|
|
169
|
+
* @param cell The cell to check.
|
|
170
|
+
* @returns If the cell is empty, the output is null, otherwise the output is the player's ID.
|
|
171
|
+
*/
|
|
172
|
+
cellOccupier(cell) {
|
|
173
|
+
for (let i = 0; i < this.numberOfPlayerBoards; i++) {
|
|
174
|
+
if (this.bitBoard.getBit(this.getBitIndex(cell, i)) === 1)
|
|
175
|
+
return i;
|
|
176
|
+
}
|
|
177
|
+
return null;
|
|
178
|
+
}
|
|
179
|
+
/**
|
|
180
|
+
* Creates a string representation of the board.
|
|
181
|
+
*
|
|
182
|
+
* @param wrap Whether or not to provide a border for the board.
|
|
183
|
+
* @param labelX Whether or not to label x.
|
|
184
|
+
* @param labelY Whether or not to label y.
|
|
185
|
+
* @param symbols The symbols to use as board pieces.
|
|
186
|
+
* @param colour Whether or not to colour the pieces.
|
|
187
|
+
* @returns The string representation.
|
|
188
|
+
*/
|
|
189
|
+
toString(wrap = true, labelX = true, labelY = true, symbols = ["X", "O"], colour = true) {
|
|
190
|
+
if (symbols.length !== this.numberOfPlayerBoards)
|
|
191
|
+
throw new Error("Too many symbols.");
|
|
192
|
+
const symbolLength = symbols[0].length;
|
|
193
|
+
if (symbols.some((s) => s.length !== symbolLength))
|
|
194
|
+
throw new Error("Symbols must be the same length.");
|
|
195
|
+
const matchCellSpace = new RegExp(`.{${symbolLength + 2}}`, "gu");
|
|
196
|
+
const alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz";
|
|
197
|
+
const xLabels = labelX
|
|
198
|
+
? `${alphabet.slice(0, this.boardWidth)
|
|
199
|
+
.split("")
|
|
200
|
+
.map((letter) => ` ${letter.padStart(symbolLength)} `)
|
|
201
|
+
.join("")
|
|
202
|
+
.match(matchCellSpace)
|
|
203
|
+
.join(" ")
|
|
204
|
+
.padStart(4 * this.boardWidth - 1 + Number(wrap) + Number(labelY))}\n`
|
|
205
|
+
: "";
|
|
206
|
+
const topBorder = wrap
|
|
207
|
+
? `${labelY ? " " : ""}${GridLines.TopLeft}${GridLines.Horizontal
|
|
208
|
+
.repeat(this.boardWidth * (symbolLength + 2))
|
|
209
|
+
.match(matchCellSpace)
|
|
210
|
+
.join(GridLines.TTop)}${GridLines.TopRight}\n`
|
|
211
|
+
: "";
|
|
212
|
+
const bottomBorder = wrap
|
|
213
|
+
? `${labelY ? " " : ""}${GridLines.BottomLeft}${GridLines.Horizontal
|
|
214
|
+
.repeat(this.boardWidth * (symbolLength + 2))
|
|
215
|
+
.match(matchCellSpace)
|
|
216
|
+
.join(GridLines.TBottom)}${GridLines.BottomRight}`
|
|
217
|
+
: "";
|
|
218
|
+
const rowSeparator = `${labelY ? " " : ""}${wrap ? GridLines.TLeft : ""}${GridLines.Horizontal
|
|
219
|
+
.repeat(this.boardWidth * (symbolLength + 2))
|
|
220
|
+
.match(matchCellSpace)
|
|
221
|
+
.join(GridLines.Cross)}${wrap ? GridLines.TRight : ""}\n`;
|
|
222
|
+
const rows = [];
|
|
223
|
+
for (let y = 0; y < this.boardHeight; y++) {
|
|
224
|
+
const yLabel = labelY ? `${y + 1}` : "";
|
|
225
|
+
const leftBorder = wrap ? GridLines.Vertical : "";
|
|
226
|
+
const rightBorder = wrap ? GridLines.Vertical : "";
|
|
227
|
+
let row = `${yLabel}${leftBorder}`;
|
|
228
|
+
for (let x = 0; x < this.boardWidth; x++) {
|
|
229
|
+
const cell = { x, y };
|
|
230
|
+
const bar = x === this.boardWidth - 1 ? "" : GridLines.Vertical;
|
|
231
|
+
const cellOccupier = this.cellOccupier(cell);
|
|
232
|
+
if (cellOccupier === null) {
|
|
233
|
+
row += ` ${" ".repeat(symbolLength)} ${bar}`;
|
|
234
|
+
}
|
|
235
|
+
else {
|
|
236
|
+
row += ` ${colour ? `\x1b[${[91, 93][cellOccupier]}m` : ""}` +
|
|
237
|
+
`${symbols[cellOccupier]}` +
|
|
238
|
+
`${colour ? "\x1b[0m" : ""} ${bar}`;
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
rows.push(`${row}${rightBorder}\n`);
|
|
242
|
+
}
|
|
243
|
+
return `${xLabels}${topBorder}${rows.join(rowSeparator)}${bottomBorder}`;
|
|
244
|
+
}
|
|
245
|
+
/**
|
|
246
|
+
* Determines if a given player has a line of pieces on the board.
|
|
247
|
+
*
|
|
248
|
+
* @param playerId The ID of the player.
|
|
249
|
+
* @param length The number of pieces needed.
|
|
250
|
+
* @param maxGaps The number of gaps allowed for a line to be valid. Defaults to 0.
|
|
251
|
+
* @returns How many lines exist.
|
|
252
|
+
*/
|
|
253
|
+
hasLine(playerId, length, maxGaps = 0) {
|
|
254
|
+
if (length > Math.max(this.boardWidth, this.boardHeight))
|
|
255
|
+
return 0;
|
|
256
|
+
const DIRECTIONS = [{ x: 1, y: 0 }, { x: 0, y: 1 }, { x: 1, y: 1 }, { x: 1, y: -1 }];
|
|
257
|
+
let lineCount = 0;
|
|
258
|
+
let gaps = [0, 0, 0, 0];
|
|
259
|
+
let lengths = [0, 0, 0, 0];
|
|
260
|
+
const checkCell = (x, y, direction) => {
|
|
261
|
+
const cell = { x, y };
|
|
262
|
+
if (this.isValidPosition(cell)) {
|
|
263
|
+
const cellOccupier = this.cellOccupier(cell);
|
|
264
|
+
if (cellOccupier === null)
|
|
265
|
+
gaps[direction]++;
|
|
266
|
+
else if (cellOccupier === playerId)
|
|
267
|
+
lengths[direction]++;
|
|
268
|
+
}
|
|
269
|
+
};
|
|
270
|
+
for (let x = 0; x < this.boardWidth; x++) {
|
|
271
|
+
for (let y = 0; y < this.boardHeight; y++) {
|
|
272
|
+
gaps = [0, 0, 0, 0];
|
|
273
|
+
lengths = [0, 0, 0, 0];
|
|
274
|
+
for (let i = 0; i < length; i++) {
|
|
275
|
+
for (let j = 0; j < 4; j++) {
|
|
276
|
+
if (gaps[j] > maxGaps)
|
|
277
|
+
continue;
|
|
278
|
+
checkCell(x + i * DIRECTIONS[j].x, y + i * DIRECTIONS[j].y, j);
|
|
279
|
+
if (lengths[j] === length)
|
|
280
|
+
lineCount++;
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
return lineCount;
|
|
286
|
+
}
|
|
287
|
+
/**
|
|
288
|
+
* Gets a bit index from its coordinates and player ID.
|
|
289
|
+
*
|
|
290
|
+
* @param move The coordinates.
|
|
291
|
+
* @param playerId The player ID.
|
|
292
|
+
* @returns The bit index.
|
|
293
|
+
*/
|
|
294
|
+
getBitIndex(move, playerId) {
|
|
295
|
+
const moveIndex = this.getIndex(move);
|
|
296
|
+
const bitBoardMoveIndex = moveIndex + this.boardWidth * this.boardHeight * playerId;
|
|
297
|
+
return bitBoardMoveIndex;
|
|
298
|
+
}
|
|
299
|
+
/**
|
|
300
|
+
* A BitBoard containing only the player's bits.
|
|
301
|
+
*
|
|
302
|
+
* @param playerId The player's ID.
|
|
303
|
+
* @returns The player's bits.
|
|
304
|
+
*/
|
|
305
|
+
getPlayerBoard(playerId) {
|
|
306
|
+
const totalBits = (this.bitBoard instanceof LongIntBitBoard ? this.bitBoard.data.wordCount : 1) * 32;
|
|
307
|
+
const boardSize = this.boardWidth * this.boardHeight;
|
|
308
|
+
return this.bitBoard.leftShift(totalBits - boardSize * (playerId + 1)).rightShift(totalBits - boardSize);
|
|
309
|
+
}
|
|
310
|
+
/**
|
|
311
|
+
* Gets a bit index from its coordinates.
|
|
312
|
+
*
|
|
313
|
+
* @param move The coordinates.
|
|
314
|
+
* @returns The bit index.
|
|
315
|
+
*/
|
|
316
|
+
getIndex(move) {
|
|
317
|
+
return this.boardWidth * move.y + move.x;
|
|
318
|
+
}
|
|
319
|
+
/**
|
|
320
|
+
* Checks if a move is valid for the given board.
|
|
321
|
+
* Does not check if that cell is already occupied.
|
|
322
|
+
*
|
|
323
|
+
* @param position The position to check.
|
|
324
|
+
* @returns Whether or not that cell exists on the board.
|
|
325
|
+
*/
|
|
326
|
+
isValidPosition(position) {
|
|
327
|
+
return position.x >= 0 && position.x < this.boardWidth && position.y >= 0 && position.y < this.boardHeight;
|
|
328
|
+
}
|
|
329
|
+
}
|
|
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"]}
|