@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,85 @@
1
+ import BitBoard from "./bitBoard.js";
2
+ import LongInt from "./longInt.js";
3
+ /**
4
+ * A BitBoard which uses an array of 32-bit numbers.
5
+ */
6
+ export default class LongIntBitBoard extends BitBoard {
7
+ constructor(args) {
8
+ switch (true) {
9
+ case args === undefined:
10
+ super(new LongInt(Array(2).fill(0)));
11
+ break;
12
+ case args instanceof LongInt:
13
+ super(args);
14
+ break;
15
+ case args instanceof Uint32Array || args instanceof Array:
16
+ super(new LongInt(args));
17
+ break;
18
+ case typeof args === "number":
19
+ super(new LongInt(Array(args).fill(0)));
20
+ break;
21
+ }
22
+ }
23
+ getBit(bit) {
24
+ return LongInt.rightShift(this._data, bit).and(1).data[0];
25
+ }
26
+ setBit(bit) {
27
+ const mask = LongInt.getMatchingLongInt(this._data, 1)
28
+ .leftShift(bit);
29
+ this._data.or(mask);
30
+ }
31
+ clearBit(bit) {
32
+ const mask = LongInt.getMatchingLongInt(this._data, 1)
33
+ .leftShift(bit)
34
+ .not();
35
+ this._data.and(mask);
36
+ }
37
+ toggleBit(bit) {
38
+ const mask = LongInt.getMatchingLongInt(this._data, 1)
39
+ .leftShift(bit);
40
+ this._data.xor(mask);
41
+ }
42
+ clearAll() {
43
+ for (let i = 0; i < this._data.data.length; i++)
44
+ this._data.data[i] = 0;
45
+ }
46
+ setAll() {
47
+ for (let i = 0; i < this._data.data.length; i++)
48
+ this._data.data[i] = ~0 >>> 0;
49
+ }
50
+ getBits(LSB, numberOfBits) {
51
+ const arr = [];
52
+ const arrLength = Math.ceil(numberOfBits / 32);
53
+ for (let i = 0; i < arrLength - 1; i++)
54
+ arr[i] = ~0 >>> 0;
55
+ arr[arrLength - 1] = 2 ** (numberOfBits - (arrLength - 1) * 32) - 1;
56
+ const mask = LongInt.getMatchingLongInt(this._data, arr).leftShift(LSB);
57
+ const bits = LongInt.rightShift(LongInt.and(this._data, mask), LSB);
58
+ return bits;
59
+ }
60
+ and(right) {
61
+ return new LongIntBitBoard(LongInt.and(this._data, right instanceof LongIntBitBoard ? right.data : right));
62
+ }
63
+ or(right) {
64
+ return new LongIntBitBoard(LongInt.or(this._data, right instanceof LongIntBitBoard ? right.data : right));
65
+ }
66
+ xor(right) {
67
+ return new LongIntBitBoard(LongInt.xor(this._data, right instanceof LongIntBitBoard ? right.data : right));
68
+ }
69
+ not() {
70
+ return new LongIntBitBoard(LongInt.not(this._data));
71
+ }
72
+ leftShift(shiftAmount) {
73
+ return new LongIntBitBoard(LongInt.leftShift(this._data, shiftAmount));
74
+ }
75
+ rightShift(shiftAmount) {
76
+ return new LongIntBitBoard(LongInt.rightShift(this._data, shiftAmount));
77
+ }
78
+ arithmeticRightShift(shiftAmount) {
79
+ return new LongIntBitBoard(LongInt.arithmeticRightShift(this._data, shiftAmount));
80
+ }
81
+ equals(value) {
82
+ return this._data.equals(value instanceof LongIntBitBoard ? value._data : value);
83
+ }
84
+ }
85
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"longIntBitBoard.js","sourceRoot":"","sources":["../../src/bitBoard/longIntBitBoard.ts"],"names":[],"mappings":"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","sourcesContent":["import BitBoard from \"./bitBoard.js\";\nimport LongInt from \"./longInt.js\";\n\n/**\n * A BitBoard which uses an array of 32-bit numbers.\n */\nexport default class LongIntBitBoard extends BitBoard<LongInt> {\n    /**\n     * Creates an instance of LongIntBitBoard.\n     *\n     * @param longInt The data to fill the BitBoard with.\n    */\n    public constructor(longInt: LongInt);\n    /**\n     * Creates an instance of LongIntBitBoard.\n     *\n     * @param uint32Array The data to fill the BitBoard with.\n     */\n    public constructor(uint32Array: Uint32Array);\n    /**\n     * Creates an instance of LongIntBitBoard.\n     *\n     * @param numberArray The data to fill the BitBoard with.\n     */\n    public constructor(numberArray: number[]);\n    /**\n     * Creates an instance of LongIntBitBoard.\n     *\n     * @param length The length of the LongInt.\n     */\n    public constructor(length: number);\n    public constructor(args?: LongInt | number[] | Uint32Array | number) {\n        switch (true) {\n            case args === undefined:\n                super(new LongInt(Array(2).fill(0)));\n                break;\n            case args instanceof LongInt:\n                super(args);\n                break;\n            case args instanceof Uint32Array || args instanceof Array:\n                super(new LongInt(args));\n                break;\n            case typeof args === \"number\":\n                super(new LongInt(Array(args).fill(0)));\n                break;\n        }\n    }\n\n    public getBit(bit: number): 0 | 1 {\n        return LongInt.rightShift(this._data, bit).and(1).data[0] as 0 | 1;\n    }\n\n    public setBit(bit: number): void {\n        const mask = LongInt.getMatchingLongInt(this._data, 1)\n            .leftShift(bit);\n        this._data.or(mask);\n    }\n\n    public clearBit(bit: number): void {\n        const mask = LongInt.getMatchingLongInt(this._data, 1)\n            .leftShift(bit)\n            .not();\n        this._data.and(mask);\n    }\n\n    public toggleBit(bit: number): void {\n        const mask = LongInt.getMatchingLongInt(this._data, 1)\n            .leftShift(bit);\n        this._data.xor(mask);\n    }\n\n    public clearAll(): void {\n        for (let i = 0; i < this._data.data.length; i++)\n            this._data.data[i] = 0;\n    }\n\n    public setAll(): void {\n        for (let i = 0; i < this._data.data.length; i++)\n            this._data.data[i] = ~0 >>> 0;\n    }\n\n    public getBits(LSB: number, numberOfBits: number): LongInt {\n        const arr: number[] = [];\n        const arrLength = Math.ceil(numberOfBits / 32);\n        for (let i = 0; i < arrLength - 1; i++)\n            arr[i] = ~0 >>> 0;\n        arr[arrLength - 1] = 2 ** (numberOfBits - (arrLength - 1) * 32) - 1;\n        const mask = LongInt.getMatchingLongInt(this._data, arr).leftShift(LSB);\n        const bits = LongInt.rightShift(LongInt.and(this._data, mask), LSB);\n        return bits;\n    }\n\n    public and(right: LongIntBitBoard | number): this {\n        return new LongIntBitBoard(LongInt.and(this._data, right instanceof LongIntBitBoard ? right.data : right)) as this;\n    }\n\n    public or(right: LongIntBitBoard | number): this {\n        return new LongIntBitBoard(LongInt.or(this._data, right instanceof LongIntBitBoard ? right.data : right)) as this;\n    }\n\n    public xor(right: LongIntBitBoard | number): this {\n        return new LongIntBitBoard(LongInt.xor(this._data, right instanceof LongIntBitBoard ? right.data : right)) as this;\n    }\n\n    public not(): this {\n        return new LongIntBitBoard(LongInt.not(this._data)) as this;\n    }\n\n    public leftShift(shiftAmount: number): this {\n        return new LongIntBitBoard(LongInt.leftShift(this._data, shiftAmount)) as this;\n    }\n\n    public rightShift(shiftAmount: number): this {\n        return new LongIntBitBoard(LongInt.rightShift(this._data, shiftAmount)) as this;\n    }\n\n    public arithmeticRightShift(shiftAmount: number): this {\n        return new LongIntBitBoard(LongInt.arithmeticRightShift(this._data, shiftAmount)) as this;\n    }\n\n    public equals(value: LongInt | LongIntBitBoard | number): boolean {\n        return this._data.equals(value instanceof LongIntBitBoard ? value._data : value);\n    }\n}\n"]}
@@ -0,0 +1,23 @@
1
+ /**
2
+ * Reads from the console.
3
+ *
4
+ * @param prompt The question to get the answer to.
5
+ * @returns The value read.
6
+ */
7
+ declare function readLine(prompt?: string): Promise<string>;
8
+ /**
9
+ * Writes to the console.
10
+ *
11
+ * @param text The text to write.
12
+ */
13
+ declare function writeLine(...text: string[]): void;
14
+ /**
15
+ * Clears the console.
16
+ */
17
+ declare function clear(): void;
18
+ declare const _default: {
19
+ clear: typeof clear;
20
+ readLine: typeof readLine;
21
+ writeLine: typeof writeLine;
22
+ };
23
+ export default _default;
@@ -0,0 +1,30 @@
1
+ /* eslint-disable no-console */
2
+ import readline from "readline/promises";
3
+ /**
4
+ * Reads from the console.
5
+ *
6
+ * @param prompt The question to get the answer to.
7
+ * @returns The value read.
8
+ */
9
+ async function readLine(prompt = "") {
10
+ const reader = readline.createInterface(process.stdin, process.stdout);
11
+ const input = await reader.question(prompt);
12
+ reader.close();
13
+ return input;
14
+ }
15
+ /**
16
+ * Writes to the console.
17
+ *
18
+ * @param text The text to write.
19
+ */
20
+ function writeLine(...text) {
21
+ return console.log(...text);
22
+ }
23
+ /**
24
+ * Clears the console.
25
+ */
26
+ function clear() {
27
+ return console.clear();
28
+ }
29
+ export default { clear, readLine, writeLine };
30
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc29sZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9jb25zb2xlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLCtCQUErQjtBQUMvQixPQUFPLFFBQVEsTUFBTSxtQkFBbUIsQ0FBQztBQUV6Qzs7Ozs7R0FLRztBQUNILEtBQUssVUFBVSxRQUFRLENBQUMsU0FBaUIsRUFBRTtJQUN2QyxNQUFNLE1BQU0sR0FBRyxRQUFRLENBQUMsZUFBZSxDQUFDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ3ZFLE1BQU0sS0FBSyxHQUFHLE1BQU0sTUFBTSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUM1QyxNQUFNLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDZixPQUFPLEtBQUssQ0FBQztBQUNqQixDQUFDO0FBRUQ7Ozs7R0FJRztBQUNILFNBQVMsU0FBUyxDQUFDLEdBQUcsSUFBYztJQUNoQyxPQUFPLE9BQU8sQ0FBQyxHQUFHLENBQUMsR0FBRyxJQUFJLENBQUMsQ0FBQztBQUNoQyxDQUFDO0FBRUQ7O0dBRUc7QUFDSCxTQUFTLEtBQUs7SUFDVixPQUFPLE9BQU8sQ0FBQyxLQUFLLEVBQUUsQ0FBQztBQUMzQixDQUFDO0FBRUQsZUFBZSxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsU0FBUyxFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBuby1jb25zb2xlICovXG5pbXBvcnQgcmVhZGxpbmUgZnJvbSBcInJlYWRsaW5lL3Byb21pc2VzXCI7XG5cbi8qKlxuICogUmVhZHMgZnJvbSB0aGUgY29uc29sZS5cbiAqXG4gKiBAcGFyYW0gcHJvbXB0IFRoZSBxdWVzdGlvbiB0byBnZXQgdGhlIGFuc3dlciB0by5cbiAqIEByZXR1cm5zIFRoZSB2YWx1ZSByZWFkLlxuICovXG5hc3luYyBmdW5jdGlvbiByZWFkTGluZShwcm9tcHQ6IHN0cmluZyA9IFwiXCIpOiBQcm9taXNlPHN0cmluZz4ge1xuICAgIGNvbnN0IHJlYWRlciA9IHJlYWRsaW5lLmNyZWF0ZUludGVyZmFjZShwcm9jZXNzLnN0ZGluLCBwcm9jZXNzLnN0ZG91dCk7XG4gICAgY29uc3QgaW5wdXQgPSBhd2FpdCByZWFkZXIucXVlc3Rpb24ocHJvbXB0KTtcbiAgICByZWFkZXIuY2xvc2UoKTtcbiAgICByZXR1cm4gaW5wdXQ7XG59XG5cbi8qKlxuICogV3JpdGVzIHRvIHRoZSBjb25zb2xlLlxuICpcbiAqIEBwYXJhbSB0ZXh0IFRoZSB0ZXh0IHRvIHdyaXRlLlxuICovXG5mdW5jdGlvbiB3cml0ZUxpbmUoLi4udGV4dDogc3RyaW5nW10pOiB2b2lkIHtcbiAgICByZXR1cm4gY29uc29sZS5sb2coLi4udGV4dCk7XG59XG5cbi8qKlxuICogQ2xlYXJzIHRoZSBjb25zb2xlLlxuICovXG5mdW5jdGlvbiBjbGVhcigpOiB2b2lkIHtcbiAgICByZXR1cm4gY29uc29sZS5jbGVhcigpO1xufVxuXG5leHBvcnQgZGVmYXVsdCB7IGNsZWFyLCByZWFkTGluZSwgd3JpdGVMaW5lIH07XG4iXX0=
@@ -0,0 +1,15 @@
1
+ import Base from "../../base/board.js";
2
+ import LongIntBitBoard from "../../bitBoard/longIntBitBoard.js";
3
+ import type { Position } from "../../base/board.js";
4
+ export default class Board extends Base<LongIntBitBoard> {
5
+ protected winningStates: LongIntBitBoard[];
6
+ private readonly FULL_BOARD;
7
+ private readonly HORIZONTAL;
8
+ private readonly VERTICAL;
9
+ private readonly LEADING_DIAGONAL;
10
+ private readonly NON_LEADING_DIAGONAL;
11
+ constructor();
12
+ get heuristic(): number;
13
+ get emptyCells(): Position[];
14
+ makeMove(move: Position, playerId: number): void;
15
+ }
@@ -0,0 +1,88 @@
1
+ import Base from "../../base/board.js";
2
+ import LongInt from "../../bitBoard/longInt.js";
3
+ import LongIntBitBoard from "../../bitBoard/longIntBitBoard.js";
4
+ export default class Board extends Base {
5
+ winningStates = [];
6
+ FULL_BOARD = new LongInt([0b1111_1111111_1111111_1111111_1111111, 0b1111111_111]);
7
+ HORIZONTAL = new LongInt([0b0000_0000000_0000000_0000000_0001111, 0b0000000_000]);
8
+ VERTICAL = new LongInt([0b0000_0000001_0000001_0000001_0000001, 0b0000000_000]);
9
+ LEADING_DIAGONAL = new LongInt([0b0000_0001000_0000100_0000010_0000001, 0b0000000_000]);
10
+ NON_LEADING_DIAGONAL = new LongInt([0b0000_0000001_0000010_0000100_0001000, 0b0000000_000]);
11
+ constructor() {
12
+ super(7, 6);
13
+ for (let i = 0; i < 4; i++) {
14
+ for (let j = 0; j < 6; j++) {
15
+ this.winningStates.push(new LongIntBitBoard(LongInt
16
+ .leftShift(this.HORIZONTAL, i + 7 * j)
17
+ .and(this.FULL_BOARD)));
18
+ }
19
+ }
20
+ for (let i = 0; i < 7; i++) {
21
+ for (let j = 0; j < 3; j++) {
22
+ this.winningStates.push(new LongIntBitBoard(LongInt
23
+ .leftShift(this.VERTICAL, i + 7 * j)
24
+ .and(this.FULL_BOARD)));
25
+ }
26
+ }
27
+ for (let i = 0; i < 4; i++) {
28
+ for (let j = 0; j < 3; j++) {
29
+ this.winningStates.push(new LongIntBitBoard(LongInt
30
+ .leftShift(this.LEADING_DIAGONAL, i + 7 * j)
31
+ .and(this.FULL_BOARD)));
32
+ }
33
+ }
34
+ for (let i = 0; i < 4; i++) {
35
+ for (let j = 0; j < 3; j++) {
36
+ this.winningStates.push(new LongIntBitBoard(LongInt
37
+ .leftShift(this.NON_LEADING_DIAGONAL, i + 7 * j)
38
+ .and(this.FULL_BOARD)));
39
+ }
40
+ }
41
+ }
42
+ get heuristic() {
43
+ const { winner } = this;
44
+ if (winner === 0)
45
+ return 1000;
46
+ if (winner === 1)
47
+ return -1000;
48
+ const p0LineOfThreeNoGaps = this.hasLine(0, 3);
49
+ const p1LineOfThreeNoGaps = this.hasLine(1, 3);
50
+ const p0LineOfThreeOneGap = this.hasLine(0, 3, 1);
51
+ const p1LineOfThreeOneGap = this.hasLine(1, 3, 1);
52
+ const p0LineOfThreeTwoGaps = this.hasLine(0, 3, 2);
53
+ const p1LineOfThreeTwoGaps = this.hasLine(1, 3, 2);
54
+ const p0LineOfTwoNoGaps = this.hasLine(0, 2);
55
+ const p1LineOfTwoNoGaps = this.hasLine(1, 2);
56
+ const p0LineOfTwoOneGap = this.hasLine(0, 2, 1);
57
+ const p1LineOfTwoOneGap = this.hasLine(1, 2, 1);
58
+ const p0LineOfTwoTwoGaps = this.hasLine(0, 2, 2);
59
+ const p1LineOfTwoTwoGaps = this.hasLine(1, 2, 2);
60
+ const p0LineOfThreeScore = 3 * p0LineOfThreeNoGaps + 2 * p0LineOfThreeOneGap + p0LineOfThreeTwoGaps;
61
+ const p1LineOfThreeScore = 3 * p1LineOfThreeNoGaps + 2 * p1LineOfThreeOneGap + p1LineOfThreeTwoGaps;
62
+ const p0LineOfTwoScore = 3 * p0LineOfTwoNoGaps + 2 * p0LineOfTwoOneGap + p0LineOfTwoTwoGaps;
63
+ const p1LineOfTwoScore = 3 * p1LineOfTwoNoGaps + 2 * p1LineOfTwoOneGap + p1LineOfTwoTwoGaps;
64
+ const p0Score = 10 * p0LineOfThreeScore + p0LineOfTwoScore;
65
+ const p1Score = 10 * p1LineOfThreeScore + p1LineOfTwoScore;
66
+ return p0Score - p1Score;
67
+ }
68
+ get emptyCells() {
69
+ const emptyCells = [];
70
+ for (let x = 0; x < this.boardWidth; x++) {
71
+ const cell = { x, y: 0 };
72
+ if (this.cellOccupier(cell) === null)
73
+ emptyCells.push(cell);
74
+ }
75
+ return emptyCells;
76
+ }
77
+ makeMove(move, playerId) {
78
+ const updatedMove = move;
79
+ for (let i = 5; i >= 0; i--) {
80
+ if (this.cellOccupier({ x: move.x, y: i }) === null) {
81
+ move.y = i;
82
+ break;
83
+ }
84
+ }
85
+ super.makeMove(updatedMove, playerId);
86
+ }
87
+ }
88
+ //# 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\nexport default class Board extends Base<LongIntBitBoard> {\n    protected winningStates: LongIntBitBoard[] = [];\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    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    public get heuristic(): number {\n        const { winner } = this;\n        if (winner === 0)\n            return 1000;\n        if (winner === 1)\n            return -1000;\n        const p0LineOfThreeNoGaps = this.hasLine(0, 3);\n        const p1LineOfThreeNoGaps = this.hasLine(1, 3);\n        const p0LineOfThreeOneGap = this.hasLine(0, 3, 1);\n        const p1LineOfThreeOneGap = this.hasLine(1, 3, 1);\n        const p0LineOfThreeTwoGaps = this.hasLine(0, 3, 2);\n        const p1LineOfThreeTwoGaps = this.hasLine(1, 3, 2);\n        const p0LineOfTwoNoGaps = this.hasLine(0, 2);\n        const p1LineOfTwoNoGaps = this.hasLine(1, 2);\n        const p0LineOfTwoOneGap = this.hasLine(0, 2, 1);\n        const p1LineOfTwoOneGap = this.hasLine(1, 2, 1);\n        const p0LineOfTwoTwoGaps = this.hasLine(0, 2, 2);\n        const p1LineOfTwoTwoGaps = this.hasLine(1, 2, 2);\n        const p0LineOfThreeScore = 3 * p0LineOfThreeNoGaps + 2 * p0LineOfThreeOneGap + p0LineOfThreeTwoGaps;\n        const p1LineOfThreeScore = 3 * p1LineOfThreeNoGaps + 2 * p1LineOfThreeOneGap + p1LineOfThreeTwoGaps;\n        const p0LineOfTwoScore = 3 * p0LineOfTwoNoGaps + 2 * p0LineOfTwoOneGap + p0LineOfTwoTwoGaps;\n        const p1LineOfTwoScore = 3 * p1LineOfTwoNoGaps + 2 * p1LineOfTwoOneGap + p1LineOfTwoTwoGaps;\n        const p0Score = 10 * p0LineOfThreeScore + p0LineOfTwoScore;\n        const p1Score = 10 * p1LineOfThreeScore + p1LineOfTwoScore;\n        return p0Score - p1Score;\n    }\n\n    public override get emptyCells(): Position[] {\n        const emptyCells: Position[] = [];\n        for (let x = 0; x < this.boardWidth; x++) {\n            const cell: Position = { x, y: 0 };\n            if (this.cellOccupier(cell) === null)\n                emptyCells.push(cell);\n        }\n        return emptyCells;\n    }\n\n    public override makeMove(move: Position, playerId: number): void {\n        const updatedMove = move;\n        for (let i = 5; i >= 0; i--) {\n            if (this.cellOccupier({ x: move.x, y: i }) === null) {\n                move.y = i;\n                break;\n            }\n        }\n        super.makeMove(updatedMove, playerId);\n    }\n}\n"]}
@@ -0,0 +1,11 @@
1
+ import type { Algorithm, GameConstructorOptions, PlayerType } from "../../base/controller.js";
2
+ import Base from "../../base/controller.js";
3
+ import type { Position } from "../../base/board.js";
4
+ export default class Connect4 extends Base {
5
+ constructor(playerOneType: PlayerType, playerTwoType: PlayerType, options?: GameConstructorOptions);
6
+ determineCPUMove(difficulty: Omit<PlayerType, "human">, algorithm?: Algorithm): Position;
7
+ findOptimalMove(options?: {
8
+ algorithm?: Algorithm;
9
+ maxDepth?: number;
10
+ }): Position;
11
+ }
@@ -0,0 +1,49 @@
1
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
6
+ };
7
+ import Base, { Game } from "../../base/controller.js";
8
+ import Board from "./board.js";
9
+ import Console from "../../console.js";
10
+ function defaultRender(controller) {
11
+ Console.clear();
12
+ Console.writeLine(controller.board.toString(true, true, false, ["⬤ ", "⬤ "]));
13
+ const { winner } = controller.board;
14
+ if (winner !== false)
15
+ Console.writeLine(winner === null ? "It's a tie!" : `Player ${winner + 1} wins!`);
16
+ }
17
+ let Connect4 = class Connect4 extends Base {
18
+ constructor(playerOneType, playerTwoType, options) {
19
+ super([playerOneType, playerTwoType], new Board(), options?.renderer ?? defaultRender, options?.id, options?.onEnd, options?.onInvalidInput);
20
+ }
21
+ determineCPUMove(difficulty, algorithm = "alphabeta") {
22
+ const { emptyCells } = this.board;
23
+ const randomMove = emptyCells[Math.floor(Math.random() * emptyCells.length)];
24
+ switch (difficulty) {
25
+ case "impossibleCPU":
26
+ return this.findOptimalMove({ algorithm, maxDepth: 10 });
27
+ case "hardCPU":
28
+ return this.findOptimalMove({ algorithm, maxDepth: 5 });
29
+ case "mediumCPU":
30
+ return this.findOptimalMove({ algorithm, maxDepth: 3 });
31
+ case "easyCPU":
32
+ return randomMove;
33
+ default:
34
+ throw new Error("Invalid difficulty.");
35
+ }
36
+ }
37
+ findOptimalMove(options) {
38
+ const { maxDepth = Infinity, algorithm = "alphabeta" } = options ?? {};
39
+ if (this.board.isEmpty)
40
+ return { x: 3, y: 5 };
41
+ const minimax = this[algorithm](maxDepth);
42
+ return minimax.move;
43
+ }
44
+ };
45
+ Connect4 = __decorate([
46
+ Game
47
+ ], Connect4);
48
+ export default Connect4;
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udHJvbGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9nYW1lcy9jb25uZWN0NC9jb250cm9sbGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7OztBQUNBLE9BQU8sSUFBSSxFQUFFLEVBQUUsSUFBSSxFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDdEQsT0FBTyxLQUFLLE1BQU0sWUFBWSxDQUFDO0FBQy9CLE9BQU8sT0FBTyxNQUFNLGtCQUFrQixDQUFDO0FBR3ZDLFNBQVMsYUFBYSxDQUFDLFVBQW9CO0lBQ3ZDLE9BQU8sQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUNoQixPQUFPLENBQUMsU0FBUyxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUM5RSxNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsVUFBVSxDQUFDLEtBQUssQ0FBQztJQUNwQyxJQUFJLE1BQU0sS0FBSyxLQUFLO1FBQ2hCLE9BQU8sQ0FBQyxTQUFTLENBQUMsTUFBTSxLQUFLLElBQUksQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxVQUFVLE1BQU0sR0FBRyxDQUFDLFFBQVEsQ0FBQyxDQUFDO0FBQzFGLENBQUM7QUFHYyxJQUFNLFFBQVEsR0FBZCxNQUFNLFFBQVMsU0FBUSxJQUFJO0lBQ3RDLFlBQW1CLGFBQXlCLEVBQUUsYUFBeUIsRUFBRSxPQUFnQztRQUNyRyxLQUFLLENBQ0QsQ0FBQyxhQUFhLEVBQUUsYUFBYSxDQUFDLEVBQzlCLElBQUksS0FBSyxFQUFFLEVBQ1gsT0FBTyxFQUFFLFFBQVEsSUFBSSxhQUFhLEVBQ2xDLE9BQU8sRUFBRSxFQUFFLEVBQ1gsT0FBTyxFQUFFLEtBQUssRUFDZCxPQUFPLEVBQUUsY0FBYyxDQUMxQixDQUFDO0lBQ04sQ0FBQztJQUVNLGdCQUFnQixDQUFDLFVBQXFDLEVBQUUsWUFBdUIsV0FBVztRQUM3RixNQUFNLEVBQUUsVUFBVSxFQUFFLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQztRQUNsQyxNQUFNLFVBQVUsR0FBRyxVQUFVLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEdBQUcsVUFBVSxDQUFDLE1BQU0sQ0FBQyxDQUFFLENBQUM7UUFDOUUsUUFBUSxVQUFVLEVBQUUsQ0FBQztZQUNqQixLQUFLLGVBQWU7Z0JBQ2hCLE9BQU8sSUFBSSxDQUFDLGVBQWUsQ0FBQyxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsRUFBRSxFQUFFLENBQUMsQ0FBQztZQUM3RCxLQUFLLFNBQVM7Z0JBQ1YsT0FBTyxJQUFJLENBQUMsZUFBZSxDQUFDLEVBQUUsU0FBUyxFQUFFLFFBQVEsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQzVELEtBQUssV0FBVztnQkFDWixPQUFPLElBQUksQ0FBQyxlQUFlLENBQUMsRUFBRSxTQUFTLEVBQUUsUUFBUSxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUM7WUFDNUQsS0FBSyxTQUFTO2dCQUNWLE9BQU8sVUFBVSxDQUFDO1lBQ3RCO2dCQUNJLE1BQU0sSUFBSSxLQUFLLENBQUMscUJBQXFCLENBQUMsQ0FBQztRQUMvQyxDQUFDO0lBQ0wsQ0FBQztJQUVNLGVBQWUsQ0FBQyxPQUd0QjtRQUNHLE1BQU0sRUFBRSxRQUFRLEdBQUcsUUFBUSxFQUFFLFNBQVMsR0FBRyxXQUFXLEVBQUUsR0FBRyxPQUFPLElBQUksRUFBRSxDQUFDO1FBQ3ZFLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPO1lBQ2xCLE9BQU8sRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQztRQUMxQixNQUFNLE9BQU8sR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDMUMsT0FBTyxPQUFPLENBQUMsSUFBSSxDQUFDO0lBQ3hCLENBQUM7Q0FDSixDQUFBO0FBdkNvQixRQUFRO0lBRDVCLElBQUk7R0FDZ0IsUUFBUSxDQXVDNUI7ZUF2Q29CLFFBQVEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IEFsZ29yaXRobSwgR2FtZUNvbnN0cnVjdG9yT3B0aW9ucywgUGxheWVyVHlwZSB9IGZyb20gXCIuLi8uLi9iYXNlL2NvbnRyb2xsZXIuanNcIjtcbmltcG9ydCBCYXNlLCB7IEdhbWUgfSBmcm9tIFwiLi4vLi4vYmFzZS9jb250cm9sbGVyLmpzXCI7XG5pbXBvcnQgQm9hcmQgZnJvbSBcIi4vYm9hcmQuanNcIjtcbmltcG9ydCBDb25zb2xlIGZyb20gXCIuLi8uLi9jb25zb2xlLmpzXCI7XG5pbXBvcnQgdHlwZSB7IFBvc2l0aW9uIH0gZnJvbSBcIi4uLy4uL2Jhc2UvYm9hcmQuanNcIjtcblxuZnVuY3Rpb24gZGVmYXVsdFJlbmRlcihjb250cm9sbGVyOiBDb25uZWN0NCk6IHZvaWQge1xuICAgIENvbnNvbGUuY2xlYXIoKTtcbiAgICBDb25zb2xlLndyaXRlTGluZShjb250cm9sbGVyLmJvYXJkLnRvU3RyaW5nKHRydWUsIHRydWUsIGZhbHNlLCBbXCLirKQgXCIsIFwi4qykIFwiXSkpO1xuICAgIGNvbnN0IHsgd2lubmVyIH0gPSBjb250cm9sbGVyLmJvYXJkO1xuICAgIGlmICh3aW5uZXIgIT09IGZhbHNlKVxuICAgICAgICBDb25zb2xlLndyaXRlTGluZSh3aW5uZXIgPT09IG51bGwgPyBcIkl0J3MgYSB0aWUhXCIgOiBgUGxheWVyICR7d2lubmVyICsgMX0gd2lucyFgKTtcbn1cblxuQEdhbWVcbmV4cG9ydCBkZWZhdWx0IGNsYXNzIENvbm5lY3Q0IGV4dGVuZHMgQmFzZSB7XG4gICAgcHVibGljIGNvbnN0cnVjdG9yKHBsYXllck9uZVR5cGU6IFBsYXllclR5cGUsIHBsYXllclR3b1R5cGU6IFBsYXllclR5cGUsIG9wdGlvbnM/OiBHYW1lQ29uc3RydWN0b3JPcHRpb25zKSB7XG4gICAgICAgIHN1cGVyKFxuICAgICAgICAgICAgW3BsYXllck9uZVR5cGUsIHBsYXllclR3b1R5cGVdLFxuICAgICAgICAgICAgbmV3IEJvYXJkKCksXG4gICAgICAgICAgICBvcHRpb25zPy5yZW5kZXJlciA/PyBkZWZhdWx0UmVuZGVyLFxuICAgICAgICAgICAgb3B0aW9ucz8uaWQsXG4gICAgICAgICAgICBvcHRpb25zPy5vbkVuZCxcbiAgICAgICAgICAgIG9wdGlvbnM/Lm9uSW52YWxpZElucHV0XG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgcHVibGljIGRldGVybWluZUNQVU1vdmUoZGlmZmljdWx0eTogT21pdDxQbGF5ZXJUeXBlLCBcImh1bWFuXCI+LCBhbGdvcml0aG06IEFsZ29yaXRobSA9IFwiYWxwaGFiZXRhXCIpOiBQb3NpdGlvbiB7XG4gICAgICAgIGNvbnN0IHsgZW1wdHlDZWxscyB9ID0gdGhpcy5ib2FyZDtcbiAgICAgICAgY29uc3QgcmFuZG9tTW92ZSA9IGVtcHR5Q2VsbHNbTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogZW1wdHlDZWxscy5sZW5ndGgpXSE7XG4gICAgICAgIHN3aXRjaCAoZGlmZmljdWx0eSkge1xuICAgICAgICAgICAgY2FzZSBcImltcG9zc2libGVDUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5maW5kT3B0aW1hbE1vdmUoeyBhbGdvcml0aG0sIG1heERlcHRoOiAxMCB9KTtcbiAgICAgICAgICAgIGNhc2UgXCJoYXJkQ1BVXCI6XG4gICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMuZmluZE9wdGltYWxNb3ZlKHsgYWxnb3JpdGhtLCBtYXhEZXB0aDogNSB9KTtcbiAgICAgICAgICAgIGNhc2UgXCJtZWRpdW1DUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5maW5kT3B0aW1hbE1vdmUoeyBhbGdvcml0aG0sIG1heERlcHRoOiAzIH0pO1xuICAgICAgICAgICAgY2FzZSBcImVhc3lDUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gcmFuZG9tTW92ZTtcbiAgICAgICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFwiSW52YWxpZCBkaWZmaWN1bHR5LlwiKTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIHB1YmxpYyBmaW5kT3B0aW1hbE1vdmUob3B0aW9ucz86IHtcbiAgICAgICAgYWxnb3JpdGhtPzogQWxnb3JpdGhtO1xuICAgICAgICBtYXhEZXB0aD86IG51bWJlcjtcbiAgICB9KTogUG9zaXRpb24ge1xuICAgICAgICBjb25zdCB7IG1heERlcHRoID0gSW5maW5pdHksIGFsZ29yaXRobSA9IFwiYWxwaGFiZXRhXCIgfSA9IG9wdGlvbnMgPz8ge307XG4gICAgICAgIGlmICh0aGlzLmJvYXJkLmlzRW1wdHkpXG4gICAgICAgICAgICByZXR1cm4geyB4OiAzLCB5OiA1IH07XG4gICAgICAgIGNvbnN0IG1pbmltYXggPSB0aGlzW2FsZ29yaXRobV0obWF4RGVwdGgpO1xuICAgICAgICByZXR1cm4gbWluaW1heC5tb3ZlO1xuICAgIH1cbn1cbiJdfQ==
@@ -0,0 +1,7 @@
1
+ import Base from "../../base/board.js";
2
+ import IntBitBoard from "../../bitBoard/intBitBoard.js";
3
+ export default class Board extends Base<IntBitBoard> {
4
+ protected winningStates: IntBitBoard[];
5
+ constructor();
6
+ get heuristic(): number;
7
+ }
@@ -0,0 +1,18 @@
1
+ import Base from "../../base/board.js";
2
+ import IntBitBoard from "../../bitBoard/intBitBoard.js";
3
+ export default class Board extends Base {
4
+ winningStates = [0x007, 0x038, 0x1C0, 0x049, 0x092, 0x124, 0x111, 0x054]
5
+ .map((data) => new IntBitBoard(data));
6
+ constructor() {
7
+ super(3, 3);
8
+ }
9
+ get heuristic() {
10
+ const { winner } = this;
11
+ if (winner === 0)
12
+ return 1;
13
+ if (winner === 1)
14
+ return -1;
15
+ return 0;
16
+ }
17
+ }
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYm9hcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZ2FtZXMvdGljdGFjdG9lL2JvYXJkLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sSUFBSSxNQUFNLHFCQUFxQixDQUFDO0FBQ3ZDLE9BQU8sV0FBVyxNQUFNLCtCQUErQixDQUFDO0FBRXhELE1BQU0sQ0FBQyxPQUFPLE9BQU8sS0FBTSxTQUFRLElBQWlCO0lBQ3RDLGFBQWEsR0FBa0IsQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxLQUFLLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxLQUFLLEVBQUUsS0FBSyxDQUFDO1NBQzVGLEdBQUcsQ0FBQyxDQUFDLElBQUksRUFBRSxFQUFFLENBQUMsSUFBSSxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUUxQztRQUNJLEtBQUssQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDaEIsQ0FBQztJQUVELElBQVcsU0FBUztRQUNoQixNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsSUFBSSxDQUFDO1FBQ3hCLElBQUksTUFBTSxLQUFLLENBQUM7WUFDWixPQUFPLENBQUMsQ0FBQztRQUNiLElBQUksTUFBTSxLQUFLLENBQUM7WUFDWixPQUFPLENBQUMsQ0FBQyxDQUFDO1FBQ2QsT0FBTyxDQUFDLENBQUM7SUFDYixDQUFDO0NBQ0oiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQmFzZSBmcm9tIFwiLi4vLi4vYmFzZS9ib2FyZC5qc1wiO1xuaW1wb3J0IEludEJpdEJvYXJkIGZyb20gXCIuLi8uLi9iaXRCb2FyZC9pbnRCaXRCb2FyZC5qc1wiO1xuXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBCb2FyZCBleHRlbmRzIEJhc2U8SW50Qml0Qm9hcmQ+IHtcbiAgICBwcm90ZWN0ZWQgd2lubmluZ1N0YXRlczogSW50Qml0Qm9hcmRbXSA9IFsweDAwNywgMHgwMzgsIDB4MUMwLCAweDA0OSwgMHgwOTIsIDB4MTI0LCAweDExMSwgMHgwNTRdXG4gICAgICAgIC5tYXAoKGRhdGEpID0+IG5ldyBJbnRCaXRCb2FyZChkYXRhKSk7XG5cbiAgICBwdWJsaWMgY29uc3RydWN0b3IoKSB7XG4gICAgICAgIHN1cGVyKDMsIDMpO1xuICAgIH1cblxuICAgIHB1YmxpYyBnZXQgaGV1cmlzdGljKCk6IG51bWJlciB7XG4gICAgICAgIGNvbnN0IHsgd2lubmVyIH0gPSB0aGlzO1xuICAgICAgICBpZiAod2lubmVyID09PSAwKVxuICAgICAgICAgICAgcmV0dXJuIDE7XG4gICAgICAgIGlmICh3aW5uZXIgPT09IDEpXG4gICAgICAgICAgICByZXR1cm4gLTE7XG4gICAgICAgIHJldHVybiAwO1xuICAgIH1cbn1cbiJdfQ==
@@ -0,0 +1,11 @@
1
+ import type { Algorithm, GameConstructorOptions, PlayerType } from "../../base/controller.js";
2
+ import Base from "../../base/controller.js";
3
+ import type { Position } from "../../base/board.js";
4
+ export default class TicTacToe extends Base {
5
+ constructor(playerOneType: PlayerType, playerTwoType: PlayerType, options?: GameConstructorOptions);
6
+ determineCPUMove(difficulty: Omit<PlayerType, "human">, algorithm?: Algorithm): Position;
7
+ findOptimalMove({ algorithm, randomMove }?: {
8
+ algorithm: Algorithm;
9
+ randomMove: Position;
10
+ }): Position;
11
+ }
@@ -0,0 +1,49 @@
1
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
6
+ };
7
+ import Base, { Game } from "../../base/controller.js";
8
+ import Board from "./board.js";
9
+ import Console from "../../console.js";
10
+ function defaultRender(controller) {
11
+ Console.clear();
12
+ Console.writeLine(controller.board.toString(false));
13
+ const { winner } = controller.board;
14
+ if (winner !== false)
15
+ Console.writeLine(winner === null ? "It's a tie!" : `Player ${winner + 1} wins!`);
16
+ }
17
+ let TicTacToe = class TicTacToe extends Base {
18
+ constructor(playerOneType, playerTwoType, options) {
19
+ super([playerOneType, playerTwoType], new Board(), options?.renderer ?? defaultRender, options?.id, options?.onEnd, options?.onInvalidInput);
20
+ }
21
+ determineCPUMove(difficulty, algorithm = "alphabeta") {
22
+ const { emptyCells } = this.board;
23
+ const randomMove = emptyCells[Math.floor(Math.random() * emptyCells.length)];
24
+ const optimalMove = this.findOptimalMove({ algorithm, randomMove });
25
+ switch (difficulty) {
26
+ case "impossibleCPU":
27
+ return optimalMove;
28
+ case "hardCPU":
29
+ return Math.random() < 0.8 ? optimalMove : randomMove;
30
+ case "mediumCPU":
31
+ return Math.random() < 0.5 ? optimalMove : randomMove;
32
+ case "easyCPU":
33
+ return randomMove;
34
+ default:
35
+ throw new Error("Invalid difficulty.");
36
+ }
37
+ }
38
+ findOptimalMove({ algorithm, randomMove } = { algorithm: "alphabeta", randomMove: { x: 2, y: 2 } }) {
39
+ if (this.board.isEmpty)
40
+ return randomMove;
41
+ const minimax = this[algorithm]();
42
+ return minimax.move;
43
+ }
44
+ };
45
+ TicTacToe = __decorate([
46
+ Game
47
+ ], TicTacToe);
48
+ export default TicTacToe;
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udHJvbGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9nYW1lcy90aWN0YWN0b2UvY29udHJvbGxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFDQSxPQUFPLElBQUksRUFBRSxFQUFFLElBQUksRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ3RELE9BQU8sS0FBSyxNQUFNLFlBQVksQ0FBQztBQUMvQixPQUFPLE9BQU8sTUFBTSxrQkFBa0IsQ0FBQztBQUd2QyxTQUFTLGFBQWEsQ0FBQyxVQUFxQjtJQUN4QyxPQUFPLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDaEIsT0FBTyxDQUFDLFNBQVMsQ0FBQyxVQUFVLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO0lBQ3BELE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxVQUFVLENBQUMsS0FBSyxDQUFDO0lBQ3BDLElBQUksTUFBTSxLQUFLLEtBQUs7UUFDaEIsT0FBTyxDQUFDLFNBQVMsQ0FBQyxNQUFNLEtBQUssSUFBSSxDQUFDLENBQUMsQ0FBQyxhQUFhLENBQUMsQ0FBQyxDQUFDLFVBQVUsTUFBTSxHQUFHLENBQUMsUUFBUSxDQUFDLENBQUM7QUFDMUYsQ0FBQztBQUdjLElBQU0sU0FBUyxHQUFmLE1BQU0sU0FBVSxTQUFRLElBQUk7SUFDdkMsWUFBbUIsYUFBeUIsRUFBRSxhQUF5QixFQUFFLE9BQWdDO1FBQ3JHLEtBQUssQ0FDRCxDQUFDLGFBQWEsRUFBRSxhQUFhLENBQUMsRUFDOUIsSUFBSSxLQUFLLEVBQUUsRUFDWCxPQUFPLEVBQUUsUUFBUSxJQUFJLGFBQWEsRUFDbEMsT0FBTyxFQUFFLEVBQUUsRUFDWCxPQUFPLEVBQUUsS0FBSyxFQUNkLE9BQU8sRUFBRSxjQUFjLENBQzFCLENBQUM7SUFDTixDQUFDO0lBRU0sZ0JBQWdCLENBQUMsVUFBcUMsRUFBRSxZQUF1QixXQUFXO1FBQzdGLE1BQU0sRUFBRSxVQUFVLEVBQUUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDO1FBQ2xDLE1BQU0sVUFBVSxHQUFHLFVBQVUsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxVQUFVLENBQUMsTUFBTSxDQUFDLENBQUUsQ0FBQztRQUM5RSxNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsZUFBZSxDQUFDLEVBQUUsU0FBUyxFQUFFLFVBQVUsRUFBRSxDQUFDLENBQUM7UUFDcEUsUUFBUSxVQUFVLEVBQUUsQ0FBQztZQUNqQixLQUFLLGVBQWU7Z0JBQ2hCLE9BQU8sV0FBVyxDQUFDO1lBQ3ZCLEtBQUssU0FBUztnQkFDVixPQUFPLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDO1lBQzFELEtBQUssV0FBVztnQkFDWixPQUFPLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDO1lBQzFELEtBQUssU0FBUztnQkFDVixPQUFPLFVBQVUsQ0FBQztZQUN0QjtnQkFDSSxNQUFNLElBQUksS0FBSyxDQUFDLHFCQUFxQixDQUFDLENBQUM7UUFDL0MsQ0FBQztJQUNMLENBQUM7SUFFTSxlQUFlLENBQUMsRUFBRSxTQUFTLEVBQUUsVUFBVSxLQUcxQyxFQUFFLFNBQVMsRUFBRSxXQUFXLEVBQUUsVUFBVSxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUU7UUFDdEQsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU87WUFDbEIsT0FBTyxVQUFVLENBQUM7UUFDdEIsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUM7UUFDbEMsT0FBTyxPQUFPLENBQUMsSUFBSSxDQUFDO0lBQ3hCLENBQUM7Q0FDSixDQUFBO0FBdkNvQixTQUFTO0lBRDdCLElBQUk7R0FDZ0IsU0FBUyxDQXVDN0I7ZUF2Q29CLFNBQVMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IEFsZ29yaXRobSwgR2FtZUNvbnN0cnVjdG9yT3B0aW9ucywgUGxheWVyVHlwZSB9IGZyb20gXCIuLi8uLi9iYXNlL2NvbnRyb2xsZXIuanNcIjtcbmltcG9ydCBCYXNlLCB7IEdhbWUgfSBmcm9tIFwiLi4vLi4vYmFzZS9jb250cm9sbGVyLmpzXCI7XG5pbXBvcnQgQm9hcmQgZnJvbSBcIi4vYm9hcmQuanNcIjtcbmltcG9ydCBDb25zb2xlIGZyb20gXCIuLi8uLi9jb25zb2xlLmpzXCI7XG5pbXBvcnQgdHlwZSB7IFBvc2l0aW9uIH0gZnJvbSBcIi4uLy4uL2Jhc2UvYm9hcmQuanNcIjtcblxuZnVuY3Rpb24gZGVmYXVsdFJlbmRlcihjb250cm9sbGVyOiBUaWNUYWNUb2UpOiB2b2lkIHtcbiAgICBDb25zb2xlLmNsZWFyKCk7XG4gICAgQ29uc29sZS53cml0ZUxpbmUoY29udHJvbGxlci5ib2FyZC50b1N0cmluZyhmYWxzZSkpO1xuICAgIGNvbnN0IHsgd2lubmVyIH0gPSBjb250cm9sbGVyLmJvYXJkO1xuICAgIGlmICh3aW5uZXIgIT09IGZhbHNlKVxuICAgICAgICBDb25zb2xlLndyaXRlTGluZSh3aW5uZXIgPT09IG51bGwgPyBcIkl0J3MgYSB0aWUhXCIgOiBgUGxheWVyICR7d2lubmVyICsgMX0gd2lucyFgKTtcbn1cblxuQEdhbWVcbmV4cG9ydCBkZWZhdWx0IGNsYXNzIFRpY1RhY1RvZSBleHRlbmRzIEJhc2Uge1xuICAgIHB1YmxpYyBjb25zdHJ1Y3RvcihwbGF5ZXJPbmVUeXBlOiBQbGF5ZXJUeXBlLCBwbGF5ZXJUd29UeXBlOiBQbGF5ZXJUeXBlLCBvcHRpb25zPzogR2FtZUNvbnN0cnVjdG9yT3B0aW9ucykge1xuICAgICAgICBzdXBlcihcbiAgICAgICAgICAgIFtwbGF5ZXJPbmVUeXBlLCBwbGF5ZXJUd29UeXBlXSxcbiAgICAgICAgICAgIG5ldyBCb2FyZCgpLFxuICAgICAgICAgICAgb3B0aW9ucz8ucmVuZGVyZXIgPz8gZGVmYXVsdFJlbmRlcixcbiAgICAgICAgICAgIG9wdGlvbnM/LmlkLFxuICAgICAgICAgICAgb3B0aW9ucz8ub25FbmQsXG4gICAgICAgICAgICBvcHRpb25zPy5vbkludmFsaWRJbnB1dFxuICAgICAgICApO1xuICAgIH1cblxuICAgIHB1YmxpYyBkZXRlcm1pbmVDUFVNb3ZlKGRpZmZpY3VsdHk6IE9taXQ8UGxheWVyVHlwZSwgXCJodW1hblwiPiwgYWxnb3JpdGhtOiBBbGdvcml0aG0gPSBcImFscGhhYmV0YVwiKTogUG9zaXRpb24ge1xuICAgICAgICBjb25zdCB7IGVtcHR5Q2VsbHMgfSA9IHRoaXMuYm9hcmQ7XG4gICAgICAgIGNvbnN0IHJhbmRvbU1vdmUgPSBlbXB0eUNlbGxzW01hdGguZmxvb3IoTWF0aC5yYW5kb20oKSAqIGVtcHR5Q2VsbHMubGVuZ3RoKV0hO1xuICAgICAgICBjb25zdCBvcHRpbWFsTW92ZSA9IHRoaXMuZmluZE9wdGltYWxNb3ZlKHsgYWxnb3JpdGhtLCByYW5kb21Nb3ZlIH0pO1xuICAgICAgICBzd2l0Y2ggKGRpZmZpY3VsdHkpIHtcbiAgICAgICAgICAgIGNhc2UgXCJpbXBvc3NpYmxlQ1BVXCI6XG4gICAgICAgICAgICAgICAgcmV0dXJuIG9wdGltYWxNb3ZlO1xuICAgICAgICAgICAgY2FzZSBcImhhcmRDUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gTWF0aC5yYW5kb20oKSA8IDAuOCA/IG9wdGltYWxNb3ZlIDogcmFuZG9tTW92ZTtcbiAgICAgICAgICAgIGNhc2UgXCJtZWRpdW1DUFVcIjpcbiAgICAgICAgICAgICAgICByZXR1cm4gTWF0aC5yYW5kb20oKSA8IDAuNSA/IG9wdGltYWxNb3ZlIDogcmFuZG9tTW92ZTtcbiAgICAgICAgICAgIGNhc2UgXCJlYXN5Q1BVXCI6XG4gICAgICAgICAgICAgICAgcmV0dXJuIHJhbmRvbU1vdmU7XG4gICAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcIkludmFsaWQgZGlmZmljdWx0eS5cIik7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwdWJsaWMgZmluZE9wdGltYWxNb3ZlKHsgYWxnb3JpdGhtLCByYW5kb21Nb3ZlIH06IHtcbiAgICAgICAgYWxnb3JpdGhtOiBBbGdvcml0aG07XG4gICAgICAgIHJhbmRvbU1vdmU6IFBvc2l0aW9uO1xuICAgIH0gPSB7IGFsZ29yaXRobTogXCJhbHBoYWJldGFcIiwgcmFuZG9tTW92ZTogeyB4OiAyLCB5OiAyIH0gfSk6IFBvc2l0aW9uIHtcbiAgICAgICAgaWYgKHRoaXMuYm9hcmQuaXNFbXB0eSlcbiAgICAgICAgICAgIHJldHVybiByYW5kb21Nb3ZlO1xuICAgICAgICBjb25zdCBtaW5pbWF4ID0gdGhpc1thbGdvcml0aG1dKCk7XG4gICAgICAgIHJldHVybiBtaW5pbWF4Lm1vdmU7XG4gICAgfVxufVxuIl19
@@ -0,0 +1,2 @@
1
+ export * from "./games/tictactoe/controller.js";
2
+ export * from "./games/connect4/controller.js";
package/build/index.js ADDED
@@ -0,0 +1,3 @@
1
+ export * from "./games/tictactoe/controller.js";
2
+ export * from "./games/connect4/controller.js";
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLGdDQUFnQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vZ2FtZXMvdGljdGFjdG9lL2NvbnRyb2xsZXIuanNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2dhbWVzL2Nvbm5lY3Q0L2NvbnRyb2xsZXIuanNcIjtcbiJdfQ==
package/package.json ADDED
@@ -0,0 +1,32 @@
1
+ {
2
+ "name": "@oathompsonjones/mini-games",
3
+ "version": "1.0.0",
4
+ "description": "A selection of MiniGames. You will soon be able to play these games on [my website](https://oathompsonjones.co.uk/arcade).",
5
+ "repository": {
6
+ "url": "https://github.com/oathompsonjones/MiniGames.git"
7
+ },
8
+ "main": "./build/index.js",
9
+ "types": "./build/index.d.ts",
10
+ "exports": "./build/index.js",
11
+ "type": "module",
12
+ "author": "oathompsonjones",
13
+ "license": "MIT",
14
+ "devDependencies": {
15
+ "@oathompsonjones/eslint-config": "^0.0.72",
16
+ "@oathompsonjones/ts-config": "^0.0.27",
17
+ "@types/node": "^20.10.5",
18
+ "typescript": "^5.3.3",
19
+ "vitest": "^1.1.0"
20
+ },
21
+ "dependencies": {
22
+ "@milahu/patch-package": "^6.4.14",
23
+ "eventemitter3": "^5.0.1"
24
+ },
25
+ "scripts": {
26
+ "lint": "eslint .",
27
+ "test": "vitest run",
28
+ "build": "tsc --project tsconfig.build.json",
29
+ "start": "node .",
30
+ "postinstall": "patch-package"
31
+ }
32
+ }
@@ -0,0 +1,49 @@
1
+ # generated by patch-package 6.4.14
2
+ #
3
+ # declared package:
4
+ # eventemitter3: ^5.0.1
5
+ #
6
+ diff --git a/node_modules/eventemitter3/index.d.ts b/node_modules/eventemitter3/index.d.ts
7
+ index fab37fd..f020700 100644
8
+ --- a/node_modules/eventemitter3/index.d.ts
9
+ +++ b/node_modules/eventemitter3/index.d.ts
10
+ @@ -81,7 +81,7 @@ declare class EventEmitter<
11
+
12
+ declare namespace EventEmitter {
13
+ export interface ListenerFn<Args extends any[] = any[]> {
14
+ - (...args: Args): void;
15
+ + (...args: Args): any;
16
+ }
17
+
18
+ export interface EventEmitterStatic {
19
+ @@ -96,7 +96,7 @@ declare namespace EventEmitter {
20
+ * ```
21
+ * interface EventTypes {
22
+ * 'event-with-parameters': any[]
23
+ - * 'event-with-example-handler': (...args: any[]) => void
24
+ + * 'event-with-example-handler': (...args: any[]) => any
25
+ * }
26
+ * ```
27
+ */
28
+ @@ -107,7 +107,7 @@ declare namespace EventEmitter {
29
+ : keyof T;
30
+
31
+ export type ArgumentMap<T extends object> = {
32
+ - [K in keyof T]: T[K] extends (...args: any[]) => void
33
+ + [K in keyof T]: T[K] extends (...args: any[]) => any
34
+ ? Parameters<T[K]>
35
+ : T[K] extends any[]
36
+ ? T[K]
37
+ @@ -118,10 +118,10 @@ declare namespace EventEmitter {
38
+ T extends ValidEventTypes,
39
+ K extends EventNames<T>
40
+ > = T extends string | symbol
41
+ - ? (...args: any[]) => void
42
+ + ? (...args: any[]) => any
43
+ : (
44
+ ...args: ArgumentMap<Exclude<T, string | symbol>>[Extract<K, keyof T>]
45
+ - ) => void;
46
+ + ) => any;
47
+
48
+ export type EventArgs<
49
+ T extends ValidEventTypes,
@@ -0,0 +1,8 @@
1
+ {
2
+ "extends": "./tsconfig.json",
3
+ "exclude": [
4
+ "src/tests",
5
+ "src/**/*.test.ts",
6
+ "src/**/*.spec.ts"
7
+ ]
8
+ }