@oathompsonjones/mini-games 1.0.6 → 1.0.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/TODO.md +6 -0
- package/build/base/board.d.ts +53 -45
- package/build/base/board.js +84 -72
- package/build/base/controller.d.ts +48 -37
- package/build/base/controller.js +32 -26
- package/build/bitBoard/bitBoard.d.ts +37 -34
- package/build/bitBoard/bitBoard.js +15 -12
- package/build/bitBoard/intBitBoard.d.ts +22 -22
- package/build/bitBoard/intBitBoard.js +16 -16
- package/build/bitBoard/longInt.d.ts +47 -40
- package/build/bitBoard/longInt.js +40 -37
- package/build/bitBoard/longIntBitBoard.d.ts +26 -26
- package/build/bitBoard/longIntBitBoard.js +16 -16
- package/build/console.d.ts +2 -3
- package/build/console.js +3 -4
- package/build/games/connect4/board.d.ts +12 -5
- package/build/games/connect4/board.js +12 -6
- package/build/games/connect4/controller.d.ts +18 -11
- package/build/games/connect4/controller.js +20 -12
- package/build/games/tictactoe/board.d.ts +5 -2
- package/build/games/tictactoe/board.js +5 -2
- package/build/games/tictactoe/controller.d.ts +17 -12
- package/build/games/tictactoe/controller.js +19 -13
- package/eslint.config.js +1 -1
- package/package.json +6 -7
- package/patches/eventemitter3@5.0.1.patch +0 -68
|
@@ -1,34 +1,39 @@
|
|
|
1
1
|
import type Board from "./board.js";
|
|
2
2
|
import { EventEmitter } from "eventemitter3";
|
|
3
|
+
import type LongInt from "../bitBoard/longInt.js";
|
|
3
4
|
import type { Position } from "./board.js";
|
|
4
5
|
export type PlayerType = "easyCPU" | "hardCPU" | "human" | "impossibleCPU" | "mediumCPU";
|
|
5
6
|
export type Algorithm = "alphabeta" | "minimax";
|
|
6
|
-
export type GameConstructorOptions = {
|
|
7
|
-
id
|
|
7
|
+
export type GameConstructorOptions<T> = {
|
|
8
|
+
id: T;
|
|
8
9
|
onEnd?: (winner: number | null) => Promise<void> | void;
|
|
9
10
|
onInvalidInput?: (position: Position) => Promise<void> | void;
|
|
10
|
-
renderer?: (controller: Controller) => Promise<void> | void;
|
|
11
|
+
renderer?: (controller: Controller<T>) => Promise<void> | void;
|
|
11
12
|
};
|
|
12
|
-
export type GameConstructor = {
|
|
13
|
-
new (playerOneType: PlayerType, playerTwoType: PlayerType, options
|
|
13
|
+
export type GameConstructor<T> = {
|
|
14
|
+
new (playerOneType: PlayerType, playerTwoType: PlayerType, options: GameConstructorOptions<T>): Controller<T>;
|
|
14
15
|
};
|
|
15
16
|
/**
|
|
16
17
|
* Decorator to check that the constructor type for the given class is correct.
|
|
17
|
-
* @
|
|
18
|
+
* @template T - The type of the game ID.
|
|
19
|
+
* @param constructor - The class to check.
|
|
18
20
|
*/
|
|
19
|
-
export declare function Game(constructor: GameConstructor): void;
|
|
20
|
-
/**
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
21
|
+
export declare function Game<T>(constructor: GameConstructor<T>): void;
|
|
22
|
+
/**
|
|
23
|
+
* Represents a game controller.
|
|
24
|
+
* @template T - The type of the game ID.
|
|
25
|
+
*/
|
|
26
|
+
export default abstract class Controller<T> extends EventEmitter<{
|
|
27
|
+
end: GameConstructorOptions<T>["onEnd"];
|
|
28
|
+
input: GameConstructorOptions<T>["onInvalidInput"];
|
|
29
|
+
invalidInput: GameConstructorOptions<T>["onInvalidInput"];
|
|
25
30
|
}> {
|
|
26
31
|
/** Contains the ID of the game. */
|
|
27
|
-
readonly gameID:
|
|
32
|
+
readonly gameID: T;
|
|
28
33
|
/** Contains the board. */
|
|
29
|
-
readonly board: Board
|
|
34
|
+
readonly board: Board<LongInt | number>;
|
|
30
35
|
/** Contains the view object. */
|
|
31
|
-
readonly render:
|
|
36
|
+
readonly render: () => Promise<void> | void;
|
|
32
37
|
/** Contains the player objects. */
|
|
33
38
|
protected readonly players: Array<{
|
|
34
39
|
id: number;
|
|
@@ -38,28 +43,32 @@ export default abstract class Controller extends EventEmitter<{
|
|
|
38
43
|
private currentPlayerId;
|
|
39
44
|
/**
|
|
40
45
|
* Creates an instance of Controller.
|
|
41
|
-
* @param playerTypes The types of player.
|
|
42
|
-
* @param
|
|
43
|
-
* @param
|
|
44
|
-
* @param
|
|
46
|
+
* @param playerTypes - The types of player for the game.
|
|
47
|
+
* @param board - The board object.
|
|
48
|
+
* @param render - The render function.
|
|
49
|
+
* @param gameID - The ID of the game instance.
|
|
50
|
+
* @param onEnd - The function to call when the game ends.
|
|
51
|
+
* @param onInvalidInput - The function to call when the input is invalid.
|
|
52
|
+
*/
|
|
53
|
+
protected constructor(playerTypes: PlayerType[], board: Board<LongInt | number>, render: Required<GameConstructorOptions<T>>["renderer"], gameID: T, onEnd?: GameConstructorOptions<T>["onEnd"], onInvalidInput?: GameConstructorOptions<T>["onInvalidInput"]);
|
|
54
|
+
/**
|
|
55
|
+
* Gets the current player object.
|
|
56
|
+
* @returns The current player object.
|
|
45
57
|
*/
|
|
46
|
-
protected constructor(playerTypes: PlayerType[], board: Board, render: Controller["render"], gameID: unknown, onEnd?: GameConstructorOptions["onEnd"], onInvalidInput?: GameConstructorOptions["onInvalidInput"]);
|
|
47
|
-
/** Gets the current player object. */
|
|
48
58
|
get currentPlayer(): {
|
|
49
59
|
id: number;
|
|
50
60
|
playerType: PlayerType;
|
|
51
61
|
};
|
|
52
62
|
/**
|
|
53
63
|
* Controls the main game flow.
|
|
54
|
-
* @param algorithm The algorithm to use.
|
|
64
|
+
* @param algorithm - The algorithm to use.
|
|
55
65
|
* @returns The winner or null in the event of a tie.
|
|
56
66
|
*/
|
|
57
67
|
play(algorithm?: Algorithm): Promise<void>;
|
|
58
68
|
/**
|
|
59
|
-
* The bog standard minimax algorithm. Left in for reference.
|
|
60
|
-
* @
|
|
61
|
-
* @param
|
|
62
|
-
* @param maximisingPlayer Whether or not the current player is the maximising player.
|
|
69
|
+
* The bog standard minimax algorithm. Left in for reference (https://en.wikipedia.org/wiki/Minimax).
|
|
70
|
+
* @param depth - The depth of the algorithm.
|
|
71
|
+
* @param maximisingPlayer - Whether or not the current player is the maximising player.
|
|
63
72
|
* @returns The optimal move.
|
|
64
73
|
*/
|
|
65
74
|
protected minimax(depth?: number, maximisingPlayer?: boolean): {
|
|
@@ -67,12 +76,11 @@ export default abstract class Controller extends EventEmitter<{
|
|
|
67
76
|
score: number;
|
|
68
77
|
};
|
|
69
78
|
/**
|
|
70
|
-
* The minimax algorithm with alpha-beta pruning.
|
|
71
|
-
* @
|
|
72
|
-
* @param
|
|
73
|
-
* @param
|
|
74
|
-
* @param
|
|
75
|
-
* @param maximisingPlayer Whether or not the current player is the maximising player.
|
|
79
|
+
* The minimax algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).
|
|
80
|
+
* @param depth - The depth of the algorithm.
|
|
81
|
+
* @param alpha - The bounds for the alpha-beta variation of the algorithm.
|
|
82
|
+
* @param beta - The bounds for the alpha-beta variation of the algorithm.
|
|
83
|
+
* @param maximisingPlayer - Whether or not the current player is the maximising player.
|
|
76
84
|
* @returns The optimal move.
|
|
77
85
|
*/
|
|
78
86
|
protected alphabeta(depth?: number, alpha?: number, beta?: number, maximisingPlayer?: boolean): {
|
|
@@ -83,20 +91,23 @@ export default abstract class Controller extends EventEmitter<{
|
|
|
83
91
|
private nextTurn;
|
|
84
92
|
/**
|
|
85
93
|
* Makes a move.
|
|
86
|
-
* @param input Either the algorithm to use to calculate the move or the move itself.
|
|
94
|
+
* @param input - Either the algorithm to use to calculate the move or the move itself.
|
|
87
95
|
*/
|
|
88
96
|
private makeMove;
|
|
89
97
|
/**
|
|
90
98
|
* Determines the CPU move.
|
|
91
|
-
* @param difficulty The difficulty of the AI.
|
|
92
|
-
* @param algorithm The algorithm to use.
|
|
99
|
+
* @param difficulty - The difficulty of the AI.
|
|
100
|
+
* @param algorithm - The algorithm to use.
|
|
93
101
|
* @returns The move.
|
|
94
102
|
*/
|
|
95
103
|
abstract determineCPUMove(difficulty: Omit<PlayerType, "human">, algorithm?: Algorithm): Position;
|
|
96
104
|
/**
|
|
97
105
|
* Finds the optimal move.
|
|
98
|
-
* @param options The options.
|
|
99
|
-
* @
|
|
106
|
+
* @param options - The options to use.
|
|
107
|
+
* @param options.algorithm - The algorithm to use.
|
|
108
|
+
* @param options.maxDepth - The maximum depth to search.
|
|
109
|
+
* @param options.randomMove - A random move to make.
|
|
110
|
+
* @returns The optimal move to make.
|
|
100
111
|
*/
|
|
101
112
|
abstract findOptimalMove(options?: {
|
|
102
113
|
algorithm?: Algorithm;
|
package/build/base/controller.js
CHANGED
|
@@ -1,12 +1,16 @@
|
|
|
1
1
|
import { EventEmitter } from "eventemitter3";
|
|
2
2
|
/**
|
|
3
3
|
* Decorator to check that the constructor type for the given class is correct.
|
|
4
|
-
* @
|
|
4
|
+
* @template T - The type of the game ID.
|
|
5
|
+
* @param constructor - The class to check.
|
|
5
6
|
*/
|
|
6
7
|
export function Game(constructor) {
|
|
7
8
|
void constructor;
|
|
8
9
|
}
|
|
9
|
-
/**
|
|
10
|
+
/**
|
|
11
|
+
* Represents a game controller.
|
|
12
|
+
* @template T - The type of the game ID.
|
|
13
|
+
*/
|
|
10
14
|
export default class Controller extends EventEmitter {
|
|
11
15
|
/** Contains the ID of the game. */
|
|
12
16
|
gameID;
|
|
@@ -20,35 +24,39 @@ export default class Controller extends EventEmitter {
|
|
|
20
24
|
currentPlayerId = 0;
|
|
21
25
|
/**
|
|
22
26
|
* Creates an instance of Controller.
|
|
23
|
-
* @param playerTypes The types of player.
|
|
24
|
-
* @param
|
|
25
|
-
* @param
|
|
26
|
-
* @param
|
|
27
|
+
* @param playerTypes - The types of player for the game.
|
|
28
|
+
* @param board - The board object.
|
|
29
|
+
* @param render - The render function.
|
|
30
|
+
* @param gameID - The ID of the game instance.
|
|
31
|
+
* @param onEnd - The function to call when the game ends.
|
|
32
|
+
* @param onInvalidInput - The function to call when the input is invalid.
|
|
27
33
|
*/
|
|
28
|
-
// eslint-disable-next-line @typescript-eslint/max-params
|
|
34
|
+
// eslint-disable-next-line @typescript-eslint/max-params
|
|
29
35
|
constructor(playerTypes, board, render, gameID, onEnd, onInvalidInput) {
|
|
30
36
|
super();
|
|
31
|
-
this.gameID = gameID
|
|
37
|
+
this.gameID = gameID;
|
|
32
38
|
this.board = board;
|
|
33
|
-
this.board.setGameID(this.gameID);
|
|
34
39
|
this.players = playerTypes.map((playerType, id) => ({ id, playerType }));
|
|
35
|
-
this.render = render;
|
|
40
|
+
this.render = render.bind(null, this);
|
|
36
41
|
if (onEnd !== undefined)
|
|
37
42
|
this.on("end", onEnd);
|
|
38
43
|
if (onInvalidInput !== undefined)
|
|
39
44
|
this.on("invalidInput", onInvalidInput);
|
|
40
45
|
}
|
|
41
|
-
/**
|
|
46
|
+
/**
|
|
47
|
+
* Gets the current player object.
|
|
48
|
+
* @returns The current player object.
|
|
49
|
+
*/
|
|
42
50
|
get currentPlayer() {
|
|
43
51
|
return this.players[this.currentPlayerId];
|
|
44
52
|
}
|
|
45
53
|
/**
|
|
46
54
|
* Controls the main game flow.
|
|
47
|
-
* @param algorithm The algorithm to use.
|
|
55
|
+
* @param algorithm - The algorithm to use.
|
|
48
56
|
* @returns The winner or null in the event of a tie.
|
|
49
57
|
*/
|
|
50
58
|
async play(algorithm = "alphabeta") {
|
|
51
|
-
await this.render(
|
|
59
|
+
await this.render();
|
|
52
60
|
this.on("input", async (move) => {
|
|
53
61
|
await this.makeMove(move);
|
|
54
62
|
if (this.currentPlayer.playerType !== "human")
|
|
@@ -59,10 +67,9 @@ export default class Controller extends EventEmitter {
|
|
|
59
67
|
await this.makeMove(algorithm);
|
|
60
68
|
}
|
|
61
69
|
/**
|
|
62
|
-
* The bog standard minimax algorithm. Left in for reference.
|
|
63
|
-
* @
|
|
64
|
-
* @param
|
|
65
|
-
* @param maximisingPlayer Whether or not the current player is the maximising player.
|
|
70
|
+
* The bog standard minimax algorithm. Left in for reference (https://en.wikipedia.org/wiki/Minimax).
|
|
71
|
+
* @param depth - The depth of the algorithm.
|
|
72
|
+
* @param maximisingPlayer - Whether or not the current player is the maximising player.
|
|
66
73
|
* @returns The optimal move.
|
|
67
74
|
*/
|
|
68
75
|
minimax(depth = Infinity, maximisingPlayer = true) {
|
|
@@ -96,12 +103,11 @@ export default class Controller extends EventEmitter {
|
|
|
96
103
|
return bestMove;
|
|
97
104
|
}
|
|
98
105
|
/**
|
|
99
|
-
* The minimax algorithm with alpha-beta pruning.
|
|
100
|
-
* @
|
|
101
|
-
* @param
|
|
102
|
-
* @param
|
|
103
|
-
* @param
|
|
104
|
-
* @param maximisingPlayer Whether or not the current player is the maximising player.
|
|
106
|
+
* The minimax algorithm with alpha-beta pruning (https://en.wikipedia.org/wiki/Minimax).
|
|
107
|
+
* @param depth - The depth of the algorithm.
|
|
108
|
+
* @param alpha - The bounds for the alpha-beta variation of the algorithm.
|
|
109
|
+
* @param beta - The bounds for the alpha-beta variation of the algorithm.
|
|
110
|
+
* @param maximisingPlayer - Whether or not the current player is the maximising player.
|
|
105
111
|
* @returns The optimal move.
|
|
106
112
|
*/
|
|
107
113
|
alphabeta(depth = Infinity, alpha = -Infinity, beta = Infinity, maximisingPlayer = true) {
|
|
@@ -148,7 +154,7 @@ export default class Controller extends EventEmitter {
|
|
|
148
154
|
}
|
|
149
155
|
/**
|
|
150
156
|
* Makes a move.
|
|
151
|
-
* @param input Either the algorithm to use to calculate the move or the move itself.
|
|
157
|
+
* @param input - Either the algorithm to use to calculate the move or the move itself.
|
|
152
158
|
*/
|
|
153
159
|
async makeMove(input) {
|
|
154
160
|
if (typeof input === "string") {
|
|
@@ -161,7 +167,7 @@ export default class Controller extends EventEmitter {
|
|
|
161
167
|
}
|
|
162
168
|
this.board.makeMove(input, this.currentPlayer.id);
|
|
163
169
|
}
|
|
164
|
-
await this.render(
|
|
170
|
+
await this.render();
|
|
165
171
|
const { winner } = this.board;
|
|
166
172
|
if (winner !== false) {
|
|
167
173
|
this.emit("end", winner);
|
|
@@ -170,4 +176,4 @@ export default class Controller extends EventEmitter {
|
|
|
170
176
|
this.nextTurn();
|
|
171
177
|
}
|
|
172
178
|
}
|
|
173
|
-
//# 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 { Position } from \"./board.js\";\n\nexport type PlayerType = \"easyCPU\" | \"hardCPU\" | \"human\" | \"impossibleCPU\" | \"mediumCPU\";\nexport type Algorithm = \"alphabeta\" | \"minimax\";\nexport type GameConstructorOptions = {\n    id?: unknown;\n    onEnd?: (winner: number | null) => Promise<void> | void;\n    onInvalidInput?: (position: Position) => Promise<void> | void;\n    renderer?: (controller: Controller) => Promise<void> | void;\n};\nexport type GameConstructor = {\n    // eslint-disable-next-line @typescript-eslint/prefer-function-type\n    new (playerOneType: PlayerType, playerTwoType: PlayerType, options?: GameConstructorOptions): Controller;\n};\n/**\n * Decorator to check that the constructor type for the given class is correct.\n * @param constructor The class to check.\n */\nexport function Game(constructor: GameConstructor): void {\n    void constructor;\n}\n\n/** Represents a game controller. */\nexport default abstract class Controller extends EventEmitter<{\n    end: GameConstructorOptions[\"onEnd\"];\n    input: GameConstructorOptions[\"onInvalidInput\"];\n    invalidInput: GameConstructorOptions[\"onInvalidInput\"];\n}> {\n    /** Contains the ID of the game. */\n    public readonly gameID: unknown;\n\n    /** Contains the board. */\n    public readonly board: Board;\n\n    /** Contains the view object. */\n    public readonly render: Required<GameConstructorOptions>[\"renderer\"];\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.\n     * @param view The view object.\n     * @param gameID The ID of the game.\n     * @param board The board.\n     */\n    // eslint-disable-next-line @typescript-eslint/max-params, require-jsdoc\n    protected constructor(\n        playerTypes: PlayerType[],\n        board: Board,\n        render: Controller[\"render\"],\n        gameID: unknown,\n        onEnd?: GameConstructorOptions[\"onEnd\"],\n        onInvalidInput?: GameConstructorOptions[\"onInvalidInput\"],\n    ) {\n        super();\n        this.gameID = gameID ?? Date.now().toString(16);\n        this.board = board;\n        this.board.setGameID(this.gameID);\n        this.players = playerTypes.map((playerType, id) => ({ id, playerType }));\n        this.render = render;\n\n        if (onEnd !== undefined)\n            this.on(\"end\", onEnd);\n\n        if (onInvalidInput !== undefined)\n            this.on(\"invalidInput\", onInvalidInput);\n    }\n\n    /** Gets the current player object. */\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(this);\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        });\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.\n     * @link 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.\n     * @link 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(this);\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.\n     * @returns The optimal move.\n     */\n    public abstract findOptimalMove(options?: { algorithm?: Algorithm; maxDepth?: number; randomMove?: Position; }): Position;\n}\n"]}
|
|
179
|
+
//# 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> = {\n    id: T;\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> = {\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 the game ID.\n * @param constructor - The class to check.\n */\nexport function Game<T>(constructor: GameConstructor<T>): void {\n    void constructor;\n}\n\n/**\n * Represents a game controller.\n * @template T - The type of the game ID.\n */\nexport default abstract class Controller<T> extends EventEmitter<{\n    end: GameConstructorOptions<T>[\"onEnd\"];\n    input: GameConstructorOptions<T>[\"onInvalidInput\"];\n    invalidInput: GameConstructorOptions<T>[\"onInvalidInput\"];\n}> {\n    /** Contains the ID of the game. */\n    public readonly gameID: T;\n\n    /** Contains the board. */\n    public readonly board: Board<LongInt | number>;\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<LongInt | number>,\n        render: Required<GameConstructorOptions<T>>[\"renderer\"],\n        gameID: T,\n        onEnd?: GameConstructorOptions<T>[\"onEnd\"],\n        onInvalidInput?: GameConstructorOptions<T>[\"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);\n\n        if (onInvalidInput !== undefined)\n            this.on(\"invalidInput\", onInvalidInput);\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        });\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"]}
|
|
@@ -1,50 +1,53 @@
|
|
|
1
1
|
import type LongInt from "./longInt.js";
|
|
2
|
-
import
|
|
2
|
+
import { StringType } from "./longInt.js";
|
|
3
3
|
/**
|
|
4
|
-
* Represents a BitBoard.
|
|
5
|
-
* @
|
|
4
|
+
* Represents a BitBoard (https://en.wikipedia.org/wiki/Bitboard).
|
|
5
|
+
* @template T - The type of the numeric data.
|
|
6
6
|
*/
|
|
7
|
-
export default abstract class BitBoard<T extends LongInt | number
|
|
7
|
+
export default abstract class BitBoard<T extends LongInt | number> {
|
|
8
8
|
/** The numeric data. */
|
|
9
9
|
protected _data: T;
|
|
10
10
|
/**
|
|
11
11
|
* Creates an instance of BitBoard.
|
|
12
|
-
* @param data The data to assign to the BitBoard.
|
|
12
|
+
* @param data - The data to assign to the BitBoard.
|
|
13
13
|
*/
|
|
14
14
|
protected constructor(data: T);
|
|
15
|
-
/**
|
|
15
|
+
/**
|
|
16
|
+
* Gets the numeric data.
|
|
17
|
+
* @returns The numeric data.
|
|
18
|
+
*/
|
|
16
19
|
get data(): T;
|
|
17
20
|
/**
|
|
18
21
|
* Returns a string representation of the BitBoard.
|
|
19
|
-
* @param type The base of the string to print.
|
|
22
|
+
* @param type - The base of the string to print.
|
|
20
23
|
* @returns The string representation.
|
|
21
24
|
*/
|
|
22
25
|
toString(type?: StringType): string;
|
|
23
26
|
/**
|
|
24
27
|
* Assigns a given bit a given value.
|
|
25
|
-
* @param bit The x coordinate.
|
|
26
|
-
* @param value The value to assign to the bit.
|
|
28
|
+
* @param bit - The x coordinate.
|
|
29
|
+
* @param value - The value to assign to the bit.
|
|
27
30
|
*/
|
|
28
31
|
assignBit(bit: number, value: 0 | 1): void;
|
|
29
32
|
/**
|
|
30
|
-
* Gets a given bit
|
|
31
|
-
* @param bit The index of the bit to get, 0 = LSB.
|
|
32
|
-
* @returns The bit.
|
|
33
|
+
* Gets a given bit.
|
|
34
|
+
* @param bit - The index of the bit to get, 0 = LSB.
|
|
35
|
+
* @returns The bit value.
|
|
33
36
|
*/
|
|
34
37
|
abstract getBit(bit: number): 0 | 1;
|
|
35
38
|
/**
|
|
36
|
-
* Sets a given bit (changes the value to 1)
|
|
37
|
-
* @param bit The index of the bit to get, 0 = LSB.
|
|
39
|
+
* Sets a given bit (changes the value to 1).
|
|
40
|
+
* @param bit - The index of the bit to get, 0 = LSB.
|
|
38
41
|
*/
|
|
39
42
|
abstract setBit(bit: number): void;
|
|
40
43
|
/**
|
|
41
|
-
* Clears a given bit (changes the value to 0)
|
|
42
|
-
* @param bit The index of the bit to get, 0 = LSB.
|
|
44
|
+
* Clears a given bit (changes the value to 0).
|
|
45
|
+
* @param bit - The index of the bit to get, 0 = LSB.
|
|
43
46
|
*/
|
|
44
47
|
abstract clearBit(bit: number): void;
|
|
45
48
|
/**
|
|
46
|
-
* Toggles a given bit (changes the value from 0 to 1 or 1 to 0)
|
|
47
|
-
* @param bit The index of the bit to get, 0 = LSB.
|
|
49
|
+
* Toggles a given bit (changes the value from 0 to 1 or 1 to 0).
|
|
50
|
+
* @param bit - The index of the bit to get, 0 = LSB.
|
|
48
51
|
*/
|
|
49
52
|
abstract toggleBit(bit: number): void;
|
|
50
53
|
/** Clears the whole BitBoard (sets all values to 0). */
|
|
@@ -53,55 +56,55 @@ export default abstract class BitBoard<T extends LongInt | number = LongInt | nu
|
|
|
53
56
|
abstract setAll(): void;
|
|
54
57
|
/**
|
|
55
58
|
* Gets a range of bits.
|
|
56
|
-
* @param LSB The least significant bit.
|
|
57
|
-
* @param numberOfBits The number of bits to get.
|
|
59
|
+
* @param LSB - The least significant bit.
|
|
60
|
+
* @param numberOfBits - The number of bits to get.
|
|
58
61
|
* @returns The range of bits.
|
|
59
62
|
*/
|
|
60
63
|
abstract getBits(LSB: number, numberOfBits: number): T;
|
|
61
64
|
/**
|
|
62
65
|
* Carries out a bitwise and (&) operation.
|
|
63
|
-
* @param right The right value.
|
|
66
|
+
* @param right - The right value.
|
|
64
67
|
* @returns The result.
|
|
65
68
|
*/
|
|
66
|
-
abstract and(right: BitBoard<T> | T | number):
|
|
69
|
+
abstract and(right: BitBoard<T> | T | number): BitBoard<T>;
|
|
67
70
|
/**
|
|
68
71
|
* Carries out a bitwise or (|) operation.
|
|
69
|
-
* @param right The right value.
|
|
72
|
+
* @param right - The right value.
|
|
70
73
|
* @returns The result.
|
|
71
74
|
*/
|
|
72
|
-
abstract or(right: BitBoard<T> | T | number):
|
|
75
|
+
abstract or(right: BitBoard<T> | T | number): BitBoard<T>;
|
|
73
76
|
/**
|
|
74
77
|
* Carries out a bitwise xor (^) operation.
|
|
75
|
-
* @param right The right value.
|
|
78
|
+
* @param right - The right value.
|
|
76
79
|
* @returns The result.
|
|
77
80
|
*/
|
|
78
|
-
abstract xor(right: BitBoard<T> | T | number):
|
|
81
|
+
abstract xor(right: BitBoard<T> | T | number): BitBoard<T>;
|
|
79
82
|
/**
|
|
80
83
|
* Carries out a bitwise not (~) operation.
|
|
81
84
|
* @returns The result.
|
|
82
85
|
*/
|
|
83
|
-
abstract not():
|
|
86
|
+
abstract not(): BitBoard<T>;
|
|
84
87
|
/**
|
|
85
88
|
* Carries out a bitwise left shift (<<) operation.
|
|
86
|
-
* @param shiftAmount How much to shift it by.
|
|
89
|
+
* @param shiftAmount - How much to shift it by.
|
|
87
90
|
* @returns The result.
|
|
88
91
|
*/
|
|
89
|
-
abstract leftShift(shiftAmount: number):
|
|
92
|
+
abstract leftShift(shiftAmount: number): BitBoard<T>;
|
|
90
93
|
/**
|
|
91
94
|
* Carries out a bitwise logical right shift (>>>) operation.
|
|
92
|
-
* @param shiftAmount How much to shift it by.
|
|
95
|
+
* @param shiftAmount - How much to shift it by.
|
|
93
96
|
* @returns The result.
|
|
94
97
|
*/
|
|
95
|
-
abstract rightShift(shiftAmount: number):
|
|
98
|
+
abstract rightShift(shiftAmount: number): BitBoard<T>;
|
|
96
99
|
/**
|
|
97
100
|
* Carries out a bitwise arithmetic right shift (>>) operation.
|
|
98
|
-
* @param shiftAmount How much to shift it by.
|
|
101
|
+
* @param shiftAmount - How much to shift it by.
|
|
99
102
|
* @returns The result.
|
|
100
103
|
*/
|
|
101
|
-
abstract arithmeticRightShift(shiftAmount: number):
|
|
104
|
+
abstract arithmeticRightShift(shiftAmount: number): BitBoard<T>;
|
|
102
105
|
/**
|
|
103
106
|
* Checks if two BitBoards have equal data values.
|
|
104
|
-
* @param value The value to compare against.
|
|
107
|
+
* @param value - The value to compare against.
|
|
105
108
|
* @returns Whether or not the two BitBoard have the same data value.
|
|
106
109
|
*/
|
|
107
110
|
abstract equals(value: BitBoard<T> | T | number): boolean;
|
|
@@ -1,36 +1,39 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Represents a BitBoard.
|
|
3
|
-
* @
|
|
2
|
+
* Represents a BitBoard (https://en.wikipedia.org/wiki/Bitboard).
|
|
3
|
+
* @template T - The type of the numeric data.
|
|
4
4
|
*/
|
|
5
5
|
export default class BitBoard {
|
|
6
6
|
/** The numeric data. */
|
|
7
7
|
_data;
|
|
8
8
|
/**
|
|
9
9
|
* Creates an instance of BitBoard.
|
|
10
|
-
* @param data The data to assign to the BitBoard.
|
|
10
|
+
* @param data - The data to assign to the BitBoard.
|
|
11
11
|
*/
|
|
12
12
|
constructor(data) {
|
|
13
13
|
this._data = data;
|
|
14
14
|
}
|
|
15
|
-
/**
|
|
15
|
+
/**
|
|
16
|
+
* Gets the numeric data.
|
|
17
|
+
* @returns The numeric data.
|
|
18
|
+
*/
|
|
16
19
|
get data() {
|
|
17
20
|
return this._data;
|
|
18
21
|
}
|
|
19
22
|
/**
|
|
20
23
|
* Returns a string representation of the BitBoard.
|
|
21
|
-
* @param type The base of the string to print.
|
|
24
|
+
* @param type - The base of the string to print.
|
|
22
25
|
* @returns The string representation.
|
|
23
26
|
*/
|
|
24
|
-
toString(type = 16) {
|
|
27
|
+
toString(type = 16 /* StringType.HEXADECIMAL */) {
|
|
25
28
|
let padLength = 0;
|
|
26
29
|
switch (type) {
|
|
27
|
-
case 2
|
|
30
|
+
case 2 /* StringType.BINARY */:
|
|
28
31
|
padLength = 32;
|
|
29
32
|
break;
|
|
30
|
-
case 10
|
|
33
|
+
case 10 /* StringType.DECIMAL */:
|
|
31
34
|
padLength = 10;
|
|
32
35
|
break;
|
|
33
|
-
case 16
|
|
36
|
+
case 16 /* StringType.HEXADECIMAL */:
|
|
34
37
|
padLength = 8;
|
|
35
38
|
break;
|
|
36
39
|
}
|
|
@@ -38,8 +41,8 @@ export default class BitBoard {
|
|
|
38
41
|
}
|
|
39
42
|
/**
|
|
40
43
|
* Assigns a given bit a given value.
|
|
41
|
-
* @param bit The x coordinate.
|
|
42
|
-
* @param value The value to assign to the bit.
|
|
44
|
+
* @param bit - The x coordinate.
|
|
45
|
+
* @param value - The value to assign to the bit.
|
|
43
46
|
*/
|
|
44
47
|
assignBit(bit, value) {
|
|
45
48
|
if (value === 0)
|
|
@@ -48,4 +51,4 @@ export default class BitBoard {
|
|
|
48
51
|
this.setBit(bit);
|
|
49
52
|
}
|
|
50
53
|
}
|
|
51
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
54
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYml0Qm9hcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYml0Qm9hcmQvYml0Qm9hcmQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBR0E7OztHQUdHO0FBQ0gsTUFBTSxDQUFDLE9BQU8sT0FBZ0IsUUFBUTtJQUNsQyx3QkFBd0I7SUFDZCxLQUFLLENBQUk7SUFFbkI7OztPQUdHO0lBQ0gsWUFBc0IsSUFBTztRQUN6QixJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQztJQUN0QixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsSUFBVyxJQUFJO1FBQ1gsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDO0lBQ3RCLENBQUM7SUFFRDs7OztPQUlHO0lBQ0ksUUFBUSxDQUFDLHNDQUF5QztRQUNyRCxJQUFJLFNBQVMsR0FBRyxDQUFDLENBQUM7UUFFbEIsUUFBUSxJQUFJLEVBQUUsQ0FBQztZQUNYO2dCQUNJLFNBQVMsR0FBRyxFQUFFLENBQUM7Z0JBQ2YsTUFBTTtZQUNWO2dCQUNJLFNBQVMsR0FBRyxFQUFFLENBQUM7Z0JBQ2YsTUFBTTtZQUNWO2dCQUNJLFNBQVMsR0FBRyxDQUFDLENBQUM7Z0JBQ2QsTUFBTTtRQUNkLENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDOUQsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxTQUFTLENBQUMsR0FBVyxFQUFFLEtBQVk7UUFDdEMsSUFBSSxLQUFLLEtBQUssQ0FBQztZQUNYLElBQUksQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUM7O1lBRW5CLElBQUksQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDekIsQ0FBQztDQStGSiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIExvbmdJbnQgZnJvbSBcIi4vbG9uZ0ludC5qc1wiO1xuaW1wb3J0IHsgU3RyaW5nVHlwZSB9IGZyb20gXCIuL2xvbmdJbnQuanNcIjtcblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgQml0Qm9hcmQgKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0JpdGJvYXJkKS5cbiAqIEB0ZW1wbGF0ZSBUIC0gVGhlIHR5cGUgb2YgdGhlIG51bWVyaWMgZGF0YS5cbiAqL1xuZXhwb3J0IGRlZmF1bHQgYWJzdHJhY3QgY2xhc3MgQml0Qm9hcmQ8VCBleHRlbmRzIExvbmdJbnQgfCBudW1iZXI+IHtcbiAgICAvKiogVGhlIG51bWVyaWMgZGF0YS4gKi9cbiAgICBwcm90ZWN0ZWQgX2RhdGE6IFQ7XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGFuIGluc3RhbmNlIG9mIEJpdEJvYXJkLlxuICAgICAqIEBwYXJhbSBkYXRhIC0gVGhlIGRhdGEgdG8gYXNzaWduIHRvIHRoZSBCaXRCb2FyZC5cbiAgICAgKi9cbiAgICBwcm90ZWN0ZWQgY29uc3RydWN0b3IoZGF0YTogVCkge1xuICAgICAgICB0aGlzLl9kYXRhID0gZGF0YTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBHZXRzIHRoZSBudW1lcmljIGRhdGEuXG4gICAgICogQHJldHVybnMgVGhlIG51bWVyaWMgZGF0YS5cbiAgICAgKi9cbiAgICBwdWJsaWMgZ2V0IGRhdGEoKTogVCB7XG4gICAgICAgIHJldHVybiB0aGlzLl9kYXRhO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybnMgYSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIEJpdEJvYXJkLlxuICAgICAqIEBwYXJhbSB0eXBlIC0gVGhlIGJhc2Ugb2YgdGhlIHN0cmluZyB0byBwcmludC5cbiAgICAgKiBAcmV0dXJucyBUaGUgc3RyaW5nIHJlcHJlc2VudGF0aW9uLlxuICAgICAqL1xuICAgIHB1YmxpYyB0b1N0cmluZyh0eXBlOiBTdHJpbmdUeXBlID0gU3RyaW5nVHlwZS5IRVhBREVDSU1BTCk6IHN0cmluZyB7XG4gICAgICAgIGxldCBwYWRMZW5ndGggPSAwO1xuXG4gICAgICAgIHN3aXRjaCAodHlwZSkge1xuICAgICAgICAgICAgY2FzZSBTdHJpbmdUeXBlLkJJTkFSWTpcbiAgICAgICAgICAgICAgICBwYWRMZW5ndGggPSAzMjtcbiAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIGNhc2UgU3RyaW5nVHlwZS5ERUNJTUFMOlxuICAgICAgICAgICAgICAgIHBhZExlbmd0aCA9IDEwO1xuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSBTdHJpbmdUeXBlLkhFWEFERUNJTUFMOlxuICAgICAgICAgICAgICAgIHBhZExlbmd0aCA9IDg7XG4gICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gdGhpcy5fZGF0YS50b1N0cmluZyh0eXBlKS5wYWRTdGFydChwYWRMZW5ndGgsIFwiMFwiKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBBc3NpZ25zIGEgZ2l2ZW4gYml0IGEgZ2l2ZW4gdmFsdWUuXG4gICAgICogQHBhcmFtIGJpdCAtIFRoZSB4IGNvb3JkaW5hdGUuXG4gICAgICogQHBhcmFtIHZhbHVlIC0gVGhlIHZhbHVlIHRvIGFzc2lnbiB0byB0aGUgYml0LlxuICAgICAqL1xuICAgIHB1YmxpYyBhc3NpZ25CaXQoYml0OiBudW1iZXIsIHZhbHVlOiAwIHwgMSk6IHZvaWQge1xuICAgICAgICBpZiAodmFsdWUgPT09IDApXG4gICAgICAgICAgICB0aGlzLmNsZWFyQml0KGJpdCk7XG4gICAgICAgIGVsc2VcbiAgICAgICAgICAgIHRoaXMuc2V0Qml0KGJpdCk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogR2V0cyBhIGdpdmVuIGJpdC5cbiAgICAgKiBAcGFyYW0gYml0IC0gVGhlIGluZGV4IG9mIHRoZSBiaXQgdG8gZ2V0LCAwID0gTFNCLlxuICAgICAqIEByZXR1cm5zIFRoZSBiaXQgdmFsdWUuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IGdldEJpdChiaXQ6IG51bWJlcik6IDAgfCAxO1xuXG4gICAgLyoqXG4gICAgICogU2V0cyBhIGdpdmVuIGJpdCAoY2hhbmdlcyB0aGUgdmFsdWUgdG8gMSkuXG4gICAgICogQHBhcmFtIGJpdCAtIFRoZSBpbmRleCBvZiB0aGUgYml0IHRvIGdldCwgMCA9IExTQi5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3Qgc2V0Qml0KGJpdDogbnVtYmVyKTogdm9pZDtcblxuICAgIC8qKlxuICAgICAqIENsZWFycyBhIGdpdmVuIGJpdCAoY2hhbmdlcyB0aGUgdmFsdWUgdG8gMCkuXG4gICAgICogQHBhcmFtIGJpdCAtIFRoZSBpbmRleCBvZiB0aGUgYml0IHRvIGdldCwgMCA9IExTQi5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgY2xlYXJCaXQoYml0OiBudW1iZXIpOiB2b2lkO1xuXG4gICAgLyoqXG4gICAgICogVG9nZ2xlcyBhIGdpdmVuIGJpdCAoY2hhbmdlcyB0aGUgdmFsdWUgZnJvbSAwIHRvIDEgb3IgMSB0byAwKS5cbiAgICAgKiBAcGFyYW0gYml0IC0gVGhlIGluZGV4IG9mIHRoZSBiaXQgdG8gZ2V0LCAwID0gTFNCLlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCB0b2dnbGVCaXQoYml0OiBudW1iZXIpOiB2b2lkO1xuXG4gICAgLyoqIENsZWFycyB0aGUgd2hvbGUgQml0Qm9hcmQgKHNldHMgYWxsIHZhbHVlcyB0byAwKS4gKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgY2xlYXJBbGwoKTogdm9pZDtcblxuICAgIC8qKiBTZXQgdGhlIHdob2xlIEJpdEJvYXJkIChzZXRzIGFsbCB2YWx1ZXMgdG8gMSkuICovXG4gICAgcHVibGljIGFic3RyYWN0IHNldEFsbCgpOiB2b2lkO1xuXG4gICAgLyoqXG4gICAgICogR2V0cyBhIHJhbmdlIG9mIGJpdHMuXG4gICAgICogQHBhcmFtIExTQiAtIFRoZSBsZWFzdCBzaWduaWZpY2FudCBiaXQuXG4gICAgICogQHBhcmFtIG51bWJlck9mQml0cyAtIFRoZSBudW1iZXIgb2YgYml0cyB0byBnZXQuXG4gICAgICogQHJldHVybnMgVGhlIHJhbmdlIG9mIGJpdHMuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IGdldEJpdHMoTFNCOiBudW1iZXIsIG51bWJlck9mQml0czogbnVtYmVyKTogVDtcblxuICAgIC8qKlxuICAgICAqIENhcnJpZXMgb3V0IGEgYml0d2lzZSBhbmQgKCYpIG9wZXJhdGlvbi5cbiAgICAgKiBAcGFyYW0gcmlnaHQgLSBUaGUgcmlnaHQgdmFsdWUuXG4gICAgICogQHJldHVybnMgVGhlIHJlc3VsdC5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgYW5kKHJpZ2h0OiBCaXRCb2FyZDxUPiB8IFQgfCBudW1iZXIpOiBCaXRCb2FyZDxUPjtcblxuICAgIC8qKlxuICAgICAqIENhcnJpZXMgb3V0IGEgYml0d2lzZSBvciAofCkgb3BlcmF0aW9uLlxuICAgICAqIEBwYXJhbSByaWdodCAtIFRoZSByaWdodCB2YWx1ZS5cbiAgICAgKiBAcmV0dXJucyBUaGUgcmVzdWx0LlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCBvcihyaWdodDogQml0Qm9hcmQ8VD4gfCBUIHwgbnVtYmVyKTogQml0Qm9hcmQ8VD47XG5cbiAgICAvKipcbiAgICAgKiBDYXJyaWVzIG91dCBhIGJpdHdpc2UgeG9yICheKSBvcGVyYXRpb24uXG4gICAgICogQHBhcmFtIHJpZ2h0IC0gVGhlIHJpZ2h0IHZhbHVlLlxuICAgICAqIEByZXR1cm5zIFRoZSByZXN1bHQuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IHhvcihyaWdodDogQml0Qm9hcmQ8VD4gfCBUIHwgbnVtYmVyKTogQml0Qm9hcmQ8VD47XG5cbiAgICAvKipcbiAgICAgKiBDYXJyaWVzIG91dCBhIGJpdHdpc2Ugbm90ICh+KSBvcGVyYXRpb24uXG4gICAgICogQHJldHVybnMgVGhlIHJlc3VsdC5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3Qgbm90KCk6IEJpdEJvYXJkPFQ+O1xuXG4gICAgLyoqXG4gICAgICogQ2FycmllcyBvdXQgYSBiaXR3aXNlIGxlZnQgc2hpZnQgKDw8KSBvcGVyYXRpb24uXG4gICAgICogQHBhcmFtIHNoaWZ0QW1vdW50IC0gSG93IG11Y2ggdG8gc2hpZnQgaXQgYnkuXG4gICAgICogQHJldHVybnMgVGhlIHJlc3VsdC5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgbGVmdFNoaWZ0KHNoaWZ0QW1vdW50OiBudW1iZXIpOiBCaXRCb2FyZDxUPjtcblxuICAgIC8qKlxuICAgICAqIENhcnJpZXMgb3V0IGEgYml0d2lzZSBsb2dpY2FsIHJpZ2h0IHNoaWZ0ICg+Pj4pIG9wZXJhdGlvbi5cbiAgICAgKiBAcGFyYW0gc2hpZnRBbW91bnQgLSBIb3cgbXVjaCB0byBzaGlmdCBpdCBieS5cbiAgICAgKiBAcmV0dXJucyBUaGUgcmVzdWx0LlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCByaWdodFNoaWZ0KHNoaWZ0QW1vdW50OiBudW1iZXIpOiBCaXRCb2FyZDxUPjtcblxuICAgIC8qKlxuICAgICAqIENhcnJpZXMgb3V0IGEgYml0d2lzZSBhcml0aG1ldGljIHJpZ2h0IHNoaWZ0ICg+Pikgb3BlcmF0aW9uLlxuICAgICAqIEBwYXJhbSBzaGlmdEFtb3VudCAtIEhvdyBtdWNoIHRvIHNoaWZ0IGl0IGJ5LlxuICAgICAqIEByZXR1cm5zIFRoZSByZXN1bHQuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IGFyaXRobWV0aWNSaWdodFNoaWZ0KHNoaWZ0QW1vdW50OiBudW1iZXIpOiBCaXRCb2FyZDxUPjtcblxuICAgIC8qKlxuICAgICAqIENoZWNrcyBpZiB0d28gQml0Qm9hcmRzIGhhdmUgZXF1YWwgZGF0YSB2YWx1ZXMuXG4gICAgICogQHBhcmFtIHZhbHVlIC0gVGhlIHZhbHVlIHRvIGNvbXBhcmUgYWdhaW5zdC5cbiAgICAgKiBAcmV0dXJucyBXaGV0aGVyIG9yIG5vdCB0aGUgdHdvIEJpdEJvYXJkIGhhdmUgdGhlIHNhbWUgZGF0YSB2YWx1ZS5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgZXF1YWxzKHZhbHVlOiBCaXRCb2FyZDxUPiB8IFQgfCBudW1iZXIpOiBib29sZWFuO1xufVxuIl19
|