@oathompsonjones/mini-games 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,124 @@
1
+ import type Board from "./board.js";
2
+ import { EventEmitter } from "eventemitter3";
3
+ import type { Position } from "./board.js";
4
+ export type PlayerType = "easyCPU" | "hardCPU" | "human" | "impossibleCPU" | "mediumCPU";
5
+ export type Algorithm = "alphabeta" | "minimax";
6
+ export interface GameConstructorOptions {
7
+ id?: string;
8
+ onEnd?: (winner: number | null) => Promise<void> | void;
9
+ onInvalidInput?: (position: Position) => Promise<void> | void;
10
+ renderer?: (controller: Controller) => Promise<void> | void;
11
+ }
12
+ export interface GameConstructor {
13
+ new (playerOneType: PlayerType, playerTwoType: PlayerType, options?: GameConstructorOptions): Controller;
14
+ }
15
+ export declare function Game(constructor: GameConstructor): void;
16
+ /**
17
+ * Represents a game controller.
18
+ */
19
+ export default abstract class Controller extends EventEmitter<{
20
+ end: GameConstructorOptions["onEnd"];
21
+ input: GameConstructorOptions["onInvalidInput"];
22
+ invalidInput: GameConstructorOptions["onInvalidInput"];
23
+ }> {
24
+ /**
25
+ * Contains the ID of the game.
26
+ */
27
+ readonly gameID: string;
28
+ /**
29
+ * Contains the board.
30
+ */
31
+ readonly board: Board;
32
+ /**
33
+ * Contains the view object.
34
+ */
35
+ readonly render: Required<GameConstructorOptions>["renderer"];
36
+ /**
37
+ * Contains the player objects.
38
+ */
39
+ protected readonly players: Array<{
40
+ id: number;
41
+ playerType: PlayerType;
42
+ }>;
43
+ /**
44
+ * Contains the ID of the current player.
45
+ */
46
+ private currentPlayerId;
47
+ /**
48
+ * Creates an instance of Controller.
49
+ *
50
+ * @param playerTypes The types of player.
51
+ * @param view The view object.
52
+ * @param gameID The ID of the game.
53
+ * @param board The board.
54
+ */
55
+ protected constructor(playerTypes: PlayerType[], board: Board, render: Controller["render"], gameID: string | undefined, onEnd?: GameConstructorOptions["onEnd"], onInvalidInput?: GameConstructorOptions["onInvalidInput"]);
56
+ /**
57
+ * Gets the current player object.
58
+ */
59
+ get currentPlayer(): {
60
+ id: number;
61
+ playerType: PlayerType;
62
+ };
63
+ /**
64
+ * Controls the main game flow.
65
+ *
66
+ * @returns The winner or null in the event of a tie.
67
+ */
68
+ play(algorithm?: Algorithm): Promise<void>;
69
+ /**
70
+ * The bog standard minimax algorithm. Left in for reference.
71
+ *
72
+ * @link https://en.wikipedia.org/wiki/Minimax
73
+ * @param depth The depth of the algorithm.
74
+ * @param maximisingPlayer Whether or not the current player is the maximising player.
75
+ * @returns The optimal move.
76
+ */
77
+ protected minimax(depth?: number, maximisingPlayer?: boolean): {
78
+ move: Position;
79
+ score: number;
80
+ };
81
+ /**
82
+ * The minimax algorithm with alpha-beta pruning.
83
+ *
84
+ * @link https://en.wikipedia.org/wiki/Minimax
85
+ * @param depth The depth of the algorithm.
86
+ * @param alpha The bounds for the alpha-beta variation of the algorithm.
87
+ * @param beta The bounds for the alpha-beta variation of the algorithm.
88
+ * @param maximisingPlayer Whether or not the current player is the maximising player.
89
+ * @returns The optimal move.
90
+ */
91
+ protected alphabeta(depth?: number, alpha?: number, beta?: number, maximisingPlayer?: boolean): {
92
+ move: Position;
93
+ score: number;
94
+ };
95
+ /**
96
+ * Changes which player's turn it is.
97
+ */
98
+ private nextTurn;
99
+ /**
100
+ * Makes a move.
101
+ *
102
+ * @param input Either the algorithm to use to calculate the move or the move itself.
103
+ */
104
+ private makeMove;
105
+ /**
106
+ * Determines the CPU move.
107
+ *
108
+ * @param difficulty The difficulty of the AI.
109
+ * @param algorithm The algorithm to use.
110
+ * @returns The move.
111
+ */
112
+ abstract determineCPUMove(difficulty: Omit<PlayerType, "human">, algorithm?: Algorithm): Position;
113
+ /**
114
+ * Finds the optimal move.
115
+ *
116
+ * @param options The options.
117
+ * @returns The optimal move.
118
+ */
119
+ abstract findOptimalMove(options?: {
120
+ algorithm?: Algorithm;
121
+ maxDepth?: number;
122
+ randomMove?: Position;
123
+ }): Position;
124
+ }
@@ -0,0 +1,185 @@
1
+ import { EventEmitter } from "eventemitter3";
2
+ export function Game(constructor) {
3
+ void constructor;
4
+ }
5
+ /**
6
+ * Represents a game controller.
7
+ */
8
+ export default class Controller extends EventEmitter {
9
+ /**
10
+ * Contains the ID of the game.
11
+ */
12
+ gameID;
13
+ /**
14
+ * Contains the board.
15
+ */
16
+ board;
17
+ /**
18
+ * Contains the view object.
19
+ */
20
+ render;
21
+ /**
22
+ * Contains the player objects.
23
+ */
24
+ players;
25
+ /**
26
+ * Contains the ID of the current player.
27
+ */
28
+ currentPlayerId = 0;
29
+ /**
30
+ * Creates an instance of Controller.
31
+ *
32
+ * @param playerTypes The types of player.
33
+ * @param view The view object.
34
+ * @param gameID The ID of the game.
35
+ * @param board The board.
36
+ */
37
+ // eslint-disable-next-line @typescript-eslint/max-params
38
+ constructor(playerTypes, board, render, gameID, onEnd, onInvalidInput) {
39
+ super();
40
+ this.gameID = gameID ?? Date.now().toString(16);
41
+ this.board = board;
42
+ this.board.setGameID(this.gameID);
43
+ this.players = playerTypes.map((playerType, id) => ({ id, playerType }));
44
+ this.render = render;
45
+ if (onEnd !== undefined)
46
+ this.on("end", onEnd);
47
+ if (onInvalidInput !== undefined)
48
+ this.on("invalidInput", onInvalidInput);
49
+ }
50
+ /**
51
+ * Gets the current player object.
52
+ */
53
+ get currentPlayer() {
54
+ return this.players[this.currentPlayerId];
55
+ }
56
+ /**
57
+ * Controls the main game flow.
58
+ *
59
+ * @returns The winner or null in the event of a tie.
60
+ */
61
+ async play(algorithm = "alphabeta") {
62
+ await this.render(this);
63
+ this.on("input", async (move) => {
64
+ await this.makeMove(move);
65
+ if (this.currentPlayer.playerType !== "human")
66
+ await this.makeMove(algorithm);
67
+ });
68
+ while (this.currentPlayer.playerType !== "human" && this.board.winner === false)
69
+ // eslint-disable-next-line no-await-in-loop
70
+ await this.makeMove(algorithm);
71
+ }
72
+ /**
73
+ * The bog standard minimax algorithm. Left in for reference.
74
+ *
75
+ * @link https://en.wikipedia.org/wiki/Minimax
76
+ * @param depth The depth of the algorithm.
77
+ * @param maximisingPlayer Whether or not the current player is the maximising player.
78
+ * @returns The optimal move.
79
+ */
80
+ minimax(depth = Infinity, maximisingPlayer = true) {
81
+ const playerIds = [(this.currentPlayerId + 1) % 2, this.currentPlayerId];
82
+ if (depth === 0 || this.board.winner !== false) {
83
+ return {
84
+ move: { x: NaN, y: NaN },
85
+ score: this.board.heuristic * (this.currentPlayerId === 0 ? 1 : -1)
86
+ };
87
+ }
88
+ let bestMove = {
89
+ move: { x: NaN, y: NaN },
90
+ score: maximisingPlayer ? -Infinity : Infinity
91
+ };
92
+ const { emptyCells } = this.board;
93
+ for (const move of emptyCells) {
94
+ this.board.makeMove(move, playerIds[Number(maximisingPlayer)]);
95
+ const score = (9 - emptyCells.length) * this.minimax(depth - 1, !maximisingPlayer).score;
96
+ this.board.undoLastMove();
97
+ if (maximisingPlayer) {
98
+ const bestScore = Math.max(score, bestMove.score);
99
+ if (bestScore !== bestMove.score)
100
+ bestMove = { move, score };
101
+ }
102
+ else {
103
+ const bestScore = Math.min(score, bestMove.score);
104
+ if (bestScore !== bestMove.score)
105
+ bestMove = { move, score };
106
+ }
107
+ }
108
+ return bestMove;
109
+ }
110
+ /**
111
+ * The minimax algorithm with alpha-beta pruning.
112
+ *
113
+ * @link https://en.wikipedia.org/wiki/Minimax
114
+ * @param depth The depth of the algorithm.
115
+ * @param alpha The bounds for the alpha-beta variation of the algorithm.
116
+ * @param beta The bounds for the alpha-beta variation of the algorithm.
117
+ * @param maximisingPlayer Whether or not the current player is the maximising player.
118
+ * @returns The optimal move.
119
+ */
120
+ alphabeta(depth = Infinity, alpha = -Infinity, beta = Infinity, maximisingPlayer = true) {
121
+ const playerIds = [(this.currentPlayerId + 1) % 2, this.currentPlayerId];
122
+ if (depth === 0 || this.board.winner !== false) {
123
+ return {
124
+ move: { x: NaN, y: NaN },
125
+ score: this.board.heuristic * (this.currentPlayerId === 0 ? 1 : -1)
126
+ };
127
+ }
128
+ let bestMove = {
129
+ move: { x: NaN, y: NaN },
130
+ score: maximisingPlayer ? -Infinity : Infinity
131
+ };
132
+ const { emptyCells } = this.board;
133
+ for (const move of emptyCells) {
134
+ this.board.makeMove(move, playerIds[Number(maximisingPlayer)]);
135
+ const score = (9 - emptyCells.length) * this.alphabeta(depth - 1, alpha, beta, !maximisingPlayer).score;
136
+ this.board.undoLastMove();
137
+ if (maximisingPlayer) {
138
+ const bestScore = Math.max(score, bestMove.score);
139
+ if (bestScore !== bestMove.score)
140
+ bestMove = { move, score };
141
+ if (bestMove.score > beta)
142
+ break;
143
+ // eslint-disable-next-line no-param-reassign
144
+ alpha = Math.max(alpha, bestScore);
145
+ }
146
+ else {
147
+ const bestScore = Math.min(score, bestMove.score);
148
+ if (bestScore !== bestMove.score)
149
+ bestMove = { move, score };
150
+ if (bestMove.score < alpha)
151
+ break;
152
+ // eslint-disable-next-line no-param-reassign
153
+ beta = Math.min(beta, bestScore);
154
+ }
155
+ }
156
+ return bestMove;
157
+ }
158
+ /**
159
+ * Changes which player's turn it is.
160
+ */
161
+ nextTurn() {
162
+ this.currentPlayerId = (this.currentPlayerId + 1) % this.players.length;
163
+ }
164
+ /**
165
+ * Makes a move.
166
+ *
167
+ * @param input Either the algorithm to use to calculate the move or the move itself.
168
+ */
169
+ async makeMove(input) {
170
+ if (typeof input === "string") {
171
+ this.board.makeMove(this.determineCPUMove(this.currentPlayer.playerType, input), this.currentPlayer.id);
172
+ }
173
+ else {
174
+ if (this.currentPlayer.playerType !== "human" || !this.board.moveIsValid(input))
175
+ return void this.emit("invalidInput", input);
176
+ this.board.makeMove(input, this.currentPlayer.id);
177
+ }
178
+ await this.render(this);
179
+ const { winner } = this.board;
180
+ if (winner !== false)
181
+ return void this.emit("end", winner);
182
+ this.nextTurn();
183
+ }
184
+ }
185
+ //# 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 interface GameConstructorOptions {\n    id?: string;\n    onEnd?: (winner: number | null) => Promise<void> | void;\n    onInvalidInput?: (position: Position) => Promise<void> | void;\n    renderer?: (controller: Controller) => Promise<void> | void;\n}\nexport interface GameConstructor {\n    // eslint-disable-next-line @typescript-eslint/prefer-function-type\n    new (playerOneType: PlayerType, playerTwoType: PlayerType, options?: GameConstructorOptions): Controller;\n}\nexport function Game(constructor: GameConstructor): void {\n    void constructor;\n}\n\n\n/**\n * Represents a game controller.\n */\nexport default abstract class Controller extends EventEmitter<{\n    end: GameConstructorOptions[\"onEnd\"];\n    input: GameConstructorOptions[\"onInvalidInput\"];\n    invalidInput: GameConstructorOptions[\"onInvalidInput\"];\n}> {\n    /**\n     * Contains the ID of the game.\n     */\n    public readonly gameID: string;\n\n    /**\n     * Contains the board.\n     */\n    public readonly board: Board;\n\n    /**\n     * Contains the view object.\n     */\n    public readonly render: Required<GameConstructorOptions>[\"renderer\"];\n\n    /**\n     * Contains the player objects.\n     */\n    protected readonly players: Array<{ id: number; playerType: PlayerType; }>;\n\n    /**\n     * Contains the ID of the current player.\n     */\n    private currentPlayerId: number = 0;\n\n    /**\n     * Creates an instance of Controller.\n     *\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\n    protected constructor(\n        playerTypes: PlayerType[],\n        board: Board,\n        render: Controller[\"render\"],\n        gameID: string | undefined,\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        if (onEnd !== undefined)\n            this.on(\"end\", onEnd);\n        if (onInvalidInput !== undefined)\n            this.on(\"invalidInput\", onInvalidInput);\n    }\n\n    /**\n     * Gets 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     *\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            if (this.currentPlayer.playerType !== \"human\")\n                await this.makeMove(algorithm);\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     *\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        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        let bestMove: ReturnType<typeof this.minimax> = {\n            move: { x: NaN, y: NaN },\n            score: maximisingPlayer ? -Infinity : Infinity\n        };\n        const { emptyCells } = this.board;\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            this.board.undoLastMove();\n            if (maximisingPlayer) {\n                const bestScore = Math.max(score, bestMove.score);\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n            } else {\n                const bestScore = Math.min(score, bestMove.score);\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n            }\n        }\n        return bestMove;\n    }\n\n    /**\n     * The minimax algorithm with alpha-beta pruning.\n     *\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        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        let bestMove: ReturnType<typeof this.alphabeta> = {\n            move: { x: NaN, y: NaN },\n            score: maximisingPlayer ? -Infinity : Infinity\n        };\n        const { emptyCells } = this.board;\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            this.board.undoLastMove();\n            if (maximisingPlayer) {\n                const bestScore = Math.max(score, bestMove.score);\n                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n                if (bestMove.score > beta)\n                    break;\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                if (bestScore !== bestMove.score)\n                    bestMove = { move, score };\n                if (bestMove.score < alpha)\n                    break;\n                // eslint-disable-next-line no-param-reassign\n                beta = Math.min(beta, bestScore);\n            }\n        }\n        return bestMove;\n    }\n\n    /**\n     * Changes which player's turn it is.\n     */\n    private nextTurn(): void {\n        this.currentPlayerId = (this.currentPlayerId + 1) % this.players.length;\n    }\n\n    /**\n     * Makes a move.\n     *\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                return void this.emit(\"invalidInput\", input);\n            this.board.makeMove(input, this.currentPlayer.id);\n        }\n        await this.render(this);\n        const { winner } = this.board;\n        if (winner !== false)\n            return void this.emit(\"end\", winner);\n        this.nextTurn();\n    }\n\n    /**\n     * Determines the CPU move.\n     *\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     *\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"]}
@@ -0,0 +1,133 @@
1
+ import type LongInt from "./longInt.js";
2
+ import type { StringType } from "./longInt.js";
3
+ /**
4
+ * Represents a BitBoard.
5
+ *
6
+ * @link https://en.wikipedia.org/wiki/Bitboard
7
+ */
8
+ export default abstract class BitBoard<T extends LongInt | number = LongInt | number> {
9
+ /**
10
+ * The numeric data.
11
+ */
12
+ protected _data: T;
13
+ /**
14
+ * Creates an instance of BitBoard.
15
+ *
16
+ * @param data The data to assign to the BitBoard.
17
+ */
18
+ protected constructor(data: T);
19
+ /**
20
+ * Gets the numeric data.
21
+ */
22
+ get data(): T;
23
+ /**
24
+ * Returns a string representation of the BitBoard.
25
+ *
26
+ * @param type The base of the string to print.
27
+ * @returns The string representation.
28
+ */
29
+ toString(type?: StringType): string;
30
+ /**
31
+ * Assigns a given bit a given value.
32
+ *
33
+ * @param bit The x coordinate.
34
+ * @param value The value to assign to the bit.
35
+ */
36
+ assignBit(bit: number, value: 0 | 1): void;
37
+ /**
38
+ * Gets a given bit, based on it's x, y coordinates.
39
+ *
40
+ * @param bit The index of the bit to get, 0 = LSB.
41
+ * @returns The bit.
42
+ */
43
+ abstract getBit(bit: number): 0 | 1;
44
+ /**
45
+ * Sets a given bit (changes the value to 1), based on it's x, y coordinates.
46
+ *
47
+ * @param bit The index of the bit to get, 0 = LSB.
48
+ */
49
+ abstract setBit(bit: number): void;
50
+ /**
51
+ * Clears a given bit (changes the value to 0), based on it's x, y coordinates.
52
+ *
53
+ * @param bit The index of the bit to get, 0 = LSB.
54
+ */
55
+ abstract clearBit(bit: number): void;
56
+ /**
57
+ * Toggles a given bit (changes the value from 0 to 1 or 1 to 0), based on it's x, y coordinates.
58
+ *
59
+ * @param bit The index of the bit to get, 0 = LSB.
60
+ */
61
+ abstract toggleBit(bit: number): void;
62
+ /**
63
+ * Clears the whole BitBoard (sets all values to 0).
64
+ */
65
+ abstract clearAll(): void;
66
+ /**
67
+ * Set the whole BitBoard (sets all values to 1).
68
+ */
69
+ abstract setAll(): void;
70
+ /**
71
+ * Gets a range of bits.
72
+ *
73
+ * @param LSB The least significant bit.
74
+ * @param numberOfBits The number of bits to get.
75
+ * @returns The range of bits.
76
+ */
77
+ abstract getBits(LSB: number, numberOfBits: number): T;
78
+ /**
79
+ * Carries out a bitwise and (&) operation.
80
+ *
81
+ * @param right The right value.
82
+ * @returns The result.
83
+ */
84
+ abstract and(right: BitBoard<T> | T | number): this;
85
+ /**
86
+ * Carries out a bitwise or (|) operation.
87
+ *
88
+ * @param right The right value.
89
+ * @returns The result.
90
+ */
91
+ abstract or(right: BitBoard<T> | T | number): this;
92
+ /**
93
+ * Carries out a bitwise xor (^) operation.
94
+ *
95
+ * @param right The right value.
96
+ * @returns The result.
97
+ */
98
+ abstract xor(right: BitBoard<T> | T | number): this;
99
+ /**
100
+ * Carries out a bitwise not (~) operation.
101
+ *
102
+ * @returns The result.
103
+ */
104
+ abstract not(): this;
105
+ /**
106
+ * Carries out a bitwise left shift (<<) operation.
107
+ *
108
+ * @param shiftAmount How much to shift it by.
109
+ * @returns The result.
110
+ */
111
+ abstract leftShift(shiftAmount: number): this;
112
+ /**
113
+ * Carries out a bitwise logical right shift (>>>) operation.
114
+ *
115
+ * @param shiftAmount How much to shift it by.
116
+ * @returns The result.
117
+ */
118
+ abstract rightShift(shiftAmount: number): this;
119
+ /**
120
+ * Carries out a bitwise arithmetic right shift (>>) operation.
121
+ *
122
+ * @param shiftAmount How much to shift it by.
123
+ * @returns The result.
124
+ */
125
+ abstract arithmeticRightShift(shiftAmount: number): this;
126
+ /**
127
+ * Checks if two BitBoards have equal data values.
128
+ *
129
+ * @param value The value to compare against.
130
+ * @returns Whether or not the two BitBoard have the same data value.
131
+ */
132
+ abstract equals(value: BitBoard<T> | T | number): boolean;
133
+ }
@@ -0,0 +1,56 @@
1
+ /**
2
+ * Represents a BitBoard.
3
+ *
4
+ * @link https://en.wikipedia.org/wiki/Bitboard
5
+ */
6
+ export default class BitBoard {
7
+ /**
8
+ * The numeric data.
9
+ */
10
+ _data;
11
+ /**
12
+ * Creates an instance of BitBoard.
13
+ *
14
+ * @param data The data to assign to the BitBoard.
15
+ */
16
+ constructor(data) {
17
+ this._data = data;
18
+ }
19
+ /**
20
+ * Gets the numeric data.
21
+ */
22
+ get data() {
23
+ return this._data;
24
+ }
25
+ /**
26
+ * Returns a string representation of the BitBoard.
27
+ *
28
+ * @param type The base of the string to print.
29
+ * @returns The string representation.
30
+ */
31
+ toString(type = 16) {
32
+ let padLength = 0;
33
+ switch (type) {
34
+ case 2:
35
+ padLength = 32;
36
+ break;
37
+ case 10:
38
+ padLength = 10;
39
+ break;
40
+ case 16:
41
+ padLength = 8;
42
+ break;
43
+ }
44
+ return this._data.toString(type).padStart(padLength, "0");
45
+ }
46
+ /**
47
+ * Assigns a given bit a given value.
48
+ *
49
+ * @param bit The x coordinate.
50
+ * @param value The value to assign to the bit.
51
+ */
52
+ assignBit(bit, value) {
53
+ return value === 0 ? this.clearBit(bit) : this.setBit(bit);
54
+ }
55
+ }
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYml0Qm9hcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYml0Qm9hcmQvYml0Qm9hcmQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBR0E7Ozs7R0FJRztBQUNILE1BQU0sQ0FBQyxPQUFPLE9BQWdCLFFBQVE7SUFDbEM7O09BRUc7SUFDTyxLQUFLLENBQUk7SUFFbkI7Ozs7T0FJRztJQUNILFlBQXNCLElBQU87UUFDekIsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUM7SUFDdEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsSUFBVyxJQUFJO1FBQ1gsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDO0lBQ3RCLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNJLFFBQVEsQ0FBQyxPQUFtQixFQUFFO1FBQ2pDLElBQUksU0FBUyxHQUFHLENBQUMsQ0FBQztRQUNsQixRQUFRLElBQUksRUFBRSxDQUFDO1lBQ1gsS0FBSyxDQUFDO2dCQUNGLFNBQVMsR0FBRyxFQUFFLENBQUM7Z0JBQ2YsTUFBTTtZQUNWLEtBQUssRUFBRTtnQkFDSCxTQUFTLEdBQUcsRUFBRSxDQUFDO2dCQUNmLE1BQU07WUFDVixLQUFLLEVBQUU7Z0JBQ0gsU0FBUyxHQUFHLENBQUMsQ0FBQztnQkFDZCxNQUFNO1FBQ2QsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUMsUUFBUSxDQUFDLFNBQVMsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUM5RCxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSSxTQUFTLENBQUMsR0FBVyxFQUFFLEtBQVk7UUFDdEMsT0FBTyxLQUFLLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQy9ELENBQUM7Q0FnSEoiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSBMb25nSW50IGZyb20gXCIuL2xvbmdJbnQuanNcIjtcbmltcG9ydCB0eXBlIHsgU3RyaW5nVHlwZSB9IGZyb20gXCIuL2xvbmdJbnQuanNcIjtcblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgQml0Qm9hcmQuXG4gKlxuICogQGxpbmsgaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvQml0Ym9hcmRcbiAqL1xuZXhwb3J0IGRlZmF1bHQgYWJzdHJhY3QgY2xhc3MgQml0Qm9hcmQ8VCBleHRlbmRzIExvbmdJbnQgfCBudW1iZXIgPSBMb25nSW50IHwgbnVtYmVyPiB7XG4gICAgLyoqXG4gICAgICogVGhlIG51bWVyaWMgZGF0YS5cbiAgICAgKi9cbiAgICBwcm90ZWN0ZWQgX2RhdGE6IFQ7XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGFuIGluc3RhbmNlIG9mIEJpdEJvYXJkLlxuICAgICAqXG4gICAgICogQHBhcmFtIGRhdGEgVGhlIGRhdGEgdG8gYXNzaWduIHRvIHRoZSBCaXRCb2FyZC5cbiAgICAgKi9cbiAgICBwcm90ZWN0ZWQgY29uc3RydWN0b3IoZGF0YTogVCkge1xuICAgICAgICB0aGlzLl9kYXRhID0gZGF0YTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBHZXRzIHRoZSBudW1lcmljIGRhdGEuXG4gICAgICovXG4gICAgcHVibGljIGdldCBkYXRhKCk6IFQge1xuICAgICAgICByZXR1cm4gdGhpcy5fZGF0YTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBSZXR1cm5zIGEgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBCaXRCb2FyZC5cbiAgICAgKlxuICAgICAqIEBwYXJhbSB0eXBlIFRoZSBiYXNlIG9mIHRoZSBzdHJpbmcgdG8gcHJpbnQuXG4gICAgICogQHJldHVybnMgVGhlIHN0cmluZyByZXByZXNlbnRhdGlvbi5cbiAgICAgKi9cbiAgICBwdWJsaWMgdG9TdHJpbmcodHlwZTogU3RyaW5nVHlwZSA9IDE2KTogc3RyaW5nIHtcbiAgICAgICAgbGV0IHBhZExlbmd0aCA9IDA7XG4gICAgICAgIHN3aXRjaCAodHlwZSkge1xuICAgICAgICAgICAgY2FzZSAyOlxuICAgICAgICAgICAgICAgIHBhZExlbmd0aCA9IDMyO1xuICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSAxMDpcbiAgICAgICAgICAgICAgICBwYWRMZW5ndGggPSAxMDtcbiAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIGNhc2UgMTY6XG4gICAgICAgICAgICAgICAgcGFkTGVuZ3RoID0gODtcbiAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gdGhpcy5fZGF0YS50b1N0cmluZyh0eXBlKS5wYWRTdGFydChwYWRMZW5ndGgsIFwiMFwiKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBBc3NpZ25zIGEgZ2l2ZW4gYml0IGEgZ2l2ZW4gdmFsdWUuXG4gICAgICpcbiAgICAgKiBAcGFyYW0gYml0IFRoZSB4IGNvb3JkaW5hdGUuXG4gICAgICogQHBhcmFtIHZhbHVlIFRoZSB2YWx1ZSB0byBhc3NpZ24gdG8gdGhlIGJpdC5cbiAgICAgKi9cbiAgICBwdWJsaWMgYXNzaWduQml0KGJpdDogbnVtYmVyLCB2YWx1ZTogMCB8IDEpOiB2b2lkIHtcbiAgICAgICAgcmV0dXJuIHZhbHVlID09PSAwID8gdGhpcy5jbGVhckJpdChiaXQpIDogdGhpcy5zZXRCaXQoYml0KTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBHZXRzIGEgZ2l2ZW4gYml0LCBiYXNlZCBvbiBpdCdzIHgsIHkgY29vcmRpbmF0ZXMuXG4gICAgICpcbiAgICAgKiBAcGFyYW0gYml0IFRoZSBpbmRleCBvZiB0aGUgYml0IHRvIGdldCwgMCA9IExTQi5cbiAgICAgKiBAcmV0dXJucyBUaGUgYml0LlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCBnZXRCaXQoYml0OiBudW1iZXIpOiAwIHwgMTtcblxuICAgIC8qKlxuICAgICAqIFNldHMgYSBnaXZlbiBiaXQgKGNoYW5nZXMgdGhlIHZhbHVlIHRvIDEpLCBiYXNlZCBvbiBpdCdzIHgsIHkgY29vcmRpbmF0ZXMuXG4gICAgICpcbiAgICAgKiBAcGFyYW0gYml0IFRoZSBpbmRleCBvZiB0aGUgYml0IHRvIGdldCwgMCA9IExTQi5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3Qgc2V0Qml0KGJpdDogbnVtYmVyKTogdm9pZDtcblxuICAgIC8qKlxuICAgICAqIENsZWFycyBhIGdpdmVuIGJpdCAoY2hhbmdlcyB0aGUgdmFsdWUgdG8gMCksIGJhc2VkIG9uIGl0J3MgeCwgeSBjb29yZGluYXRlcy5cbiAgICAgKlxuICAgICAqIEBwYXJhbSBiaXQgVGhlIGluZGV4IG9mIHRoZSBiaXQgdG8gZ2V0LCAwID0gTFNCLlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCBjbGVhckJpdChiaXQ6IG51bWJlcik6IHZvaWQ7XG5cbiAgICAvKipcbiAgICAgKiBUb2dnbGVzIGEgZ2l2ZW4gYml0IChjaGFuZ2VzIHRoZSB2YWx1ZSBmcm9tIDAgdG8gMSBvciAxIHRvIDApLCBiYXNlZCBvbiBpdCdzIHgsIHkgY29vcmRpbmF0ZXMuXG4gICAgICpcbiAgICAgKiBAcGFyYW0gYml0IFRoZSBpbmRleCBvZiB0aGUgYml0IHRvIGdldCwgMCA9IExTQi5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgdG9nZ2xlQml0KGJpdDogbnVtYmVyKTogdm9pZDtcblxuICAgIC8qKlxuICAgICAqIENsZWFycyB0aGUgd2hvbGUgQml0Qm9hcmQgKHNldHMgYWxsIHZhbHVlcyB0byAwKS5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgY2xlYXJBbGwoKTogdm9pZDtcblxuICAgIC8qKlxuICAgICAqIFNldCB0aGUgd2hvbGUgQml0Qm9hcmQgKHNldHMgYWxsIHZhbHVlcyB0byAxKS5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3Qgc2V0QWxsKCk6IHZvaWQ7XG5cbiAgICAvKipcbiAgICAgKiBHZXRzIGEgcmFuZ2Ugb2YgYml0cy5cbiAgICAgKlxuICAgICAqIEBwYXJhbSBMU0IgVGhlIGxlYXN0IHNpZ25pZmljYW50IGJpdC5cbiAgICAgKiBAcGFyYW0gbnVtYmVyT2ZCaXRzIFRoZSBudW1iZXIgb2YgYml0cyB0byBnZXQuXG4gICAgICogQHJldHVybnMgVGhlIHJhbmdlIG9mIGJpdHMuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IGdldEJpdHMoTFNCOiBudW1iZXIsIG51bWJlck9mQml0czogbnVtYmVyKTogVDtcblxuICAgIC8qKlxuICAgICAqIENhcnJpZXMgb3V0IGEgYml0d2lzZSBhbmQgKCYpIG9wZXJhdGlvbi5cbiAgICAgKlxuICAgICAqIEBwYXJhbSByaWdodCBUaGUgcmlnaHQgdmFsdWUuXG4gICAgICogQHJldHVybnMgVGhlIHJlc3VsdC5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgYW5kKHJpZ2h0OiBCaXRCb2FyZDxUPiB8IFQgfCBudW1iZXIpOiB0aGlzO1xuXG4gICAgLyoqXG4gICAgICogQ2FycmllcyBvdXQgYSBiaXR3aXNlIG9yICh8KSBvcGVyYXRpb24uXG4gICAgICpcbiAgICAgKiBAcGFyYW0gcmlnaHQgVGhlIHJpZ2h0IHZhbHVlLlxuICAgICAqIEByZXR1cm5zIFRoZSByZXN1bHQuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IG9yKHJpZ2h0OiBCaXRCb2FyZDxUPiB8IFQgfCBudW1iZXIpOiB0aGlzO1xuXG4gICAgLyoqXG4gICAgICogQ2FycmllcyBvdXQgYSBiaXR3aXNlIHhvciAoXikgb3BlcmF0aW9uLlxuICAgICAqXG4gICAgICogQHBhcmFtIHJpZ2h0IFRoZSByaWdodCB2YWx1ZS5cbiAgICAgKiBAcmV0dXJucyBUaGUgcmVzdWx0LlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCB4b3IocmlnaHQ6IEJpdEJvYXJkPFQ+IHwgVCB8IG51bWJlcik6IHRoaXM7XG5cbiAgICAvKipcbiAgICAgKiBDYXJyaWVzIG91dCBhIGJpdHdpc2Ugbm90ICh+KSBvcGVyYXRpb24uXG4gICAgICpcbiAgICAgKiBAcmV0dXJucyBUaGUgcmVzdWx0LlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCBub3QoKTogdGhpcztcblxuICAgIC8qKlxuICAgICAqIENhcnJpZXMgb3V0IGEgYml0d2lzZSBsZWZ0IHNoaWZ0ICg8PCkgb3BlcmF0aW9uLlxuICAgICAqXG4gICAgICogQHBhcmFtIHNoaWZ0QW1vdW50IEhvdyBtdWNoIHRvIHNoaWZ0IGl0IGJ5LlxuICAgICAqIEByZXR1cm5zIFRoZSByZXN1bHQuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IGxlZnRTaGlmdChzaGlmdEFtb3VudDogbnVtYmVyKTogdGhpcztcblxuICAgIC8qKlxuICAgICAqIENhcnJpZXMgb3V0IGEgYml0d2lzZSBsb2dpY2FsIHJpZ2h0IHNoaWZ0ICg+Pj4pIG9wZXJhdGlvbi5cbiAgICAgKlxuICAgICAqIEBwYXJhbSBzaGlmdEFtb3VudCBIb3cgbXVjaCB0byBzaGlmdCBpdCBieS5cbiAgICAgKiBAcmV0dXJucyBUaGUgcmVzdWx0LlxuICAgICAqL1xuICAgIHB1YmxpYyBhYnN0cmFjdCByaWdodFNoaWZ0KHNoaWZ0QW1vdW50OiBudW1iZXIpOiB0aGlzO1xuXG4gICAgLyoqXG4gICAgICogQ2FycmllcyBvdXQgYSBiaXR3aXNlIGFyaXRobWV0aWMgcmlnaHQgc2hpZnQgKD4+KSBvcGVyYXRpb24uXG4gICAgICpcbiAgICAgKiBAcGFyYW0gc2hpZnRBbW91bnQgSG93IG11Y2ggdG8gc2hpZnQgaXQgYnkuXG4gICAgICogQHJldHVybnMgVGhlIHJlc3VsdC5cbiAgICAgKi9cbiAgICBwdWJsaWMgYWJzdHJhY3QgYXJpdGhtZXRpY1JpZ2h0U2hpZnQoc2hpZnRBbW91bnQ6IG51bWJlcik6IHRoaXM7XG5cbiAgICAvKipcbiAgICAgKiBDaGVja3MgaWYgdHdvIEJpdEJvYXJkcyBoYXZlIGVxdWFsIGRhdGEgdmFsdWVzLlxuICAgICAqXG4gICAgICogQHBhcmFtIHZhbHVlIFRoZSB2YWx1ZSB0byBjb21wYXJlIGFnYWluc3QuXG4gICAgICogQHJldHVybnMgV2hldGhlciBvciBub3QgdGhlIHR3byBCaXRCb2FyZCBoYXZlIHRoZSBzYW1lIGRhdGEgdmFsdWUuXG4gICAgICovXG4gICAgcHVibGljIGFic3RyYWN0IGVxdWFscyh2YWx1ZTogQml0Qm9hcmQ8VD4gfCBUIHwgbnVtYmVyKTogYm9vbGVhbjtcbn1cbiJdfQ==
@@ -0,0 +1,27 @@
1
+ import BitBoard from "./bitBoard.js";
2
+ /**
3
+ * Represents a BitBoard which uses just one 32-bit number.
4
+ */
5
+ export default class IntBitBoard extends BitBoard<number> {
6
+ /**
7
+ * Creates an instance of NumberBitBoard.
8
+ *
9
+ * @param data The data to fill the BitBoard with.
10
+ */
11
+ constructor(data?: number);
12
+ getBit(bit: number): 0 | 1;
13
+ setBit(bit: number): void;
14
+ clearBit(bit: number): void;
15
+ toggleBit(bit: number): void;
16
+ clearAll(): void;
17
+ setAll(): void;
18
+ getBits(LSB: number, numberOfBits: number): number;
19
+ and(right: IntBitBoard | number): this;
20
+ or(right: IntBitBoard | number): this;
21
+ xor(right: IntBitBoard | number): this;
22
+ not(): this;
23
+ leftShift(shiftAmount: number): this;
24
+ rightShift(shiftAmount: number): this;
25
+ arithmeticRightShift(shiftAmount: number): this;
26
+ equals(value: IntBitBoard | number): boolean;
27
+ }
@@ -0,0 +1,65 @@
1
+ import BitBoard from "./bitBoard.js";
2
+ /**
3
+ * Represents a BitBoard which uses just one 32-bit number.
4
+ */
5
+ export default class IntBitBoard extends BitBoard {
6
+ /**
7
+ * Creates an instance of NumberBitBoard.
8
+ *
9
+ * @param data The data to fill the BitBoard with.
10
+ */
11
+ constructor(data = 0) {
12
+ super(data);
13
+ }
14
+ getBit(bit) {
15
+ return (this._data >>> bit & 1);
16
+ }
17
+ setBit(bit) {
18
+ const mask = 1 << bit;
19
+ this._data |= mask;
20
+ }
21
+ clearBit(bit) {
22
+ const mask = ~(1 << bit);
23
+ this._data &= mask;
24
+ }
25
+ toggleBit(bit) {
26
+ const mask = 1 << bit;
27
+ this._data ^= mask;
28
+ }
29
+ clearAll() {
30
+ this._data = 0;
31
+ }
32
+ setAll() {
33
+ this._data = ~0 >>> 0;
34
+ }
35
+ getBits(LSB, numberOfBits) {
36
+ const mask = 2 ** numberOfBits - 1 << LSB;
37
+ const bits = (this._data & mask) >>> LSB;
38
+ return bits;
39
+ }
40
+ and(right) {
41
+ return new IntBitBoard(this._data & (right instanceof IntBitBoard ? right.data : right));
42
+ }
43
+ or(right) {
44
+ return new IntBitBoard(this._data | (right instanceof IntBitBoard ? right.data : right));
45
+ }
46
+ xor(right) {
47
+ return new IntBitBoard(this._data ^ (right instanceof IntBitBoard ? right.data : right));
48
+ }
49
+ not() {
50
+ return new IntBitBoard(~this._data);
51
+ }
52
+ leftShift(shiftAmount) {
53
+ return new IntBitBoard(this._data << shiftAmount);
54
+ }
55
+ rightShift(shiftAmount) {
56
+ return new IntBitBoard(this._data >>> shiftAmount);
57
+ }
58
+ arithmeticRightShift(shiftAmount) {
59
+ return new IntBitBoard(this._data >> shiftAmount);
60
+ }
61
+ equals(value) {
62
+ return this._data === (value instanceof IntBitBoard ? value._data : value);
63
+ }
64
+ }
65
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50Qml0Qm9hcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYml0Qm9hcmQvaW50Qml0Qm9hcmQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxRQUFRLE1BQU0sZUFBZSxDQUFDO0FBRXJDOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE9BQU8sT0FBTyxXQUFZLFNBQVEsUUFBZ0I7SUFDckQ7Ozs7T0FJRztJQUNILFlBQW1CLE9BQWUsQ0FBQztRQUMvQixLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDaEIsQ0FBQztJQUVNLE1BQU0sQ0FBQyxHQUFXO1FBQ3JCLE9BQU8sQ0FBQyxJQUFJLENBQUMsS0FBSyxLQUFLLEdBQUcsR0FBRyxDQUFDLENBQVUsQ0FBQztJQUM3QyxDQUFDO0lBRU0sTUFBTSxDQUFDLEdBQVc7UUFDckIsTUFBTSxJQUFJLEdBQUcsQ0FBQyxJQUFJLEdBQUcsQ0FBQztRQUN0QixJQUFJLENBQUMsS0FBSyxJQUFJLElBQUksQ0FBQztJQUN2QixDQUFDO0lBRU0sUUFBUSxDQUFDLEdBQVc7UUFDdkIsTUFBTSxJQUFJLEdBQUcsQ0FBQyxDQUFDLENBQUMsSUFBSSxHQUFHLENBQUMsQ0FBQztRQUN6QixJQUFJLENBQUMsS0FBSyxJQUFJLElBQUksQ0FBQztJQUN2QixDQUFDO0lBRU0sU0FBUyxDQUFDLEdBQVc7UUFDeEIsTUFBTSxJQUFJLEdBQUcsQ0FBQyxJQUFJLEdBQUcsQ0FBQztRQUN0QixJQUFJLENBQUMsS0FBSyxJQUFJLElBQUksQ0FBQztJQUN2QixDQUFDO0lBRU0sUUFBUTtRQUNYLElBQUksQ0FBQyxLQUFLLEdBQUcsQ0FBQyxDQUFDO0lBQ25CLENBQUM7SUFFTSxNQUFNO1FBQ1QsSUFBSSxDQUFDLEtBQUssR0FBRyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDMUIsQ0FBQztJQUVNLE9BQU8sQ0FBQyxHQUFXLEVBQUUsWUFBb0I7UUFDNUMsTUFBTSxJQUFJLEdBQUcsQ0FBQyxJQUFJLFlBQVksR0FBRyxDQUFDLElBQUksR0FBRyxDQUFDO1FBQzFDLE1BQU0sSUFBSSxHQUFHLENBQUMsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxHQUFHLENBQUM7UUFDekMsT0FBTyxJQUFJLENBQUM7SUFDaEIsQ0FBQztJQUVNLEdBQUcsQ0FBQyxLQUEyQjtRQUNsQyxPQUFPLElBQUksV0FBVyxDQUFDLElBQUksQ0FBQyxLQUFLLEdBQUcsQ0FBQyxLQUFLLFlBQVksV0FBVyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBUyxDQUFDO0lBQ3JHLENBQUM7SUFFTSxFQUFFLENBQUMsS0FBMkI7UUFDakMsT0FBTyxJQUFJLFdBQVcsQ0FBQyxJQUFJLENBQUMsS0FBSyxHQUFHLENBQUMsS0FBSyxZQUFZLFdBQVcsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQVMsQ0FBQztJQUNyRyxDQUFDO0lBRU0sR0FBRyxDQUFDLEtBQTJCO1FBQ2xDLE9BQU8sSUFBSSxXQUFXLENBQUMsSUFBSSxDQUFDLEtBQUssR0FBRyxDQUFDLEtBQUssWUFBWSxXQUFXLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFTLENBQUM7SUFDckcsQ0FBQztJQUVNLEdBQUc7UUFDTixPQUFPLElBQUksV0FBVyxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBUyxDQUFDO0lBQ2hELENBQUM7SUFFTSxTQUFTLENBQUMsV0FBbUI7UUFDaEMsT0FBTyxJQUFJLFdBQVcsQ0FBQyxJQUFJLENBQUMsS0FBSyxJQUFJLFdBQVcsQ0FBUyxDQUFDO0lBQzlELENBQUM7SUFFTSxVQUFVLENBQUMsV0FBbUI7UUFDakMsT0FBTyxJQUFJLFdBQVcsQ0FBQyxJQUFJLENBQUMsS0FBSyxLQUFLLFdBQVcsQ0FBUyxDQUFDO0lBQy9ELENBQUM7SUFFTSxvQkFBb0IsQ0FBQyxXQUFtQjtRQUMzQyxPQUFPLElBQUksV0FBVyxDQUFDLElBQUksQ0FBQyxLQUFLLElBQUksV0FBVyxDQUFTLENBQUM7SUFDOUQsQ0FBQztJQUVNLE1BQU0sQ0FBQyxLQUEyQjtRQUNyQyxPQUFPLElBQUksQ0FBQyxLQUFLLEtBQUssQ0FBQyxLQUFLLFlBQVksV0FBVyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMvRSxDQUFDO0NBQ0oiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQml0Qm9hcmQgZnJvbSBcIi4vYml0Qm9hcmQuanNcIjtcblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgQml0Qm9hcmQgd2hpY2ggdXNlcyBqdXN0IG9uZSAzMi1iaXQgbnVtYmVyLlxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBJbnRCaXRCb2FyZCBleHRlbmRzIEJpdEJvYXJkPG51bWJlcj4ge1xuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYW4gaW5zdGFuY2Ugb2YgTnVtYmVyQml0Qm9hcmQuXG4gICAgICpcbiAgICAgKiBAcGFyYW0gZGF0YSBUaGUgZGF0YSB0byBmaWxsIHRoZSBCaXRCb2FyZCB3aXRoLlxuICAgICAqL1xuICAgIHB1YmxpYyBjb25zdHJ1Y3RvcihkYXRhOiBudW1iZXIgPSAwKSB7XG4gICAgICAgIHN1cGVyKGRhdGEpO1xuICAgIH1cblxuICAgIHB1YmxpYyBnZXRCaXQoYml0OiBudW1iZXIpOiAwIHwgMSB7XG4gICAgICAgIHJldHVybiAodGhpcy5fZGF0YSA+Pj4gYml0ICYgMSkgYXMgMCB8IDE7XG4gICAgfVxuXG4gICAgcHVibGljIHNldEJpdChiaXQ6IG51bWJlcik6IHZvaWQge1xuICAgICAgICBjb25zdCBtYXNrID0gMSA8PCBiaXQ7XG4gICAgICAgIHRoaXMuX2RhdGEgfD0gbWFzaztcbiAgICB9XG5cbiAgICBwdWJsaWMgY2xlYXJCaXQoYml0OiBudW1iZXIpOiB2b2lkIHtcbiAgICAgICAgY29uc3QgbWFzayA9IH4oMSA8PCBiaXQpO1xuICAgICAgICB0aGlzLl9kYXRhICY9IG1hc2s7XG4gICAgfVxuXG4gICAgcHVibGljIHRvZ2dsZUJpdChiaXQ6IG51bWJlcik6IHZvaWQge1xuICAgICAgICBjb25zdCBtYXNrID0gMSA8PCBiaXQ7XG4gICAgICAgIHRoaXMuX2RhdGEgXj0gbWFzaztcbiAgICB9XG5cbiAgICBwdWJsaWMgY2xlYXJBbGwoKTogdm9pZCB7XG4gICAgICAgIHRoaXMuX2RhdGEgPSAwO1xuICAgIH1cblxuICAgIHB1YmxpYyBzZXRBbGwoKTogdm9pZCB7XG4gICAgICAgIHRoaXMuX2RhdGEgPSB+MCA+Pj4gMDtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0Qml0cyhMU0I6IG51bWJlciwgbnVtYmVyT2ZCaXRzOiBudW1iZXIpOiBudW1iZXIge1xuICAgICAgICBjb25zdCBtYXNrID0gMiAqKiBudW1iZXJPZkJpdHMgLSAxIDw8IExTQjtcbiAgICAgICAgY29uc3QgYml0cyA9ICh0aGlzLl9kYXRhICYgbWFzaykgPj4+IExTQjtcbiAgICAgICAgcmV0dXJuIGJpdHM7XG4gICAgfVxuXG4gICAgcHVibGljIGFuZChyaWdodDogSW50Qml0Qm9hcmQgfCBudW1iZXIpOiB0aGlzIHtcbiAgICAgICAgcmV0dXJuIG5ldyBJbnRCaXRCb2FyZCh0aGlzLl9kYXRhICYgKHJpZ2h0IGluc3RhbmNlb2YgSW50Qml0Qm9hcmQgPyByaWdodC5kYXRhIDogcmlnaHQpKSBhcyB0aGlzO1xuICAgIH1cblxuICAgIHB1YmxpYyBvcihyaWdodDogSW50Qml0Qm9hcmQgfCBudW1iZXIpOiB0aGlzIHtcbiAgICAgICAgcmV0dXJuIG5ldyBJbnRCaXRCb2FyZCh0aGlzLl9kYXRhIHwgKHJpZ2h0IGluc3RhbmNlb2YgSW50Qml0Qm9hcmQgPyByaWdodC5kYXRhIDogcmlnaHQpKSBhcyB0aGlzO1xuICAgIH1cblxuICAgIHB1YmxpYyB4b3IocmlnaHQ6IEludEJpdEJvYXJkIHwgbnVtYmVyKTogdGhpcyB7XG4gICAgICAgIHJldHVybiBuZXcgSW50Qml0Qm9hcmQodGhpcy5fZGF0YSBeIChyaWdodCBpbnN0YW5jZW9mIEludEJpdEJvYXJkID8gcmlnaHQuZGF0YSA6IHJpZ2h0KSkgYXMgdGhpcztcbiAgICB9XG5cbiAgICBwdWJsaWMgbm90KCk6IHRoaXMge1xuICAgICAgICByZXR1cm4gbmV3IEludEJpdEJvYXJkKH50aGlzLl9kYXRhKSBhcyB0aGlzO1xuICAgIH1cblxuICAgIHB1YmxpYyBsZWZ0U2hpZnQoc2hpZnRBbW91bnQ6IG51bWJlcik6IHRoaXMge1xuICAgICAgICByZXR1cm4gbmV3IEludEJpdEJvYXJkKHRoaXMuX2RhdGEgPDwgc2hpZnRBbW91bnQpIGFzIHRoaXM7XG4gICAgfVxuXG4gICAgcHVibGljIHJpZ2h0U2hpZnQoc2hpZnRBbW91bnQ6IG51bWJlcik6IHRoaXMge1xuICAgICAgICByZXR1cm4gbmV3IEludEJpdEJvYXJkKHRoaXMuX2RhdGEgPj4+IHNoaWZ0QW1vdW50KSBhcyB0aGlzO1xuICAgIH1cblxuICAgIHB1YmxpYyBhcml0aG1ldGljUmlnaHRTaGlmdChzaGlmdEFtb3VudDogbnVtYmVyKTogdGhpcyB7XG4gICAgICAgIHJldHVybiBuZXcgSW50Qml0Qm9hcmQodGhpcy5fZGF0YSA+PiBzaGlmdEFtb3VudCkgYXMgdGhpcztcbiAgICB9XG5cbiAgICBwdWJsaWMgZXF1YWxzKHZhbHVlOiBJbnRCaXRCb2FyZCB8IG51bWJlcik6IGJvb2xlYW4ge1xuICAgICAgICByZXR1cm4gdGhpcy5fZGF0YSA9PT0gKHZhbHVlIGluc3RhbmNlb2YgSW50Qml0Qm9hcmQgPyB2YWx1ZS5fZGF0YSA6IHZhbHVlKTtcbiAgICB9XG59XG4iXX0=