@oathompsonjones/mini-games 1.0.13 → 1.0.15

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.
@@ -31,6 +31,11 @@ export default abstract class Board<T extends LongInt | number> {
31
31
  protected readonly height: number;
32
32
  /** A stack of moves. */
33
33
  protected readonly moves: number[];
34
+ /**
35
+ * Cached result of the last `winner` computation.
36
+ * Set to `undefined` whenever the board changes so the getter recomputes.
37
+ */
38
+ private _cachedWinner;
34
39
  /** How many boards there are representing player positions (most likely 2). */
35
40
  private readonly numberOfPlayerBoards;
36
41
  /** Number of boards in total (most likely also 2). */
@@ -60,6 +65,12 @@ export default abstract class Board<T extends LongInt | number> {
60
65
  * @returns `false` if the game is not over, the player ID if there is a winner, and `null` if there is a draw.
61
66
  */
62
67
  get winner(): 0 | 1 | false | null;
68
+ /**
69
+ * Returns a compact string key uniquely identifying the current board state.
70
+ * Used as a key in the alpha-beta transposition table.
71
+ * @returns The unique board state key.
72
+ */
73
+ get hashKey(): string;
63
74
  /**
64
75
  * Calculates which cells are empty.
65
76
  * @returns The empty cells on the board.
@@ -101,14 +112,6 @@ export default abstract class Board<T extends LongInt | number> {
101
112
  * @throws {Error} - If the symbols are not the same length.
102
113
  */
103
114
  toString(wrap?: boolean, labelX?: boolean, labelY?: boolean, symbols?: string[], colour?: boolean): string;
104
- /**
105
- * Determines if a given player has a line of pieces on the board.
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.
109
- * @returns How many lines exist.
110
- */
111
- hasLine(playerId: number, length: number, maxGaps?: number): number;
112
115
  /**
113
116
  * Gets a bit index from its coordinates and player ID.
114
117
  * @param move - The coordinates.
@@ -14,6 +14,11 @@ export default class Board {
14
14
  height;
15
15
  /** A stack of moves. */
16
16
  moves = [];
17
+ /**
18
+ * Cached result of the last `winner` computation.
19
+ * Set to `undefined` whenever the board changes so the getter recomputes.
20
+ */
21
+ _cachedWinner = undefined;
17
22
  /** How many boards there are representing player positions (most likely 2). */
18
23
  numberOfPlayerBoards;
19
24
  /** Number of boards in total (most likely also 2). */
@@ -70,17 +75,31 @@ export default class Board {
70
75
  * @returns `false` if the game is not over, the player ID if there is a winner, and `null` if there is a draw.
71
76
  */
72
77
  get winner() {
78
+ if (this._cachedWinner !== undefined)
79
+ return this._cachedWinner;
73
80
  const playerOneBoard = this.getPlayerBoard(0);
74
81
  const playerTwoBoard = this.getPlayerBoard(1);
75
82
  for (const state of this.winningStates) {
76
- if (playerOneBoard.and(state).equals(state))
83
+ if (playerOneBoard.and(state).equals(state)) {
84
+ this._cachedWinner = 0;
77
85
  return 0;
78
- if (playerTwoBoard.and(state).equals(state))
86
+ }
87
+ if (playerTwoBoard.and(state).equals(state)) {
88
+ this._cachedWinner = 1;
79
89
  return 1;
90
+ }
80
91
  }
81
- if (this.isFull)
82
- return null;
83
- return false;
92
+ this._cachedWinner = this.isFull ? null : false;
93
+ return this._cachedWinner;
94
+ }
95
+ /**
96
+ * Returns a compact string key uniquely identifying the current board state.
97
+ * Used as a key in the alpha-beta transposition table.
98
+ * @returns The unique board state key.
99
+ */
100
+ get hashKey() {
101
+ const { data } = this.bitBoard;
102
+ return data instanceof LongInt ? data.data.join(",") : String(data);
84
103
  }
85
104
  /**
86
105
  * Calculates which cells are empty.
@@ -102,6 +121,7 @@ export default class Board {
102
121
  * @param playerId - The player who's making the move.
103
122
  */
104
123
  makeMove(move, playerId) {
124
+ this._cachedWinner = undefined;
105
125
  const bit = this.getBitIndex(move, playerId);
106
126
  this.moves.push(bit);
107
127
  this.bitBoard.setBit(bit);
@@ -111,6 +131,7 @@ export default class Board {
111
131
  * @throws {Error} - If there is no move to undo.
112
132
  */
113
133
  undoLastMove() {
134
+ this._cachedWinner = undefined;
114
135
  const lastMove = this.moves.pop();
115
136
  if (lastMove === undefined)
116
137
  throw new Error("No move to undo.");
@@ -202,52 +223,6 @@ export default class Board {
202
223
  }
203
224
  return `${xLabels}${topBorder}${rows.join(rowSeparator)}${bottomBorder}`;
204
225
  }
205
- /**
206
- * Determines if a given player has a line of pieces on the board.
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.
210
- * @returns How many lines exist.
211
- */
212
- hasLine(playerId, length, maxGaps = 0) {
213
- if (length > Math.max(this.width, this.height))
214
- return 0;
215
- const DIRECTIONS = [
216
- { x: 1, y: 0 },
217
- { x: 0, y: 1 },
218
- { x: 1, y: 1 },
219
- { x: 1, y: -1 },
220
- ];
221
- let lineCount = 0;
222
- let gaps = [0, 0, 0, 0];
223
- let lengths = [0, 0, 0, 0];
224
- const checkCell = (x, y, direction) => {
225
- const cell = { x, y };
226
- if (this.isValidPosition(cell)) {
227
- const cellOccupier = this.cellOccupier(cell);
228
- if (cellOccupier === null)
229
- gaps[direction]++;
230
- else if (cellOccupier === playerId)
231
- lengths[direction]++;
232
- }
233
- };
234
- for (let x = 0; x < this.width; x++) {
235
- for (let y = 0; y < this.height; y++) {
236
- gaps = [0, 0, 0, 0];
237
- lengths = [0, 0, 0, 0];
238
- for (let i = 0; i < length; i++) {
239
- for (let j = 0; j < 4; j++) {
240
- if (gaps[j] > maxGaps)
241
- continue;
242
- checkCell(x + i * DIRECTIONS[j].x, y + i * DIRECTIONS[j].y, j);
243
- if (lengths[j] === length)
244
- lineCount++;
245
- }
246
- }
247
- }
248
- }
249
- return lineCount;
250
- }
251
226
  /**
252
227
  * Gets a bit index from its coordinates and player ID.
253
228
  * @param move - The coordinates.
@@ -287,4 +262,4 @@ export default class Board {
287
262
  return position.x >= 0 && position.x < this.width && position.y >= 0 && position.y < this.height;
288
263
  }
289
264
  }
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"]}
265
+ //# 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    /**\n     * Cached result of the last `winner` computation.\n     * Set to `undefined` whenever the board changes so the getter recomputes.\n     */\n    private _cachedWinner: 0 | 1 | false | null | undefined = undefined;\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        if (this._cachedWinner !== undefined)\n            return this._cachedWinner;\n\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                this._cachedWinner = 0;\n\n                return 0;\n            }\n\n            if (playerTwoBoard.and(state).equals(state)) {\n                this._cachedWinner = 1;\n\n                return 1;\n            }\n        }\n\n        this._cachedWinner = this.isFull ? null : false;\n\n        return this._cachedWinner;\n    }\n\n    /**\n     * Returns a compact string key uniquely identifying the current board state.\n     * Used as a key in the alpha-beta transposition table.\n     * @returns The unique board state key.\n     */\n    public get hashKey(): string {\n        const { data } = this.bitBoard;\n\n        return data instanceof LongInt ? data.data.join(\",\") : String(data);\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        this._cachedWinner = undefined;\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        this._cachedWinner = undefined;\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     * 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"]}
@@ -4,36 +4,37 @@ import type LongInt from "../bitBoard/longInt.js";
4
4
  import type { Position } from "./board.js";
5
5
  export type PlayerType = "easyCPU" | "hardCPU" | "human" | "impossibleCPU" | "mediumCPU";
6
6
  export type Algorithm = "alphabeta" | "minimax";
7
- export type GameConstructorOptions<T, U extends LongInt | number> = {
8
- id: T;
7
+ /** Entry stored in the alpha-beta transposition table. */
8
+ type TTEntry = {
9
+ depth: number;
10
+ flag: "exact" | "lower" | "upper";
11
+ score: number;
12
+ };
13
+ export type GameConstructorOptions<T extends Board<LongInt | number>> = {
9
14
  onEnd?: (winner: number | null) => Promise<void> | void;
10
15
  onInvalidInput?: (position: Position) => Promise<void> | void;
11
- renderer?: (controller: Controller<T, U>) => Promise<void> | void;
16
+ renderer?: (controller: Controller<T>) => Promise<void> | void;
12
17
  };
13
- export type GameConstructor<T, U extends LongInt | number> = {
14
- new (playerOneType: PlayerType, playerTwoType: PlayerType, options: GameConstructorOptions<T, U>): Controller<T, U>;
18
+ export type GameConstructor<T extends Board<LongInt | number>> = {
19
+ new (playerOneType: PlayerType, playerTwoType: PlayerType, options?: GameConstructorOptions<T>): Controller<T>;
15
20
  };
16
21
  /**
17
22
  * Decorator to check that the constructor type for the given class is correct.
18
- * @template T - The type of the game ID.
19
- * @template U - The type of the board values.
23
+ * @template T - The type of board.
20
24
  * @param constructor - The class to check.
21
25
  */
22
- export declare function Game<T, U extends LongInt | number>(constructor: GameConstructor<T, U>): void;
26
+ export declare function Game<T extends Board<LongInt | number>>(constructor: GameConstructor<T>): void;
23
27
  /**
24
28
  * Represents a game controller.
25
- * @template T - The type of the game ID.
26
- * @template U - The type of the board values.
29
+ * @template T - The type of board.
27
30
  */
28
- export default abstract class Controller<T, U extends LongInt | number> extends EventEmitter<{
29
- end: GameConstructorOptions<T, U>["onEnd"];
30
- input: GameConstructorOptions<T, U>["onInvalidInput"];
31
- invalidInput: GameConstructorOptions<T, U>["onInvalidInput"];
31
+ export default abstract class Controller<T extends Board<LongInt | number>> extends EventEmitter<{
32
+ end: GameConstructorOptions<T>["onEnd"];
33
+ input: GameConstructorOptions<T>["onInvalidInput"];
34
+ invalidInput: GameConstructorOptions<T>["onInvalidInput"];
32
35
  }> {
33
- /** Contains the ID of the game. */
34
- readonly gameID: T;
35
36
  /** Contains the board. */
36
- readonly board: Board<U>;
37
+ readonly board: T;
37
38
  /** Contains the view object. */
38
39
  readonly render: () => Promise<void> | void;
39
40
  /** Contains the player objects. */
@@ -48,11 +49,10 @@ export default abstract class Controller<T, U extends LongInt | number> extends
48
49
  * @param playerTypes - The types of player for the game.
49
50
  * @param board - The board object.
50
51
  * @param render - The render function.
51
- * @param gameID - The ID of the game instance.
52
52
  * @param onEnd - The function to call when the game ends.
53
53
  * @param onInvalidInput - The function to call when the input is invalid.
54
54
  */
55
- protected constructor(playerTypes: PlayerType[], board: Board<U>, render: Required<GameConstructorOptions<T, U>>["renderer"], gameID: T, onEnd?: GameConstructorOptions<T, U>["onEnd"], onInvalidInput?: GameConstructorOptions<T, U>["onInvalidInput"]);
55
+ protected constructor(playerTypes: PlayerType[], board: T, render: Required<GameConstructorOptions<T>>["renderer"], onEnd?: GameConstructorOptions<T>["onEnd"], onInvalidInput?: GameConstructorOptions<T>["onInvalidInput"]);
56
56
  /**
57
57
  * Gets the current player object.
58
58
  * @returns The current player object.
@@ -79,13 +79,23 @@ export default abstract class Controller<T, U extends LongInt | number> extends
79
79
  };
80
80
  /**
81
81
  * The minimax algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).
82
+ *
83
+ * A transposition table (TT) is passed through all recursive calls so that
84
+ * identical board positions reached via different move orderings are not
85
+ * re-evaluated. The TT is created fresh on the root call (default parameter)
86
+ * and shared by every child, so each call to `findOptimalMove` starts clean.
87
+ *
88
+ * Each TT entry stores the score, the remaining depth when it was computed
89
+ * (deeper = more reliable), and a flag indicating whether the score is exact,
90
+ * a lower bound (beta cut-off), or an upper bound (never exceeded alpha).
82
91
  * @param depth - The depth of the algorithm.
83
92
  * @param alpha - The bounds for the alpha-beta variation of the algorithm.
84
93
  * @param beta - The bounds for the alpha-beta variation of the algorithm.
85
94
  * @param maximisingPlayer - Whether or not the current player is the maximising player.
95
+ * @param cache - The transposition table shared across the whole search.
86
96
  * @returns The optimal move.
87
97
  */
88
- protected alphabeta(depth?: number, alpha?: number, beta?: number, maximisingPlayer?: boolean): {
98
+ protected alphabeta(depth?: number, alpha?: number, beta?: number, maximisingPlayer?: boolean, cache?: Map<string, TTEntry>): {
89
99
  move: Position;
90
100
  score: number;
91
101
  };
@@ -117,3 +127,4 @@ export default abstract class Controller<T, U extends LongInt | number> extends
117
127
  randomMove?: Position;
118
128
  }): Position;
119
129
  }
130
+ export {};
@@ -1,8 +1,7 @@
1
1
  import { EventEmitter } from "eventemitter3";
2
2
  /**
3
3
  * Decorator to check that the constructor type for the given class is correct.
4
- * @template T - The type of the game ID.
5
- * @template U - The type of the board values.
4
+ * @template T - The type of board.
6
5
  * @param constructor - The class to check.
7
6
  */
8
7
  export function Game(constructor) {
@@ -10,12 +9,9 @@ export function Game(constructor) {
10
9
  }
11
10
  /**
12
11
  * Represents a game controller.
13
- * @template T - The type of the game ID.
14
- * @template U - The type of the board values.
12
+ * @template T - The type of board.
15
13
  */
16
14
  export default class Controller extends EventEmitter {
17
- /** Contains the ID of the game. */
18
- gameID;
19
15
  /** Contains the board. */
20
16
  board;
21
17
  /** Contains the view object. */
@@ -29,14 +25,11 @@ export default class Controller extends EventEmitter {
29
25
  * @param playerTypes - The types of player for the game.
30
26
  * @param board - The board object.
31
27
  * @param render - The render function.
32
- * @param gameID - The ID of the game instance.
33
28
  * @param onEnd - The function to call when the game ends.
34
29
  * @param onInvalidInput - The function to call when the input is invalid.
35
30
  */
36
- // eslint-disable-next-line @typescript-eslint/max-params
37
- constructor(playerTypes, board, render, gameID, onEnd, onInvalidInput) {
31
+ constructor(playerTypes, board, render, onEnd, onInvalidInput) {
38
32
  super();
39
- this.gameID = gameID;
40
33
  this.board = board;
41
34
  this.players = playerTypes.map((playerType, id) => ({ id, playerType }));
42
35
  this.render = render.bind(null, this);
@@ -89,7 +82,7 @@ export default class Controller extends EventEmitter {
89
82
  const { emptyCells } = this.board;
90
83
  for (const move of emptyCells) {
91
84
  this.board.makeMove(move, playerIds[Number(maximisingPlayer)]);
92
- const score = (9 - emptyCells.length) * this.minimax(depth - 1, !maximisingPlayer).score;
85
+ const { score } = this.minimax(depth - 1, !maximisingPlayer);
93
86
  this.board.undoLastMove();
94
87
  if (maximisingPlayer) {
95
88
  const bestScore = Math.max(score, bestMove.score);
@@ -106,14 +99,40 @@ export default class Controller extends EventEmitter {
106
99
  }
107
100
  /**
108
101
  * The minimax algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).
102
+ *
103
+ * A transposition table (TT) is passed through all recursive calls so that
104
+ * identical board positions reached via different move orderings are not
105
+ * re-evaluated. The TT is created fresh on the root call (default parameter)
106
+ * and shared by every child, so each call to `findOptimalMove` starts clean.
107
+ *
108
+ * Each TT entry stores the score, the remaining depth when it was computed
109
+ * (deeper = more reliable), and a flag indicating whether the score is exact,
110
+ * a lower bound (beta cut-off), or an upper bound (never exceeded alpha).
109
111
  * @param depth - The depth of the algorithm.
110
112
  * @param alpha - The bounds for the alpha-beta variation of the algorithm.
111
113
  * @param beta - The bounds for the alpha-beta variation of the algorithm.
112
114
  * @param maximisingPlayer - Whether or not the current player is the maximising player.
115
+ * @param cache - The transposition table shared across the whole search.
113
116
  * @returns The optimal move.
114
117
  */
115
- alphabeta(depth = Infinity, alpha = -Infinity, beta = Infinity, maximisingPlayer = true) {
118
+ // eslint-disable-next-line max-statements
119
+ alphabeta(depth = Infinity, alpha = -Infinity, beta = Infinity, maximisingPlayer = true, cache = new Map()) {
116
120
  const playerIds = [(this.currentPlayerId + 1) % 2, this.currentPlayerId];
121
+ const originalAlpha = alpha;
122
+ const key = `${this.board.hashKey}:${Number(maximisingPlayer)}`;
123
+ const cached = cache.get(key);
124
+ if (cached !== undefined && cached.depth >= depth) {
125
+ if (cached.flag === "exact")
126
+ return { move: { x: NaN, y: NaN }, score: cached.score };
127
+ if (cached.flag === "lower")
128
+ // eslint-disable-next-line no-param-reassign
129
+ alpha = Math.max(alpha, cached.score);
130
+ else
131
+ // eslint-disable-next-line no-param-reassign
132
+ beta = Math.min(beta, cached.score);
133
+ if (alpha >= beta)
134
+ return { move: { x: NaN, y: NaN }, score: cached.score };
135
+ }
117
136
  if (depth === 0 || this.board.winner !== false) {
118
137
  return {
119
138
  move: { x: NaN, y: NaN },
@@ -127,7 +146,7 @@ export default class Controller extends EventEmitter {
127
146
  const { emptyCells } = this.board;
128
147
  for (const move of emptyCells) {
129
148
  this.board.makeMove(move, playerIds[Number(maximisingPlayer)]);
130
- const score = (9 - emptyCells.length) * this.alphabeta(depth - 1, alpha, beta, !maximisingPlayer).score;
149
+ const { score } = this.alphabeta(depth - 1, alpha, beta, !maximisingPlayer, cache);
131
150
  this.board.undoLastMove();
132
151
  if (maximisingPlayer) {
133
152
  const bestScore = Math.max(score, bestMove.score);
@@ -148,6 +167,15 @@ export default class Controller extends EventEmitter {
148
167
  beta = Math.min(beta, bestScore);
149
168
  }
150
169
  }
170
+ // Store in the transposition table with the appropriate bound flag.
171
+ let flag;
172
+ if (bestMove.score <= originalAlpha)
173
+ flag = "upper";
174
+ else if (bestMove.score >= beta)
175
+ flag = "lower";
176
+ else
177
+ flag = "exact";
178
+ cache.set(key, { depth, flag, score: bestMove.score });
151
179
  return bestMove;
152
180
  }
153
181
  /** Changes which player's turn it is. */
@@ -178,4 +206,4 @@ export default class Controller extends EventEmitter {
178
206
  this.nextTurn();
179
207
  }
180
208
  }
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"controller.js","sourceRoot":"","sources":["../../src/base/controller.ts"],"names":[],"mappings":"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","sourcesContent":["import type Board from \"./board.js\";\nimport { EventEmitter } from \"eventemitter3\";\nimport type LongInt from \"../bitBoard/longInt.js\";\nimport type { Position } from \"./board.js\";\n\nexport type PlayerType = \"easyCPU\" | \"hardCPU\" | \"human\" | \"impossibleCPU\" | \"mediumCPU\";\nexport type Algorithm = \"alphabeta\" | \"minimax\";\nexport type GameConstructorOptions<T, U extends LongInt | number> = {\n    id: T;\n    onEnd?: (winner: number | null) => Promise<void> | void;\n    onInvalidInput?: (position: Position) => Promise<void> | void;\n    renderer?: (controller: Controller<T, U>) => Promise<void> | void;\n};\nexport type GameConstructor<T, U extends LongInt | number> = {\n    // eslint-disable-next-line @typescript-eslint/prefer-function-type\n    new(playerOneType: PlayerType, playerTwoType: PlayerType, options: GameConstructorOptions<T, U>): Controller<T, U>;\n};\n/**\n * Decorator to check that the constructor type for the given class is correct.\n * @template T - The type of the game ID.\n * @template U - The type of the board values.\n * @param constructor - The class to check.\n */\nexport function Game<T, U extends LongInt | number>(constructor: GameConstructor<T, U>): void {\n    void constructor;\n}\n\n/**\n * Represents a game controller.\n * @template T - The type of the game ID.\n * @template U - The type of the board values.\n */\nexport default abstract class Controller<T, U extends LongInt | number> extends EventEmitter<{\n    end: GameConstructorOptions<T, U>[\"onEnd\"];\n    input: GameConstructorOptions<T, U>[\"onInvalidInput\"];\n    invalidInput: GameConstructorOptions<T, U>[\"onInvalidInput\"];\n}> {\n    /** Contains the ID of the game. */\n    public readonly gameID: T;\n\n    /** Contains the board. */\n    public readonly board: Board<U>;\n\n    /** Contains the view object. */\n    public readonly render: () => Promise<void> | void;\n\n    /** Contains the player objects. */\n    protected readonly players: Array<{ id: number; playerType: PlayerType; }>;\n\n    /** Contains the ID of the current player. */\n    private currentPlayerId: number = 0;\n\n    /**\n     * Creates an instance of Controller.\n     * @param playerTypes - The types of player for the game.\n     * @param board - The board object.\n     * @param render - The render function.\n     * @param gameID - The ID of the game instance.\n     * @param onEnd - The function to call when the game ends.\n     * @param onInvalidInput - The function to call when the input is invalid.\n     */\n    // eslint-disable-next-line @typescript-eslint/max-params\n    protected constructor(\n        playerTypes: PlayerType[],\n        board: Board<U>,\n        render: Required<GameConstructorOptions<T, U>>[\"renderer\"],\n        gameID: T,\n        onEnd?: GameConstructorOptions<T, U>[\"onEnd\"],\n        onInvalidInput?: GameConstructorOptions<T, U>[\"onInvalidInput\"],\n    ) {\n        super();\n        this.gameID = gameID;\n        this.board = board;\n        this.players = playerTypes.map((playerType, id) => ({ id, playerType }));\n        this.render = render.bind(null, this);\n\n        if (onEnd !== undefined)\n            this.on(\"end\", onEnd as () => void);\n\n        if (onInvalidInput !== undefined)\n            this.on(\"invalidInput\", onInvalidInput as () => void);\n    }\n\n    /**\n     * Gets the current player object.\n     * @returns The current player object.\n     */\n    public get currentPlayer(): { id: number; playerType: PlayerType; } {\n        return this.players[this.currentPlayerId]!;\n    }\n\n    /**\n     * Controls the main game flow.\n     * @param algorithm - The algorithm to use.\n     * @returns The winner or null in the event of a tie.\n     */\n    public async play(algorithm: Algorithm = \"alphabeta\"): Promise<void> {\n        await this.render();\n        this.on(\"input\", (async (move: Position): Promise<void> => {\n            await this.makeMove(move);\n\n            if (this.currentPlayer.playerType !== \"human\")\n                await this.makeMove(algorithm);\n        }) as (move: Position) => void);\n\n        while (this.currentPlayer.playerType !== \"human\" && this.board.winner === false)\n            // eslint-disable-next-line no-await-in-loop\n            await this.makeMove(algorithm);\n    }\n\n    /**\n     * The bog standard minimax algorithm. Left in for reference (https://en.wikipedia.org/wiki/Minimax).\n     * @param depth - The depth of the algorithm.\n     * @param maximisingPlayer - Whether or not the current player is the maximising player.\n     * @returns The optimal move.\n     */\n    protected minimax(depth: number = Infinity, maximisingPlayer: boolean = true): { move: Position; score: number; } {\n        const playerIds = [(this.currentPlayerId + 1) % 2, this.currentPlayerId] as const;\n\n        if (depth === 0 || this.board.winner !== false) {\n            return {\n                move: { x: NaN, y: NaN },\n                score: this.board.heuristic * (this.currentPlayerId === 0 ? 1 : -1),\n            };\n        }\n\n        let bestMove: ReturnType<typeof this.minimax> = {\n            move: { x: NaN, y: NaN },\n            score: maximisingPlayer ? -Infinity : Infinity,\n        };\n        const { emptyCells } = this.board;\n\n        for (const move of emptyCells) {\n            this.board.makeMove(move, playerIds[Number(maximisingPlayer)]!);\n            const score = (9 - emptyCells.length) * this.minimax(depth - 1, !maximisingPlayer).score;\n\n            this.board.undoLastMove();\n\n            if (maximisingPlayer) {\n                const bestScore = Math.max(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n            } else {\n                const bestScore = Math.min(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n            }\n        }\n\n        return bestMove;\n    }\n\n    /**\n     * The minimax algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).\n     * @param depth - The depth of the algorithm.\n     * @param alpha - The bounds for the alpha-beta variation of the algorithm.\n     * @param beta - The bounds for the alpha-beta variation of the algorithm.\n     * @param maximisingPlayer - Whether or not the current player is the maximising player.\n     * @returns The optimal move.\n     */\n    protected alphabeta(\n        depth: number = Infinity,\n        alpha: number = -Infinity,\n        beta: number = Infinity,\n        maximisingPlayer: boolean = true,\n    ): { move: Position; score: number; } {\n        const playerIds = [(this.currentPlayerId + 1) % 2, this.currentPlayerId] as const;\n\n        if (depth === 0 || this.board.winner !== false) {\n            return {\n                move: { x: NaN, y: NaN },\n                score: this.board.heuristic * (this.currentPlayerId === 0 ? 1 : -1),\n            };\n        }\n\n        let bestMove: ReturnType<typeof this.alphabeta> = {\n            move: { x: NaN, y: NaN },\n            score: maximisingPlayer ? -Infinity : Infinity,\n        };\n        const { emptyCells } = this.board;\n\n        for (const move of emptyCells) {\n            this.board.makeMove(move, playerIds[Number(maximisingPlayer)]!);\n            const score = (9 - emptyCells.length) * this.alphabeta(depth - 1, alpha, beta, !maximisingPlayer).score;\n\n            this.board.undoLastMove();\n\n            if (maximisingPlayer) {\n                const bestScore = Math.max(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n\n                if (bestMove.score > beta)\n                    break;\n\n                // eslint-disable-next-line no-param-reassign\n                alpha = Math.max(alpha, bestScore);\n            } else {\n                const bestScore = Math.min(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n\n                if (bestMove.score < alpha)\n                    break;\n\n                // eslint-disable-next-line no-param-reassign\n                beta = Math.min(beta, bestScore);\n            }\n        }\n\n        return bestMove;\n    }\n\n    /** Changes which player's turn it is. */\n    private nextTurn(): void {\n        this.currentPlayerId = (this.currentPlayerId + 1) % this.players.length;\n    }\n\n    /**\n     * Makes a move.\n     * @param input - Either the algorithm to use to calculate the move or the move itself.\n     */\n    private async makeMove(input: Algorithm | Position): Promise<void> {\n        if (typeof input === \"string\") {\n            this.board.makeMove(this.determineCPUMove(this.currentPlayer.playerType, input), this.currentPlayer.id);\n        } else {\n            if (this.currentPlayer.playerType !== \"human\" || !this.board.moveIsValid(input)) {\n                this.emit(\"invalidInput\", input);\n\n                return;\n            }\n\n            this.board.makeMove(input, this.currentPlayer.id);\n        }\n\n        await this.render();\n        const { winner } = this.board;\n\n        if (winner !== false) {\n            this.emit(\"end\", winner);\n\n            return;\n        }\n\n        this.nextTurn();\n    }\n\n    /**\n     * Determines the CPU move.\n     * @param difficulty - The difficulty of the AI.\n     * @param algorithm - The algorithm to use.\n     * @returns The move.\n     */\n    public abstract determineCPUMove(difficulty: Omit<PlayerType, \"human\">, algorithm?: Algorithm): Position;\n\n    /**\n     * Finds the optimal move.\n     * @param options - The options to use.\n     * @param options.algorithm - The algorithm to use.\n     * @param options.maxDepth - The maximum depth to search.\n     * @param options.randomMove - A random move to make.\n     * @returns The optimal move to make.\n     */\n    public abstract findOptimalMove(options?: {\n        algorithm?: Algorithm;\n        maxDepth?: number;\n        randomMove?: Position;\n    }): Position;\n}\n"]}
209
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"controller.js","sourceRoot":"","sources":["../../src/base/controller.ts"],"names":[],"mappings":"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","sourcesContent":["import type Board from \"./board.js\";\nimport { EventEmitter } from \"eventemitter3\";\nimport type LongInt from \"../bitBoard/longInt.js\";\nimport type { Position } from \"./board.js\";\n\nexport type PlayerType = \"easyCPU\" | \"hardCPU\" | \"human\" | \"impossibleCPU\" | \"mediumCPU\";\nexport type Algorithm = \"alphabeta\" | \"minimax\";\n\n/** Entry stored in the alpha-beta transposition table. */\ntype TTEntry = { depth: number; flag: \"exact\" | \"lower\" | \"upper\"; score: number; };\n\nexport type GameConstructorOptions<T extends Board<LongInt | number>> = {\n    onEnd?: (winner: number | null) => Promise<void> | void;\n    onInvalidInput?: (position: Position) => Promise<void> | void;\n    renderer?: (controller: Controller<T>) => Promise<void> | void;\n};\nexport type GameConstructor<T extends Board<LongInt | number>> = {\n    // eslint-disable-next-line @typescript-eslint/prefer-function-type\n    new(playerOneType: PlayerType, playerTwoType: PlayerType, options?: GameConstructorOptions<T>): Controller<T>;\n};\n/**\n * Decorator to check that the constructor type for the given class is correct.\n * @template T - The type of board.\n * @param constructor - The class to check.\n */\nexport function Game<T extends Board<LongInt | number>>(constructor: GameConstructor<T>): void {\n    void constructor;\n}\n\n/**\n * Represents a game controller.\n * @template T - The type of board.\n */\nexport default abstract class Controller<T extends Board<LongInt | number>> extends EventEmitter<{\n    end: GameConstructorOptions<T>[\"onEnd\"];\n    input: GameConstructorOptions<T>[\"onInvalidInput\"];\n    invalidInput: GameConstructorOptions<T>[\"onInvalidInput\"];\n}> {\n    /** Contains the board. */\n    public readonly board: T;\n\n    /** Contains the view object. */\n    public readonly render: () => Promise<void> | void;\n\n    /** Contains the player objects. */\n    protected readonly players: Array<{ id: number; playerType: PlayerType; }>;\n\n    /** Contains the ID of the current player. */\n    private currentPlayerId: number = 0;\n\n    /**\n     * Creates an instance of Controller.\n     * @param playerTypes - The types of player for the game.\n     * @param board - The board object.\n     * @param render - The render function.\n     * @param onEnd - The function to call when the game ends.\n     * @param onInvalidInput - The function to call when the input is invalid.\n     */\n\n    protected constructor(\n        playerTypes: PlayerType[],\n        board: T,\n        render: Required<GameConstructorOptions<T>>[\"renderer\"],\n        onEnd?: GameConstructorOptions<T>[\"onEnd\"],\n        onInvalidInput?: GameConstructorOptions<T>[\"onInvalidInput\"],\n    ) {\n        super();\n        this.board = board;\n        this.players = playerTypes.map((playerType, id) => ({ id, playerType }));\n        this.render = render.bind(null, this);\n\n        if (onEnd !== undefined)\n            this.on(\"end\", onEnd as () => void);\n\n        if (onInvalidInput !== undefined)\n            this.on(\"invalidInput\", onInvalidInput as () => void);\n    }\n\n    /**\n     * Gets the current player object.\n     * @returns The current player object.\n     */\n    public get currentPlayer(): { id: number; playerType: PlayerType; } {\n        return this.players[this.currentPlayerId]!;\n    }\n\n    /**\n     * Controls the main game flow.\n     * @param algorithm - The algorithm to use.\n     * @returns The winner or null in the event of a tie.\n     */\n    public async play(algorithm: Algorithm = \"alphabeta\"): Promise<void> {\n        await this.render();\n        this.on(\"input\", (async (move: Position): Promise<void> => {\n            await this.makeMove(move);\n\n            if (this.currentPlayer.playerType !== \"human\")\n                await this.makeMove(algorithm);\n        }) as (move: Position) => void);\n\n        while (this.currentPlayer.playerType !== \"human\" && this.board.winner === false)\n            // eslint-disable-next-line no-await-in-loop\n            await this.makeMove(algorithm);\n    }\n\n    /**\n     * The bog standard minimax algorithm. Left in for reference (https://en.wikipedia.org/wiki/Minimax).\n     * @param depth - The depth of the algorithm.\n     * @param maximisingPlayer - Whether or not the current player is the maximising player.\n     * @returns The optimal move.\n     */\n    protected minimax(depth: number = Infinity, maximisingPlayer: boolean = true): { move: Position; score: number; } {\n        const playerIds = [(this.currentPlayerId + 1) % 2, this.currentPlayerId] as const;\n\n        if (depth === 0 || this.board.winner !== false) {\n            return {\n                move: { x: NaN, y: NaN },\n                score: this.board.heuristic * (this.currentPlayerId === 0 ? 1 : -1),\n            };\n        }\n\n        let bestMove: ReturnType<typeof this.minimax> = {\n            move: { x: NaN, y: NaN },\n            score: maximisingPlayer ? -Infinity : Infinity,\n        };\n        const { emptyCells } = this.board;\n\n        for (const move of emptyCells) {\n            this.board.makeMove(move, playerIds[Number(maximisingPlayer)]!);\n            const { score } = this.minimax(depth - 1, !maximisingPlayer);\n\n            this.board.undoLastMove();\n\n            if (maximisingPlayer) {\n                const bestScore = Math.max(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n            } else {\n                const bestScore = Math.min(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n            }\n        }\n\n        return bestMove;\n    }\n\n    /**\n     * The minimax algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).\n     *\n     * A transposition table (TT) is passed through all recursive calls so that\n     * identical board positions reached via different move orderings are not\n     * re-evaluated. The TT is created fresh on the root call (default parameter)\n     * and shared by every child, so each call to `findOptimalMove` starts clean.\n     *\n     * Each TT entry stores the score, the remaining depth when it was computed\n     * (deeper = more reliable), and a flag indicating whether the score is exact,\n     * a lower bound (beta cut-off), or an upper bound (never exceeded alpha).\n     * @param depth - The depth of the algorithm.\n     * @param alpha - The bounds for the alpha-beta variation of the algorithm.\n     * @param beta - The bounds for the alpha-beta variation of the algorithm.\n     * @param maximisingPlayer - Whether or not the current player is the maximising player.\n     * @param cache - The transposition table shared across the whole search.\n     * @returns The optimal move.\n     */\n    // eslint-disable-next-line max-statements\n    protected alphabeta(\n        depth: number = Infinity,\n        alpha: number = -Infinity,\n        beta: number = Infinity,\n        maximisingPlayer: boolean = true,\n        cache: Map<string, TTEntry> = new Map<string, TTEntry>(),\n    ): { move: Position; score: number; } {\n        const playerIds = [(this.currentPlayerId + 1) % 2, this.currentPlayerId];\n        const originalAlpha = alpha;\n\n        const key = `${this.board.hashKey}:${Number(maximisingPlayer)}`;\n        const cached = cache.get(key);\n\n        if (cached !== undefined && cached.depth >= depth) {\n            if (cached.flag === \"exact\")\n                return { move: { x: NaN, y: NaN }, score: cached.score };\n\n            if (cached.flag === \"lower\")\n                // eslint-disable-next-line no-param-reassign\n                alpha = Math.max(alpha, cached.score);\n            else\n                // eslint-disable-next-line no-param-reassign\n                beta = Math.min(beta, cached.score);\n\n            if (alpha >= beta)\n                return { move: { x: NaN, y: NaN }, score: cached.score };\n        }\n\n        if (depth === 0 || this.board.winner !== false) {\n            return {\n                move: { x: NaN, y: NaN },\n                score: this.board.heuristic * (this.currentPlayerId === 0 ? 1 : -1),\n            };\n        }\n\n        let bestMove: ReturnType<typeof this.alphabeta> = {\n            move: { x: NaN, y: NaN },\n            score: maximisingPlayer ? -Infinity : Infinity,\n        };\n        const { emptyCells } = this.board;\n\n        for (const move of emptyCells) {\n            this.board.makeMove(move, playerIds[Number(maximisingPlayer)]!);\n            const { score } = this.alphabeta(depth - 1, alpha, beta, !maximisingPlayer, cache);\n\n            this.board.undoLastMove();\n\n            if (maximisingPlayer) {\n                const bestScore = Math.max(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n\n                if (bestMove.score > beta)\n                    break;\n\n                // eslint-disable-next-line no-param-reassign\n                alpha = Math.max(alpha, bestScore);\n            } else {\n                const bestScore = Math.min(score, bestMove.score);\n\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n\n                if (bestMove.score < alpha)\n                    break;\n\n                // eslint-disable-next-line no-param-reassign\n                beta = Math.min(beta, bestScore);\n            }\n        }\n\n        // Store in the transposition table with the appropriate bound flag.\n        let flag: \"exact\" | \"lower\" | \"upper\";\n\n        if (bestMove.score <= originalAlpha)\n            flag = \"upper\";\n        else if (bestMove.score >= beta)\n            flag = \"lower\";\n        else\n            flag = \"exact\";\n\n        cache.set(key, { depth, flag, score: bestMove.score });\n\n        return bestMove;\n    }\n\n    /** Changes which player's turn it is. */\n    private nextTurn(): void {\n        this.currentPlayerId = (this.currentPlayerId + 1) % this.players.length;\n    }\n\n    /**\n     * Makes a move.\n     * @param input - Either the algorithm to use to calculate the move or the move itself.\n     */\n    private async makeMove(input: Algorithm | Position): Promise<void> {\n        if (typeof input === \"string\") {\n            this.board.makeMove(this.determineCPUMove(this.currentPlayer.playerType, input), this.currentPlayer.id);\n        } else {\n            if (this.currentPlayer.playerType !== \"human\" || !this.board.moveIsValid(input)) {\n                this.emit(\"invalidInput\", input);\n\n                return;\n            }\n\n            this.board.makeMove(input, this.currentPlayer.id);\n        }\n\n        await this.render();\n        const { winner } = this.board;\n\n        if (winner !== false) {\n            this.emit(\"end\", winner);\n\n            return;\n        }\n\n        this.nextTurn();\n    }\n\n    /**\n     * Determines the CPU move.\n     * @param difficulty - The difficulty of the AI.\n     * @param algorithm - The algorithm to use.\n     * @returns The move.\n     */\n    public abstract determineCPUMove(difficulty: Omit<PlayerType, \"human\">, algorithm?: Algorithm): Position;\n\n    /**\n     * Finds the optimal move.\n     * @param options - The options to use.\n     * @param options.algorithm - The algorithm to use.\n     * @param options.maxDepth - The maximum depth to search.\n     * @param options.randomMove - A random move to make.\n     * @returns The optimal move to make.\n     */\n    public abstract findOptimalMove(options?: {\n        algorithm?: Algorithm;\n        maxDepth?: number;\n        randomMove?: Position;\n    }): Position;\n}\n"]}