@oathompsonjones/mini-games 1.0.16 → 1.0.17
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/build/base/board.d.ts +9 -4
- package/build/base/board.js +22 -8
- package/build/base/controller.d.ts +16 -8
- package/build/base/controller.js +68 -38
- package/build/games/connect4/board.d.ts +1 -0
- package/build/games/connect4/board.js +14 -10
- package/build/games/connect4/controller.js +2 -2
- package/build/games/tictactoe/controller.js +2 -2
- package/package.json +1 -1
package/build/base/board.d.ts
CHANGED
|
@@ -30,7 +30,9 @@ export default abstract class Board<T extends LongInt | number> {
|
|
|
30
30
|
/** The height of the game board. */
|
|
31
31
|
protected readonly height: number;
|
|
32
32
|
/** A stack of moves. */
|
|
33
|
-
protected readonly moves:
|
|
33
|
+
protected readonly moves: Array<Position & {
|
|
34
|
+
playerId: number;
|
|
35
|
+
}>;
|
|
34
36
|
/**
|
|
35
37
|
* Cached result of the last `winner` computation.
|
|
36
38
|
* Set to `undefined` whenever the board changes so the getter recomputes.
|
|
@@ -40,6 +42,10 @@ export default abstract class Board<T extends LongInt | number> {
|
|
|
40
42
|
private readonly numberOfPlayerBoards;
|
|
41
43
|
/** Number of boards in total (most likely also 2). */
|
|
42
44
|
private readonly numberOfBoards;
|
|
45
|
+
/** The zobrist hash table. */
|
|
46
|
+
private readonly zobrist;
|
|
47
|
+
/** The hash for the current board state. */
|
|
48
|
+
private _hash;
|
|
43
49
|
/** The board states which represent a winning state. */
|
|
44
50
|
protected abstract readonly winningStates: Array<BitBoard<T>>;
|
|
45
51
|
/**
|
|
@@ -66,11 +72,10 @@ export default abstract class Board<T extends LongInt | number> {
|
|
|
66
72
|
*/
|
|
67
73
|
get winner(): 0 | 1 | false | null;
|
|
68
74
|
/**
|
|
69
|
-
*
|
|
70
|
-
* Used as a key in the alpha-beta transposition table.
|
|
75
|
+
* Gets a unique key representing the current board state.
|
|
71
76
|
* @returns The unique board state key.
|
|
72
77
|
*/
|
|
73
|
-
get
|
|
78
|
+
get hash(): number;
|
|
74
79
|
/**
|
|
75
80
|
* Calculates which cells are empty.
|
|
76
81
|
* @returns The empty cells on the board.
|
package/build/base/board.js
CHANGED
|
@@ -23,6 +23,10 @@ export default class Board {
|
|
|
23
23
|
numberOfPlayerBoards;
|
|
24
24
|
/** Number of boards in total (most likely also 2). */
|
|
25
25
|
numberOfBoards;
|
|
26
|
+
/** The zobrist hash table. */
|
|
27
|
+
zobrist = [];
|
|
28
|
+
/** The hash for the current board state. */
|
|
29
|
+
_hash = 0;
|
|
26
30
|
/**
|
|
27
31
|
* Creates an instance of Board.
|
|
28
32
|
* @param width - The width of the game board.
|
|
@@ -39,6 +43,15 @@ export default class Board {
|
|
|
39
43
|
this.bitBoard = (totalBits > 32
|
|
40
44
|
? new LongIntBitBoard(Math.ceil(totalBits / 32))
|
|
41
45
|
: new IntBitBoard());
|
|
46
|
+
for (let x = 0; x < width; x++) {
|
|
47
|
+
this.zobrist[x] = [];
|
|
48
|
+
for (let y = 0; y < height; y++) {
|
|
49
|
+
this.zobrist[x][y] = [
|
|
50
|
+
Math.floor(Math.random() * 0xFFFFFFFF),
|
|
51
|
+
Math.floor(Math.random() * 0xFFFFFFFF),
|
|
52
|
+
];
|
|
53
|
+
}
|
|
54
|
+
}
|
|
42
55
|
}
|
|
43
56
|
/**
|
|
44
57
|
* Calculates whether or not the board is full.
|
|
@@ -93,13 +106,11 @@ export default class Board {
|
|
|
93
106
|
return this._cachedWinner;
|
|
94
107
|
}
|
|
95
108
|
/**
|
|
96
|
-
*
|
|
97
|
-
* Used as a key in the alpha-beta transposition table.
|
|
109
|
+
* Gets a unique key representing the current board state.
|
|
98
110
|
* @returns The unique board state key.
|
|
99
111
|
*/
|
|
100
|
-
get
|
|
101
|
-
|
|
102
|
-
return data instanceof LongInt ? data.data.join(",") : String(data);
|
|
112
|
+
get hash() {
|
|
113
|
+
return this._hash;
|
|
103
114
|
}
|
|
104
115
|
/**
|
|
105
116
|
* Calculates which cells are empty.
|
|
@@ -123,8 +134,9 @@ export default class Board {
|
|
|
123
134
|
makeMove(move, playerId) {
|
|
124
135
|
this._cachedWinner = undefined;
|
|
125
136
|
const bit = this.getBitIndex(move, playerId);
|
|
126
|
-
this.moves.push(
|
|
137
|
+
this.moves.push({ ...move, playerId });
|
|
127
138
|
this.bitBoard.setBit(bit);
|
|
139
|
+
this._hash ^= this.zobrist[move.x][move.y][playerId];
|
|
128
140
|
}
|
|
129
141
|
/**
|
|
130
142
|
* Reverses the last move.
|
|
@@ -135,7 +147,9 @@ export default class Board {
|
|
|
135
147
|
const lastMove = this.moves.pop();
|
|
136
148
|
if (lastMove === undefined)
|
|
137
149
|
throw new Error("No move to undo.");
|
|
138
|
-
this.
|
|
150
|
+
const bit = this.getBitIndex(lastMove, lastMove.playerId);
|
|
151
|
+
this.bitBoard.clearBit(bit);
|
|
152
|
+
this._hash ^= this.zobrist[lastMove.x][lastMove.y][lastMove.playerId];
|
|
139
153
|
}
|
|
140
154
|
/**
|
|
141
155
|
* Checks if a move is valid.
|
|
@@ -262,4 +276,4 @@ export default class Board {
|
|
|
262
276
|
return position.x >= 0 && position.x < this.width && position.y >= 0 && position.y < this.height;
|
|
263
277
|
}
|
|
264
278
|
}
|
|
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"]}
|
|
279
|
+
//# 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: Array<Position & { playerId: 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 zobrist hash table. */\n    private readonly zobrist: number[][][] = [];\n\n    /** The hash for the current board state. */\n    private _hash: number = 0;\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        for (let x = 0; x < width; x++) {\n            this.zobrist[x] = [];\n            for (let y = 0; y < height; y++) {\n                this.zobrist[x]![y] = [\n                    Math.floor(Math.random() * 0xFFFFFFFF),\n                    Math.floor(Math.random() * 0xFFFFFFFF),\n                ];\n            }\n        }\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     * Gets a unique key representing the current board state.\n     * @returns The unique board state key.\n     */\n    public get hash(): number {\n        return this._hash;\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({ ...move, playerId });\n        this.bitBoard.setBit(bit);\n        this._hash ^= this.zobrist[move.x]![move.y]![playerId]!;\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        const bit = this.getBitIndex(lastMove, lastMove.playerId);\n\n        this.bitBoard.clearBit(bit);\n        this._hash ^= this.zobrist[lastMove.x]![lastMove.y]![lastMove.playerId]!;\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"]}
|
|
@@ -37,8 +37,10 @@ export default abstract class Controller<T extends Board<LongInt | number>> exte
|
|
|
37
37
|
}>;
|
|
38
38
|
/** Contains the ID of the current player. */
|
|
39
39
|
private currentPlayerId;
|
|
40
|
-
/**
|
|
41
|
-
private readonly
|
|
40
|
+
/** Contains the transposition table for the minimax algorithm. */
|
|
41
|
+
private readonly transposition;
|
|
42
|
+
/** Contains the best moves for the minimax algorithm. */
|
|
43
|
+
private readonly bestMoves;
|
|
42
44
|
/**
|
|
43
45
|
* Creates an instance of Controller.
|
|
44
46
|
* @param playerTypes - The types of player for the game.
|
|
@@ -62,17 +64,23 @@ export default abstract class Controller<T extends Board<LongInt | number>> exte
|
|
|
62
64
|
*/
|
|
63
65
|
play(): Promise<void>;
|
|
64
66
|
/**
|
|
65
|
-
*
|
|
67
|
+
* A wrapper for the minimax algorithm to find the optimal move for the current board state.
|
|
68
|
+
* @param maxDepth - The maximum depth to search.
|
|
69
|
+
* @returns The optimal move for the current board state.
|
|
70
|
+
*/
|
|
71
|
+
protected search(maxDepth: number): {
|
|
72
|
+
move: Position;
|
|
73
|
+
score: number;
|
|
74
|
+
};
|
|
75
|
+
/**
|
|
76
|
+
* The minimax (negamax) algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).
|
|
66
77
|
* @param depth - The depth of the algorithm.
|
|
67
78
|
* @param alpha - The bounds for the alpha-beta variation of the algorithm.
|
|
68
79
|
* @param beta - The bounds for the alpha-beta variation of the algorithm.
|
|
69
|
-
* @param
|
|
80
|
+
* @param playerId - The player to calculate the move for.
|
|
70
81
|
* @returns The optimal move.
|
|
71
82
|
*/
|
|
72
|
-
|
|
73
|
-
move: Position;
|
|
74
|
-
score: number;
|
|
75
|
-
};
|
|
83
|
+
private minimax;
|
|
76
84
|
/** Changes which player's turn it is. */
|
|
77
85
|
private nextTurn;
|
|
78
86
|
/**
|
package/build/base/controller.js
CHANGED
|
@@ -20,8 +20,10 @@ export default class Controller extends EventEmitter {
|
|
|
20
20
|
players;
|
|
21
21
|
/** Contains the ID of the current player. */
|
|
22
22
|
currentPlayerId = 0;
|
|
23
|
-
/**
|
|
24
|
-
|
|
23
|
+
/** Contains the transposition table for the minimax algorithm. */
|
|
24
|
+
transposition = new Map();
|
|
25
|
+
/** Contains the best moves for the minimax algorithm. */
|
|
26
|
+
bestMoves = new Map();
|
|
25
27
|
/**
|
|
26
28
|
* Creates an instance of Controller.
|
|
27
29
|
* @param playerTypes - The types of player for the game.
|
|
@@ -63,53 +65,81 @@ export default class Controller extends EventEmitter {
|
|
|
63
65
|
await this.makeMove();
|
|
64
66
|
}
|
|
65
67
|
/**
|
|
66
|
-
*
|
|
68
|
+
* A wrapper for the minimax algorithm to find the optimal move for the current board state.
|
|
69
|
+
* @param maxDepth - The maximum depth to search.
|
|
70
|
+
* @returns The optimal move for the current board state.
|
|
71
|
+
*/
|
|
72
|
+
search(maxDepth) {
|
|
73
|
+
let best = {
|
|
74
|
+
move: { x: NaN, y: NaN },
|
|
75
|
+
score: -Infinity,
|
|
76
|
+
};
|
|
77
|
+
for (let depth = 1; depth <= maxDepth; depth++)
|
|
78
|
+
best = this.minimax(depth, -Infinity, Infinity, this.currentPlayerId);
|
|
79
|
+
return best;
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* The minimax (negamax) algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).
|
|
67
83
|
* @param depth - The depth of the algorithm.
|
|
68
84
|
* @param alpha - The bounds for the alpha-beta variation of the algorithm.
|
|
69
85
|
* @param beta - The bounds for the alpha-beta variation of the algorithm.
|
|
70
|
-
* @param
|
|
86
|
+
* @param playerId - The player to calculate the move for.
|
|
71
87
|
* @returns The optimal move.
|
|
72
88
|
*/
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
89
|
+
// eslint-disable-next-line max-statements
|
|
90
|
+
minimax(depth, alpha, beta, playerId) {
|
|
91
|
+
const { board } = this;
|
|
92
|
+
const { hash } = board;
|
|
93
|
+
const cached = this.transposition.get(hash);
|
|
94
|
+
if (cached && cached.depth >= depth) {
|
|
95
|
+
return {
|
|
96
|
+
move: this.bestMoves.get(hash) ?? { x: NaN, y: NaN },
|
|
97
|
+
score: cached.score,
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
if (depth === 0 || board.winner !== false) {
|
|
101
|
+
const score = board.heuristic * (this.currentPlayerId === 0 ? 1 : -1);
|
|
102
|
+
this.transposition.set(hash, { depth, score });
|
|
76
103
|
return {
|
|
77
104
|
move: { x: NaN, y: NaN },
|
|
78
|
-
score
|
|
105
|
+
score,
|
|
79
106
|
};
|
|
80
107
|
}
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
108
|
+
const moves = [...board.emptyCells];
|
|
109
|
+
// Move ordering: try previously best move first
|
|
110
|
+
const bestStored = this.bestMoves.get(hash);
|
|
111
|
+
if (bestStored) {
|
|
112
|
+
const index = moves.findIndex((m) => m.x === bestStored.x && m.y === bestStored.y);
|
|
113
|
+
if (index > 0)
|
|
114
|
+
moves.unshift(moves.splice(index, 1)[0]);
|
|
115
|
+
}
|
|
116
|
+
let bestScore = -Infinity;
|
|
117
|
+
let bestMove = null;
|
|
118
|
+
// eslint-disable-next-line @typescript-eslint/prefer-for-of
|
|
119
|
+
for (let i = 0; i < moves.length; i++) {
|
|
120
|
+
const move = moves[i];
|
|
121
|
+
board.makeMove(move, playerId);
|
|
122
|
+
const { score } = this.minimax(depth - 1, -beta, -alpha, playerId ^ 1);
|
|
123
|
+
const value = -score;
|
|
124
|
+
board.undoLastMove();
|
|
125
|
+
if (value > bestScore) {
|
|
126
|
+
bestScore = value;
|
|
127
|
+
bestMove = move;
|
|
100
128
|
}
|
|
101
|
-
|
|
102
|
-
const bestScore = Math.min(score, bestMove.score);
|
|
103
|
-
if (bestScore !== bestMove.score)
|
|
104
|
-
bestMove = { move, score };
|
|
105
|
-
if (bestMove.score < alpha)
|
|
106
|
-
break;
|
|
129
|
+
if (value > alpha)
|
|
107
130
|
// eslint-disable-next-line no-param-reassign
|
|
108
|
-
|
|
109
|
-
|
|
131
|
+
alpha = value;
|
|
132
|
+
if (alpha >= beta)
|
|
133
|
+
break;
|
|
134
|
+
}
|
|
135
|
+
if (bestMove) {
|
|
136
|
+
this.transposition.set(hash, { depth, score: bestScore });
|
|
137
|
+
this.bestMoves.set(hash, bestMove);
|
|
110
138
|
}
|
|
111
|
-
|
|
112
|
-
|
|
139
|
+
return {
|
|
140
|
+
move: bestMove ?? { x: NaN, y: NaN },
|
|
141
|
+
score: bestScore,
|
|
142
|
+
};
|
|
113
143
|
}
|
|
114
144
|
/** Changes which player's turn it is. */
|
|
115
145
|
nextTurn() {
|
|
@@ -139,4 +169,4 @@ export default class Controller extends EventEmitter {
|
|
|
139
169
|
this.nextTurn();
|
|
140
170
|
}
|
|
141
171
|
}
|
|
142
|
-
//# 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\";\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    /** Stores precomputed values to optimise minimax/alphabeta. */\n    private readonly cache = new Map<string, { move: Position; score: number; }>();\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     * @returns The winner or null in the event of a tie.\n     */\n    public async play(): 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();\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();\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];\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        if (this.cache.has(this.board.toString()))\n            return this.cache.get(this.board.toString())!;\n\n        let bestMove = {\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);\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        this.cache.set(this.board.toString(), bestMove);\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 - The move to make. If ommitted, the move will be calculated for the CPU.\n     */\n    private async makeMove(input?: Position): Promise<void> {\n        if (input === undefined) {\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.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        maxDepth?: number;\n        randomMove?: Position;\n    }): Position;\n}\n"]}
|
|
172
|
+
//# 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\";\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    /** Contains the transposition table for the minimax algorithm. */\n    private readonly transposition = new Map<number, { depth: number; score: number; }>();\n\n    /** Contains the best moves for the minimax algorithm. */\n    private readonly bestMoves = new Map<number, Position>();\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     * @returns The winner or null in the event of a tie.\n     */\n    public async play(): 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();\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();\n    }\n\n    /**\n     * A wrapper for the minimax algorithm to find the optimal move for the current board state.\n     * @param maxDepth - The maximum depth to search.\n     * @returns The optimal move for the current board state.\n     */\n    protected search(maxDepth: number): { move: Position; score: number; } {\n        let best: { move: Position; score: number; } = {\n            move: { x: NaN, y: NaN },\n            score: -Infinity,\n        };\n\n        for (let depth = 1; depth <= maxDepth; depth++)\n            best = this.minimax(depth, -Infinity, Infinity, this.currentPlayerId);\n\n        return best;\n    }\n\n    /**\n     * The minimax (negamax) 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 playerId - The player to calculate the move for.\n     * @returns The optimal move.\n     */\n    // eslint-disable-next-line max-statements\n    private minimax(depth: number, alpha: number, beta: number, playerId: number): { move: Position; score: number; } {\n        const { board } = this;\n        const { hash } = board;\n\n        const cached = this.transposition.get(hash);\n\n        if (cached && cached.depth >= depth) {\n            return {\n                move: this.bestMoves.get(hash) ?? { x: NaN, y: NaN },\n                score: cached.score,\n            };\n        }\n\n        if (depth === 0 || board.winner !== false) {\n            const score = board.heuristic * (this.currentPlayerId === 0 ? 1 : -1);\n\n            this.transposition.set(hash, { depth, score });\n\n            return {\n                move: { x: NaN, y: NaN },\n                score,\n            };\n        }\n\n        const moves = [...board.emptyCells];\n\n        // Move ordering: try previously best move first\n        const bestStored = this.bestMoves.get(hash);\n\n        if (bestStored) {\n            const index = moves.findIndex((m) => m.x === bestStored.x && m.y === bestStored.y);\n\n            if (index > 0)\n                moves.unshift(moves.splice(index, 1)[0]!);\n        }\n\n        let bestScore = -Infinity;\n        let bestMove: Position | null = null;\n\n        // eslint-disable-next-line @typescript-eslint/prefer-for-of\n        for (let i = 0; i < moves.length; i++) {\n            const move = moves[i]!;\n\n            board.makeMove(move, playerId);\n\n            const { score } = this.minimax(\n                depth - 1,\n                -beta,\n                -alpha,\n                playerId ^ 1,\n            );\n\n            const value = -score;\n\n            board.undoLastMove();\n\n            if (value > bestScore) {\n                bestScore = value;\n                bestMove = move;\n            }\n\n            if (value > alpha)\n                // eslint-disable-next-line no-param-reassign\n                alpha = value;\n\n            if (alpha >= beta)\n                break;\n        }\n\n        if (bestMove) {\n            this.transposition.set(hash, { depth, score: bestScore });\n            this.bestMoves.set(hash, bestMove);\n        }\n\n        return {\n            move: bestMove ?? { x: NaN, y: NaN },\n            score: bestScore,\n        };\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 - The move to make. If ommitted, the move will be calculated for the CPU.\n     */\n    private async makeMove(input?: Position): Promise<void> {\n        if (input === undefined) {\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.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        maxDepth?: number;\n        randomMove?: Position;\n    }): Position;\n}\n"]}
|
|
@@ -20,6 +20,7 @@ export default class Board extends Base<LongInt> {
|
|
|
20
20
|
constructor();
|
|
21
21
|
/**
|
|
22
22
|
* Calculates the heuristic score for the current board state.
|
|
23
|
+
* Favors blocking opponent's 3s over making own 3s.
|
|
23
24
|
* @returns The heuristic score.
|
|
24
25
|
*/
|
|
25
26
|
get heuristic(): number;
|
|
@@ -49,6 +49,7 @@ export default class Board extends Base {
|
|
|
49
49
|
}
|
|
50
50
|
/**
|
|
51
51
|
* Calculates the heuristic score for the current board state.
|
|
52
|
+
* Favors blocking opponent's 3s over making own 3s.
|
|
52
53
|
* @returns The heuristic score.
|
|
53
54
|
*/
|
|
54
55
|
// eslint-disable-next-line max-statements
|
|
@@ -60,6 +61,9 @@ export default class Board extends Base {
|
|
|
60
61
|
return -(10000 - this.moves.length);
|
|
61
62
|
let p0Score = 0;
|
|
62
63
|
let p1Score = 0;
|
|
64
|
+
// Scoring: defensive moves (blocking) weighted 1.5x higher than offensive moves
|
|
65
|
+
const offensiveScores = [0, 1, 10, 100];
|
|
66
|
+
const defensiveScores = [0, 1.5, 15, 150];
|
|
63
67
|
// Horizontal →
|
|
64
68
|
for (let y = 0; y < this.height; y++) {
|
|
65
69
|
for (let x = 0; x < this.width - 3; x++) {
|
|
@@ -72,9 +76,9 @@ export default class Board extends Base {
|
|
|
72
76
|
const p0Count = window.filter((occupier) => occupier === 0).length;
|
|
73
77
|
const p1Count = window.filter((occupier) => occupier === 1).length;
|
|
74
78
|
if (p0Count > 0 && p1Count === 0)
|
|
75
|
-
p0Score += [
|
|
79
|
+
p0Score += defensiveScores[p0Count];
|
|
76
80
|
else if (p1Count > 0 && p0Count === 0)
|
|
77
|
-
p1Score += [
|
|
81
|
+
p1Score += offensiveScores[p1Count];
|
|
78
82
|
}
|
|
79
83
|
}
|
|
80
84
|
// Vertical ↓
|
|
@@ -89,9 +93,9 @@ export default class Board extends Base {
|
|
|
89
93
|
const p0Count = window.filter((occupier) => occupier === 0).length;
|
|
90
94
|
const p1Count = window.filter((occupier) => occupier === 1).length;
|
|
91
95
|
if (p0Count > 0 && p1Count === 0)
|
|
92
|
-
p0Score += [
|
|
96
|
+
p0Score += defensiveScores[p0Count];
|
|
93
97
|
else if (p1Count > 0 && p0Count === 0)
|
|
94
|
-
p1Score += [
|
|
98
|
+
p1Score += offensiveScores[p1Count];
|
|
95
99
|
}
|
|
96
100
|
}
|
|
97
101
|
// Diagonal ↘
|
|
@@ -106,9 +110,9 @@ export default class Board extends Base {
|
|
|
106
110
|
const p0Count = window.filter((occupier) => occupier === 0).length;
|
|
107
111
|
const p1Count = window.filter((occupier) => occupier === 1).length;
|
|
108
112
|
if (p0Count > 0 && p1Count === 0)
|
|
109
|
-
p0Score += [
|
|
113
|
+
p0Score += defensiveScores[p0Count];
|
|
110
114
|
else if (p1Count > 0 && p0Count === 0)
|
|
111
|
-
p1Score += [
|
|
115
|
+
p1Score += offensiveScores[p1Count];
|
|
112
116
|
}
|
|
113
117
|
}
|
|
114
118
|
// Diagonal ↗
|
|
@@ -123,9 +127,9 @@ export default class Board extends Base {
|
|
|
123
127
|
const p0Count = window.filter((occupier) => occupier === 0).length;
|
|
124
128
|
const p1Count = window.filter((occupier) => occupier === 1).length;
|
|
125
129
|
if (p0Count > 0 && p1Count === 0)
|
|
126
|
-
p0Score += [
|
|
130
|
+
p0Score += defensiveScores[p0Count];
|
|
127
131
|
else if (p1Count > 0 && p0Count === 0)
|
|
128
|
-
p1Score += [
|
|
132
|
+
p1Score += offensiveScores[p1Count];
|
|
129
133
|
}
|
|
130
134
|
}
|
|
131
135
|
return p1Score - p0Score;
|
|
@@ -160,9 +164,9 @@ export default class Board extends Base {
|
|
|
160
164
|
*/
|
|
161
165
|
undoLastMove() {
|
|
162
166
|
// X-coordinate is recoverable from the bit index: bit % width = x.
|
|
163
|
-
const x = this.moves[this.moves.length - 1]
|
|
167
|
+
const { x } = (this.moves[this.moves.length - 1]);
|
|
164
168
|
super.undoLastMove();
|
|
165
169
|
this.heights[x]++;
|
|
166
170
|
}
|
|
167
171
|
}
|
|
168
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.js","sourceRoot":"","sources":["../../../src/games/connect4/board.ts"],"names":[],"mappings":"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","sourcesContent":["import Base from \"../../base/board.js\";\nimport LongInt from \"../../bitBoard/longInt.js\";\nimport LongIntBitBoard from \"../../bitBoard/longIntBitBoard.js\";\nimport type { Position } from \"../../base/board.js\";\n\n/** Represents a Connect 4 board. */\nexport default class Board extends Base<LongInt> {\n    protected winningStates: LongIntBitBoard[] = [];\n\n    /**\n     * Tracks the next available row index per column (row 5 = bottom, row 0 = top).\n     * A value of -1 means the column is full.\n     * Maintained incrementally so makeMove and emptyCells are O(1).\n     */\n    private readonly heights: number[] = [5, 5, 5, 5, 5, 5, 5];\n\n    private readonly FULL_BOARD = new LongInt([0b1111_1111111_1111111_1111111_1111111, 0b1111111_111]);\n\n    private readonly HORIZONTAL = new LongInt([0b0000_0000000_0000000_0000000_0001111, 0b0000000_000]);\n\n    private readonly VERTICAL = new LongInt([0b0000_0000001_0000001_0000001_0000001, 0b0000000_000]);\n\n    private readonly LEADING_DIAGONAL = new LongInt([0b0000_0001000_0000100_0000010_0000001, 0b0000000_000]);\n\n    private readonly NON_LEADING_DIAGONAL = new LongInt([0b0000_0000001_0000010_0000100_0001000, 0b0000000_000]);\n\n    /** Creates an instance of Board. */\n    public constructor() {\n        super(7, 6);\n        for (let i = 0; i < 4; i++) {\n            for (let j = 0; j < 6; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.HORIZONTAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n        for (let i = 0; i < 7; i++) {\n            for (let j = 0; j < 3; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.VERTICAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n        for (let i = 0; i < 4; i++) {\n            for (let j = 0; j < 3; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.LEADING_DIAGONAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n        for (let i = 0; i < 4; i++) {\n            for (let j = 0; j < 3; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.NON_LEADING_DIAGONAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n    }\n\n    /**\n     * Calculates the heuristic score for the current board state.\n     * @returns The heuristic score.\n     */\n    // eslint-disable-next-line max-statements\n    public get heuristic(): number {\n        const { winner } = this;\n\n        if (winner === 0)\n            return 10000 - this.moves.length;\n\n        if (winner === 1)\n            return -(10000 - this.moves.length);\n\n        let p0Score = 0;\n        let p1Score = 0;\n\n        // Horizontal →\n        for (let y = 0; y < this.height; y++) {\n            for (let x = 0; x < this.width - 3; x++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x: x + 1, y } as Position),\n                    this.cellOccupier({ x: x + 2, y } as Position),\n                    this.cellOccupier({ x: x + 3, y } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += [0, 1, 10, 100][p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += [0, 1, 10, 100][p1Count]!;\n            }\n        }\n\n        // Vertical ↓\n        for (let x = 0; x < this.width; x++) {\n            for (let y = 0; y < this.height - 3; y++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x, y: y + 1 } as Position),\n                    this.cellOccupier({ x, y: y + 2 } as Position),\n                    this.cellOccupier({ x, y: y + 3 } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += [0, 1, 10, 100][p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += [0, 1, 10, 100][p1Count]!;\n            }\n        }\n\n        // Diagonal ↘\n        for (let x = 0; x < this.width - 3; x++) {\n            for (let y = 0; y < this.height - 3; y++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x: x + 1, y: y + 1 } as Position),\n                    this.cellOccupier({ x: x + 2, y: y + 2 } as Position),\n                    this.cellOccupier({ x: x + 3, y: y + 3 } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += [0, 1, 10, 100][p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += [0, 1, 10, 100][p1Count]!;\n            }\n        }\n\n        // Diagonal ↗\n        for (let x = 0; x < this.width - 3; x++) {\n            for (let y = 3; y < this.height; y++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x: x + 1, y: y - 1 } as Position),\n                    this.cellOccupier({ x: x + 2, y: y - 2 } as Position),\n                    this.cellOccupier({ x: x + 3, y: y - 3 } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += [0, 1, 10, 100][p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += [0, 1, 10, 100][p1Count]!;\n            }\n        }\n\n        return p1Score - p0Score;\n    }\n\n    /**\n     * Generates a list of available column moves ordered centre-first so that\n     * alpha-beta pruning encounters the strongest moves earliest.\n     * Uses the incremental heights array — O(width), no bitboard reads.\n     * @returns The list of empty cells.\n     */\n    public override get emptyCells(): Position[] {\n        const emptyCells: Position[] = [];\n\n        for (const x of [3, 2, 4, 1, 5, 0, 6]) {\n            if (this.heights[x]! >= 0)\n                emptyCells.push({ x, y: 0 });\n        }\n\n        return emptyCells;\n    }\n\n    /**\n     * Makes a move on the board.\n     * Uses the heights array for O(1) row resolution instead of scanning the column.\n     * @param move - The move to make.\n     * @param playerId - The player making the move.\n     */\n    public override makeMove(move: Position, playerId: number): void {\n        move.y = this.heights[move.x]!;\n        this.heights[move.x]!--;\n        super.makeMove(move, playerId);\n    }\n\n    /**\n     * Reverses the last move and restores the column height.\n     */\n    public override undoLastMove(): void {\n        // X-coordinate is recoverable from the bit index: bit % width = x.\n        const x = this.moves[this.moves.length - 1]! % this.width;\n\n        super.undoLastMove();\n        this.heights[x]!++;\n    }\n}\n"]}
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.js","sourceRoot":"","sources":["../../../src/games/connect4/board.ts"],"names":[],"mappings":"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","sourcesContent":["import Base from \"../../base/board.js\";\nimport LongInt from \"../../bitBoard/longInt.js\";\nimport LongIntBitBoard from \"../../bitBoard/longIntBitBoard.js\";\nimport type { Position } from \"../../base/board.js\";\n\n/** Represents a Connect 4 board. */\nexport default class Board extends Base<LongInt> {\n    protected winningStates: LongIntBitBoard[] = [];\n\n    /**\n     * Tracks the next available row index per column (row 5 = bottom, row 0 = top).\n     * A value of -1 means the column is full.\n     * Maintained incrementally so makeMove and emptyCells are O(1).\n     */\n    private readonly heights: number[] = [5, 5, 5, 5, 5, 5, 5];\n\n    private readonly FULL_BOARD = new LongInt([0b1111_1111111_1111111_1111111_1111111, 0b1111111_111]);\n\n    private readonly HORIZONTAL = new LongInt([0b0000_0000000_0000000_0000000_0001111, 0b0000000_000]);\n\n    private readonly VERTICAL = new LongInt([0b0000_0000001_0000001_0000001_0000001, 0b0000000_000]);\n\n    private readonly LEADING_DIAGONAL = new LongInt([0b0000_0001000_0000100_0000010_0000001, 0b0000000_000]);\n\n    private readonly NON_LEADING_DIAGONAL = new LongInt([0b0000_0000001_0000010_0000100_0001000, 0b0000000_000]);\n\n    /** Creates an instance of Board. */\n    public constructor() {\n        super(7, 6);\n        for (let i = 0; i < 4; i++) {\n            for (let j = 0; j < 6; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.HORIZONTAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n        for (let i = 0; i < 7; i++) {\n            for (let j = 0; j < 3; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.VERTICAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n        for (let i = 0; i < 4; i++) {\n            for (let j = 0; j < 3; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.LEADING_DIAGONAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n        for (let i = 0; i < 4; i++) {\n            for (let j = 0; j < 3; j++) {\n                this.winningStates.push(new LongIntBitBoard(LongInt\n                    .leftShift(this.NON_LEADING_DIAGONAL, i + 7 * j)\n                    .and(this.FULL_BOARD)));\n            }\n        }\n    }\n\n    /**\n     * Calculates the heuristic score for the current board state.\n     * Favors blocking opponent's 3s over making own 3s.\n     * @returns The heuristic score.\n     */\n    // eslint-disable-next-line max-statements\n    public get heuristic(): number {\n        const { winner } = this;\n\n        if (winner === 0)\n            return 10000 - this.moves.length;\n\n        if (winner === 1)\n            return -(10000 - this.moves.length);\n\n        let p0Score = 0;\n        let p1Score = 0;\n\n        // Scoring: defensive moves (blocking) weighted 1.5x higher than offensive moves\n        const offensiveScores = [0, 1, 10, 100];\n        const defensiveScores = [0, 1.5, 15, 150];\n\n        // Horizontal →\n        for (let y = 0; y < this.height; y++) {\n            for (let x = 0; x < this.width - 3; x++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x: x + 1, y } as Position),\n                    this.cellOccupier({ x: x + 2, y } as Position),\n                    this.cellOccupier({ x: x + 3, y } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += defensiveScores[p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += offensiveScores[p1Count]!;\n            }\n        }\n\n        // Vertical ↓\n        for (let x = 0; x < this.width; x++) {\n            for (let y = 0; y < this.height - 3; y++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x, y: y + 1 } as Position),\n                    this.cellOccupier({ x, y: y + 2 } as Position),\n                    this.cellOccupier({ x, y: y + 3 } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += defensiveScores[p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += offensiveScores[p1Count]!;\n            }\n        }\n\n        // Diagonal ↘\n        for (let x = 0; x < this.width - 3; x++) {\n            for (let y = 0; y < this.height - 3; y++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x: x + 1, y: y + 1 } as Position),\n                    this.cellOccupier({ x: x + 2, y: y + 2 } as Position),\n                    this.cellOccupier({ x: x + 3, y: y + 3 } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += defensiveScores[p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += offensiveScores[p1Count]!;\n            }\n        }\n\n        // Diagonal ↗\n        for (let x = 0; x < this.width - 3; x++) {\n            for (let y = 3; y < this.height; y++) {\n                const window = [\n                    this.cellOccupier({ x, y } as Position),\n                    this.cellOccupier({ x: x + 1, y: y - 1 } as Position),\n                    this.cellOccupier({ x: x + 2, y: y - 2 } as Position),\n                    this.cellOccupier({ x: x + 3, y: y - 3 } as Position),\n                ];\n\n                const p0Count = window.filter((occupier) => occupier === 0).length;\n                const p1Count = window.filter((occupier) => occupier === 1).length;\n\n                if (p0Count > 0 && p1Count === 0)\n                    p0Score += defensiveScores[p0Count]!;\n                else if (p1Count > 0 && p0Count === 0)\n                    p1Score += offensiveScores[p1Count]!;\n            }\n        }\n\n        return p1Score - p0Score;\n    }\n\n    /**\n     * Generates a list of available column moves ordered centre-first so that\n     * alpha-beta pruning encounters the strongest moves earliest.\n     * Uses the incremental heights array — O(width), no bitboard reads.\n     * @returns The list of empty cells.\n     */\n    public override get emptyCells(): Position[] {\n        const emptyCells: Position[] = [];\n\n        for (const x of [3, 2, 4, 1, 5, 0, 6]) {\n            if (this.heights[x]! >= 0)\n                emptyCells.push({ x, y: 0 });\n        }\n\n        return emptyCells;\n    }\n\n    /**\n     * Makes a move on the board.\n     * Uses the heights array for O(1) row resolution instead of scanning the column.\n     * @param move - The move to make.\n     * @param playerId - The player making the move.\n     */\n    public override makeMove(move: Position, playerId: number): void {\n        move.y = this.heights[move.x]!;\n        this.heights[move.x]!--;\n        super.makeMove(move, playerId);\n    }\n\n    /**\n     * Reverses the last move and restores the column height.\n     */\n    public override undoLastMove(): void {\n        // X-coordinate is recoverable from the bit index: bit % width = x.\n        const { x } = (this.moves[this.moves.length - 1]!);\n\n        super.undoLastMove();\n        this.heights[x]!++;\n    }\n}\n"]}
|
|
@@ -62,11 +62,11 @@ let Connect4 = class Connect4 extends Base {
|
|
|
62
62
|
findOptimalMove({ maxDepth } = { maxDepth: Infinity }) {
|
|
63
63
|
return this.board.isEmpty
|
|
64
64
|
? { x: 3, y: 5 }
|
|
65
|
-
: this.
|
|
65
|
+
: this.search(maxDepth).move;
|
|
66
66
|
}
|
|
67
67
|
};
|
|
68
68
|
Connect4 = __decorate([
|
|
69
69
|
Game
|
|
70
70
|
], Connect4);
|
|
71
71
|
export default Connect4;
|
|
72
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
72
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udHJvbGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9nYW1lcy9jb25uZWN0NC9jb250cm9sbGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBLE9BQU8sSUFBSSxFQUFFLEVBQUUsSUFBSSxFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFFdEQsT0FBTyxLQUFLLE1BQU0sWUFBWSxDQUFDO0FBRy9COzs7R0FHRztBQUNILFNBQVMsYUFBYSxDQUFDLFVBQW9CO0lBQ3ZDLCtCQUErQjtJQUMvQixPQUFPLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDaEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxJQUFJLEVBQUUsSUFBSSxFQUFFLEtBQUssRUFBRSxDQUFDLElBQUksRUFBRSxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDeEUsTUFBTSxFQUFFLE1BQU0sRUFBRSxHQUFHLFVBQVUsQ0FBQyxLQUFLLENBQUM7SUFFcEMsSUFBSSxNQUFNLEtBQUssS0FBSztRQUNoQixPQUFPLENBQUMsR0FBRyxDQUFDLE1BQU0sS0FBSyxJQUFJLENBQUMsQ0FBQyxDQUFDLGFBQWEsQ0FBQyxDQUFDLENBQUMsVUFBVSxNQUFNLEdBQUcsQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUNoRiw4QkFBOEI7QUFDbEMsQ0FBQztBQUVELCtDQUErQztBQUVoQyxJQUFNLFFBQVEsR0FBZCxNQUFNLFFBQVMsU0FBUSxJQUFXO0lBQzdDOzs7OztPQUtHO0lBQ0gsWUFDSSxhQUF5QixFQUN6QixhQUF5QixFQUN6QixPQUF1QztRQUV2QyxLQUFLLENBQ0QsQ0FBQyxhQUFhLEVBQUUsYUFBYSxDQUFDLEVBQzlCLElBQUksS0FBSyxFQUFFLEVBQ1gsT0FBTyxFQUFFLFFBQVEsSUFBSSxhQUFhLEVBQ2xDLE9BQU8sRUFBRSxLQUFLLEVBQ2QsT0FBTyxFQUFFLGNBQWMsQ0FDMUIsQ0FBQztJQUNOLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLGdCQUFnQixDQUFDLFVBQXFDO1FBQ3pELE1BQU0sRUFBRSxVQUFVLEVBQUUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDO1FBQ2xDLE1BQU0sVUFBVSxHQUFHLFVBQVUsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxVQUFVLENBQUMsTUFBTSxDQUFDLENBQUUsQ0FBQztRQUU5RSxRQUFRLFVBQVUsRUFBRSxDQUFDO1lBQ2pCLEtBQUssZUFBZTtnQkFDaEIsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDLEVBQUUsUUFBUSxFQUFFLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDbEQsS0FBSyxTQUFTO2dCQUNWLE9BQU8sSUFBSSxDQUFDLGVBQWUsQ0FBQyxFQUFFLFFBQVEsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQ2pELEtBQUssV0FBVztnQkFDWixPQUFPLElBQUksQ0FBQyxlQUFlLENBQUMsRUFBRSxRQUFRLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQztZQUNqRCxLQUFLLFNBQVM7Z0JBQ1YsT0FBTyxVQUFVLENBQUM7WUFDdEI7Z0JBQ0ksTUFBTSxJQUFJLEtBQUssQ0FBQyxxQkFBcUIsQ0FBQyxDQUFDO1FBQy9DLENBQUM7SUFDTCxDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0ksZUFBZSxDQUFDLEVBQUUsUUFBUSxLQUE0QixFQUFFLFFBQVEsRUFBRSxRQUFRLEVBQUU7UUFDL0UsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU87WUFDckIsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFO1lBQ2hCLENBQUMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDLElBQUksQ0FBQztJQUNyQyxDQUFDO0NBQ0osQ0FBQTtBQXpEb0IsUUFBUTtJQUQ1QixJQUFJO0dBQ2dCLFFBQVEsQ0F5RDVCO2VBekRvQixRQUFRIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEJhc2UsIHsgR2FtZSB9IGZyb20gXCIuLi8uLi9iYXNlL2NvbnRyb2xsZXIuanNcIjtcbmltcG9ydCB0eXBlIHsgR2FtZUNvbnN0cnVjdG9yT3B0aW9ucywgUGxheWVyVHlwZSB9IGZyb20gXCIuLi8uLi9iYXNlL2NvbnRyb2xsZXIuanNcIjtcbmltcG9ydCBCb2FyZCBmcm9tIFwiLi9ib2FyZC5qc1wiO1xuaW1wb3J0IHR5cGUgeyBQb3NpdGlvbiB9IGZyb20gXCIuLi8uLi9iYXNlL2JvYXJkLmpzXCI7XG5cbi8qKlxuICogVGhlIGRlZmF1bHQgcmVuZGVyZXIgZm9yIGNvbm5lY3QgNC5cbiAqIEBwYXJhbSBjb250cm9sbGVyIC0gVGhlIGNvbnRyb2xsZXIgdG8gcmVuZGVyLlxuICovXG5mdW5jdGlvbiBkZWZhdWx0UmVuZGVyKGNvbnRyb2xsZXI6IENvbm5lY3Q0KTogdm9pZCB7XG4gICAgLyogZXNsaW50LWRpc2FibGUgbm8tY29uc29sZSAqL1xuICAgIGNvbnNvbGUuY2xlYXIoKTtcbiAgICBjb25zb2xlLmxvZyhjb250cm9sbGVyLmJvYXJkLnRvU3RyaW5nKHRydWUsIHRydWUsIGZhbHNlLCBbXCLirKQgXCIsIFwi4qykIFwiXSkpO1xuICAgIGNvbnN0IHsgd2lubmVyIH0gPSBjb250cm9sbGVyLmJvYXJkO1xuXG4gICAgaWYgKHdpbm5lciAhPT0gZmFsc2UpXG4gICAgICAgIGNvbnNvbGUubG9nKHdpbm5lciA9PT0gbnVsbCA/IFwiSXQncyBhIHRpZSFcIiA6IGBQbGF5ZXIgJHt3aW5uZXIgKyAxfSB3aW5zIWApO1xuICAgIC8qIGVzbGludC1lbmFibGUgbm8tY29uc29sZSAqL1xufVxuXG4vKiogUmVwcmVzZW50cyB0aGUgY29udHJvbGxlciBmb3IgY29ubmVjdCA0LiAqL1xuQEdhbWVcbmV4cG9ydCBkZWZhdWx0IGNsYXNzIENvbm5lY3Q0IGV4dGVuZHMgQmFzZTxCb2FyZD4ge1xuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYW4gaW5zdGFuY2Ugb2YgQ29ubmVjdDQuXG4gICAgICogQHBhcmFtIHBsYXllck9uZVR5cGUgLSBUaGUgdHlwZSBvZiBwbGF5ZXIgb25lIChodW1hbiBvciBDUFUpLlxuICAgICAqIEBwYXJhbSBwbGF5ZXJUd29UeXBlIC0gVGhlIHR5cGUgb2YgcGxheWVyIHR3byAoaHVtYW4gb3IgQ1BVKS5cbiAgICAgKiBAcGFyYW0gb3B0aW9ucyAtIFRoZSBvcHRpb25zIGZvciB0aGUgZ2FtZS5cbiAgICAgKi9cbiAgICBwdWJsaWMgY29uc3RydWN0b3IoXG4gICAgICAgIHBsYXllck9uZVR5cGU6IFBsYXllclR5cGUsXG4gICAgICAgIHBsYXllclR3b1R5cGU6IFBsYXllclR5cGUsXG4gICAgICAgIG9wdGlvbnM/OiBHYW1lQ29uc3RydWN0b3JPcHRpb25zPEJvYXJkPixcbiAgICApIHtcbiAgICAgICAgc3VwZXIoXG4gICAgICAgICAgICBbcGxheWVyT25lVHlwZSwgcGxheWVyVHdvVHlwZV0sXG4gICAgICAgICAgICBuZXcgQm9hcmQoKSxcbiAgICAgICAgICAgIG9wdGlvbnM/LnJlbmRlcmVyID8/IGRlZmF1bHRSZW5kZXIsXG4gICAgICAgICAgICBvcHRpb25zPy5vbkVuZCxcbiAgICAgICAgICAgIG9wdGlvbnM/Lm9uSW52YWxpZElucHV0LFxuICAgICAgICApO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENhbGN1bGF0ZXMgdGhlIENQVSdzIG1vdmUuXG4gICAgICogQHBhcmFtIGRpZmZpY3VsdHkgLSBUaGUgZGlmZmljdWx0eSBvZiB0aGUgQ1BVLlxuICAgICAqIEByZXR1cm5zIFRoZSBDUFUncyBtb3ZlLlxuICAgICAqIEB0aHJvd3Mge0Vycm9yfSBBbiBlcnJvciBpZiB0aGUgZGlmZmljdWx0eSBpcyBpbnZhbGlkLlxuICAgICAqL1xuICAgIHB1YmxpYyBkZXRlcm1pbmVDUFVNb3ZlKGRpZmZpY3VsdHk6IE9taXQ8UGxheWVyVHlwZSwgXCJodW1hblwiPik6IFBvc2l0aW9uIHtcbiAgICAgICAgY29uc3QgeyBlbXB0eUNlbGxzIH0gPSB0aGlzLmJvYXJkO1xuICAgICAgICBjb25zdCByYW5kb21Nb3ZlID0gZW1wdHlDZWxsc1tNYXRoLmZsb29yKE1hdGgucmFuZG9tKCkgKiBlbXB0eUNlbGxzLmxlbmd0aCldITtcblxuICAgICAgICBzd2l0Y2ggKGRpZmZpY3VsdHkpIHtcbiAgICAgICAgICAgIGNhc2UgXCJpbXBvc3NpYmxlQ1BVXCI6XG4gICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMuZmluZE9wdGltYWxNb3ZlKHsgbWF4RGVwdGg6IDEwIH0pO1xuICAgICAgICAgICAgY2FzZSBcImhhcmRDUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5maW5kT3B0aW1hbE1vdmUoeyBtYXhEZXB0aDogNyB9KTtcbiAgICAgICAgICAgIGNhc2UgXCJtZWRpdW1DUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5maW5kT3B0aW1hbE1vdmUoeyBtYXhEZXB0aDogNSB9KTtcbiAgICAgICAgICAgIGNhc2UgXCJlYXN5Q1BVXCI6XG4gICAgICAgICAgICAgICAgcmV0dXJuIHJhbmRvbU1vdmU7XG4gICAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcIkludmFsaWQgZGlmZmljdWx0eS5cIik7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBGaW5kcyB0aGUgb3B0aW1hbCBtb3ZlIGZvciB0aGUgY3VycmVudCBib2FyZCBzdGF0ZS5cbiAgICAgKiBAcGFyYW0gb3B0aW9ucyAtIFRoZSBvcHRpb25zIGZvciB0aGUgYWxnb3JpdGhtLlxuICAgICAqIEBwYXJhbSBvcHRpb25zLm1heERlcHRoIC0gVGhlIG1heGltdW0gZGVwdGggdG8gc2VhcmNoLlxuICAgICAqIEByZXR1cm5zIFRoZSBvcHRpbWFsIG1vdmUgZm9yIHRoZSBjdXJyZW50IGJvYXJkIHN0YXRlLlxuICAgICAqIEB0aHJvd3Mge0Vycm9yfSBBbiBlcnJvciBpZiB0aGUgYWxnb3JpdGhtIGlzIGludmFsaWQuXG4gICAgICovXG4gICAgcHVibGljIGZpbmRPcHRpbWFsTW92ZSh7IG1heERlcHRoIH06IHsgbWF4RGVwdGg6IG51bWJlcjsgfSA9IHsgbWF4RGVwdGg6IEluZmluaXR5IH0pOiBQb3NpdGlvbiB7XG4gICAgICAgIHJldHVybiB0aGlzLmJvYXJkLmlzRW1wdHlcbiAgICAgICAgICAgID8geyB4OiAzLCB5OiA1IH1cbiAgICAgICAgICAgIDogdGhpcy5zZWFyY2gobWF4RGVwdGgpLm1vdmU7XG4gICAgfVxufVxuIl19
|
|
@@ -63,11 +63,11 @@ let TicTacToe = class TicTacToe extends Base {
|
|
|
63
63
|
findOptimalMove({ randomMove } = { randomMove: { x: 2, y: 2 } }) {
|
|
64
64
|
return this.board.isEmpty
|
|
65
65
|
? randomMove
|
|
66
|
-
: this.
|
|
66
|
+
: this.search(9).move;
|
|
67
67
|
}
|
|
68
68
|
};
|
|
69
69
|
TicTacToe = __decorate([
|
|
70
70
|
Game
|
|
71
71
|
], TicTacToe);
|
|
72
72
|
export default TicTacToe;
|
|
73
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
73
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udHJvbGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9nYW1lcy90aWN0YWN0b2UvY29udHJvbGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSxPQUFPLElBQUksRUFBRSxFQUFFLElBQUksRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBRXRELE9BQU8sS0FBSyxNQUFNLFlBQVksQ0FBQztBQUcvQjs7O0dBR0c7QUFDSCxTQUFTLGFBQWEsQ0FBQyxVQUFxQjtJQUN4QywrQkFBK0I7SUFDL0IsT0FBTyxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ2hCLE9BQU8sQ0FBQyxHQUFHLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztJQUM5QyxNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsVUFBVSxDQUFDLEtBQUssQ0FBQztJQUVwQyxJQUFJLE1BQU0sS0FBSyxLQUFLO1FBQ2hCLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxLQUFLLElBQUksQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxVQUFVLE1BQU0sR0FBRyxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ2hGLDhCQUE4QjtBQUNsQyxDQUFDO0FBRUQsMkJBQTJCO0FBRVosSUFBTSxTQUFTLEdBQWYsTUFBTSxTQUFVLFNBQVEsSUFBVztJQUM5Qzs7Ozs7T0FLRztJQUNILFlBQ0ksYUFBeUIsRUFDekIsYUFBeUIsRUFDekIsT0FBdUM7UUFFdkMsS0FBSyxDQUNELENBQUMsYUFBYSxFQUFFLGFBQWEsQ0FBQyxFQUM5QixJQUFJLEtBQUssRUFBRSxFQUNYLE9BQU8sRUFBRSxRQUFRLElBQUksYUFBYSxFQUNsQyxPQUFPLEVBQUUsS0FBSyxFQUNkLE9BQU8sRUFBRSxjQUFjLENBQzFCLENBQUM7SUFDTixDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSSxnQkFBZ0IsQ0FBQyxVQUFxQztRQUN6RCxNQUFNLEVBQUUsVUFBVSxFQUFFLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQztRQUNsQyxNQUFNLFVBQVUsR0FBRyxVQUFVLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEdBQUcsVUFBVSxDQUFDLE1BQU0sQ0FBQyxDQUFFLENBQUM7UUFDOUUsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLGVBQWUsQ0FBQyxFQUFFLFVBQVUsRUFBRSxDQUFDLENBQUM7UUFFekQsUUFBUSxVQUFVLEVBQUUsQ0FBQztZQUNqQixLQUFLLGVBQWU7Z0JBQ2hCLE9BQU8sV0FBVyxDQUFDO1lBQ3ZCLEtBQUssU0FBUztnQkFDVixPQUFPLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDO1lBQzFELEtBQUssV0FBVztnQkFDWixPQUFPLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDO1lBQzFELEtBQUssU0FBUztnQkFDVixPQUFPLFVBQVUsQ0FBQztZQUN0QjtnQkFDSSxNQUFNLElBQUksS0FBSyxDQUFDLHFCQUFxQixDQUFDLENBQUM7UUFDL0MsQ0FBQztJQUNMLENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSSxlQUFlLENBQUMsRUFBRSxVQUFVLEtBQWdDLEVBQUUsVUFBVSxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUU7UUFDN0YsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU87WUFDckIsQ0FBQyxDQUFDLFVBQVU7WUFDWixDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUM7SUFDOUIsQ0FBQztDQUNKLENBQUE7QUExRG9CLFNBQVM7SUFEN0IsSUFBSTtHQUNnQixTQUFTLENBMEQ3QjtlQTFEb0IsU0FBUyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBCYXNlLCB7IEdhbWUgfSBmcm9tIFwiLi4vLi4vYmFzZS9jb250cm9sbGVyLmpzXCI7XG5pbXBvcnQgdHlwZSB7IEdhbWVDb25zdHJ1Y3Rvck9wdGlvbnMsIFBsYXllclR5cGUgfSBmcm9tIFwiLi4vLi4vYmFzZS9jb250cm9sbGVyLmpzXCI7XG5pbXBvcnQgQm9hcmQgZnJvbSBcIi4vYm9hcmQuanNcIjtcbmltcG9ydCB0eXBlIHsgUG9zaXRpb24gfSBmcm9tIFwiLi4vLi4vYmFzZS9ib2FyZC5qc1wiO1xuXG4vKipcbiAqIFRoZSBkZWZhdWx0IHJlbmRlcmluZyBmdW5jdGlvbiBmb3IgVGljVGFjVG9lLlxuICogQHBhcmFtIGNvbnRyb2xsZXIgLSBUaGUgY29udHJvbGxlciB0byByZW5kZXIuXG4gKi9cbmZ1bmN0aW9uIGRlZmF1bHRSZW5kZXIoY29udHJvbGxlcjogVGljVGFjVG9lKTogdm9pZCB7XG4gICAgLyogZXNsaW50LWRpc2FibGUgbm8tY29uc29sZSAqL1xuICAgIGNvbnNvbGUuY2xlYXIoKTtcbiAgICBjb25zb2xlLmxvZyhjb250cm9sbGVyLmJvYXJkLnRvU3RyaW5nKGZhbHNlKSk7XG4gICAgY29uc3QgeyB3aW5uZXIgfSA9IGNvbnRyb2xsZXIuYm9hcmQ7XG5cbiAgICBpZiAod2lubmVyICE9PSBmYWxzZSlcbiAgICAgICAgY29uc29sZS5sb2cod2lubmVyID09PSBudWxsID8gXCJJdCdzIGEgdGllIVwiIDogYFBsYXllciAke3dpbm5lciArIDF9IHdpbnMhYCk7XG4gICAgLyogZXNsaW50LWVuYWJsZSBuby1jb25zb2xlICovXG59XG5cbi8qKiBBIGdhbWUgb2YgVGljVGFjVG9lLiAqL1xuQEdhbWVcbmV4cG9ydCBkZWZhdWx0IGNsYXNzIFRpY1RhY1RvZSBleHRlbmRzIEJhc2U8Qm9hcmQ+IHtcbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgbmV3IFRpY1RhY1RvZSBnYW1lLlxuICAgICAqIEBwYXJhbSBwbGF5ZXJPbmVUeXBlIC0gVGhlIHR5cGUgb2YgcGxheWVyIG9uZSAoaHVtYW4gb3IgQ1BVKS5cbiAgICAgKiBAcGFyYW0gcGxheWVyVHdvVHlwZSAtIFRoZSB0eXBlIG9mIHBsYXllciB0d28gKGh1bWFuIG9yIENQVSkuXG4gICAgICogQHBhcmFtIG9wdGlvbnMgLSBUaGUgb3B0aW9ucyBmb3IgdGhlIGdhbWUuXG4gICAgICovXG4gICAgcHVibGljIGNvbnN0cnVjdG9yKFxuICAgICAgICBwbGF5ZXJPbmVUeXBlOiBQbGF5ZXJUeXBlLFxuICAgICAgICBwbGF5ZXJUd29UeXBlOiBQbGF5ZXJUeXBlLFxuICAgICAgICBvcHRpb25zPzogR2FtZUNvbnN0cnVjdG9yT3B0aW9uczxCb2FyZD4sXG4gICAgKSB7XG4gICAgICAgIHN1cGVyKFxuICAgICAgICAgICAgW3BsYXllck9uZVR5cGUsIHBsYXllclR3b1R5cGVdLFxuICAgICAgICAgICAgbmV3IEJvYXJkKCksXG4gICAgICAgICAgICBvcHRpb25zPy5yZW5kZXJlciA/PyBkZWZhdWx0UmVuZGVyLFxuICAgICAgICAgICAgb3B0aW9ucz8ub25FbmQsXG4gICAgICAgICAgICBvcHRpb25zPy5vbkludmFsaWRJbnB1dCxcbiAgICAgICAgKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDYWxjdWxhdGVzIHRoZSBDUFUncyBtb3ZlLlxuICAgICAqIEBwYXJhbSBkaWZmaWN1bHR5IC0gVGhlIGRpZmZpY3VsdHkgb2YgdGhlIENQVS5cbiAgICAgKiBAcmV0dXJucyBUaGUgQ1BVJ3MgbW92ZS5cbiAgICAgKiBAdGhyb3dzIHtFcnJvcn0gQW4gZXJyb3IgaWYgdGhlIGRpZmZpY3VsdHkgaXMgaW52YWxpZC5cbiAgICAgKi9cbiAgICBwdWJsaWMgZGV0ZXJtaW5lQ1BVTW92ZShkaWZmaWN1bHR5OiBPbWl0PFBsYXllclR5cGUsIFwiaHVtYW5cIj4pOiBQb3NpdGlvbiB7XG4gICAgICAgIGNvbnN0IHsgZW1wdHlDZWxscyB9ID0gdGhpcy5ib2FyZDtcbiAgICAgICAgY29uc3QgcmFuZG9tTW92ZSA9IGVtcHR5Q2VsbHNbTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogZW1wdHlDZWxscy5sZW5ndGgpXSE7XG4gICAgICAgIGNvbnN0IG9wdGltYWxNb3ZlID0gdGhpcy5maW5kT3B0aW1hbE1vdmUoeyByYW5kb21Nb3ZlIH0pO1xuXG4gICAgICAgIHN3aXRjaCAoZGlmZmljdWx0eSkge1xuICAgICAgICAgICAgY2FzZSBcImltcG9zc2libGVDUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gb3B0aW1hbE1vdmU7XG4gICAgICAgICAgICBjYXNlIFwiaGFyZENQVVwiOlxuICAgICAgICAgICAgICAgIHJldHVybiBNYXRoLnJhbmRvbSgpIDwgMC44ID8gb3B0aW1hbE1vdmUgOiByYW5kb21Nb3ZlO1xuICAgICAgICAgICAgY2FzZSBcIm1lZGl1bUNQVVwiOlxuICAgICAgICAgICAgICAgIHJldHVybiBNYXRoLnJhbmRvbSgpIDwgMC41ID8gb3B0aW1hbE1vdmUgOiByYW5kb21Nb3ZlO1xuICAgICAgICAgICAgY2FzZSBcImVhc3lDUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gcmFuZG9tTW92ZTtcbiAgICAgICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFwiSW52YWxpZCBkaWZmaWN1bHR5LlwiKTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIEZpbmRzIHRoZSBvcHRpbWFsIG1vdmUgZm9yIHRoZSBjdXJyZW50IHBsYXllci5cbiAgICAgKiBAcGFyYW0gYWxnb3JpdGhtIC0gVGhlIGFsZ29yaXRobSB0byB1c2UuXG4gICAgICogQHBhcmFtIGFsZ29yaXRobS5yYW5kb21Nb3ZlIC0gVGhlIG1vdmUgdG8gcmV0dXJuIGlmIHRoZSBib2FyZCBpcyBlbXB0eS5cbiAgICAgKiBAcmV0dXJucyBUaGUgb3B0aW1hbCBtb3ZlIGZvciB0aGUgY3VycmVudCBwbGF5ZXIuXG4gICAgICogQHRocm93cyB7RXJyb3J9IEFuIGVycm9yIGlmIHRoZSBhbGdvcml0aG0gaXMgaW52YWxpZC5cbiAgICAgKi9cbiAgICBwdWJsaWMgZmluZE9wdGltYWxNb3ZlKHsgcmFuZG9tTW92ZSB9OiB7IHJhbmRvbU1vdmU6IFBvc2l0aW9uOyB9ID0geyByYW5kb21Nb3ZlOiB7IHg6IDIsIHk6IDIgfSB9KTogUG9zaXRpb24ge1xuICAgICAgICByZXR1cm4gdGhpcy5ib2FyZC5pc0VtcHR5XG4gICAgICAgICAgICA/IHJhbmRvbU1vdmVcbiAgICAgICAgICAgIDogdGhpcy5zZWFyY2goOSkubW92ZTtcbiAgICB9XG59XG4iXX0=
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@oathompsonjones/mini-games",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.17",
|
|
4
4
|
"description": "A selection of MiniGames. You will soon be able to play these games on [my website](https://oathompsonjones.co.uk/arcade).",
|
|
5
5
|
"repository": {
|
|
6
6
|
"url": "https://github.com/oathompsonjones/MiniGames.git"
|