typings on ai...
arsen@cntrl.site
X
PGNhbnZhcyBpZD0iZ2FtZUNhbnZhcyI+PC9jYW52YXM+CjxzY3JpcHQ+CiAgICBjbGFzcyBCaXRGaWVsZCB7CiAgICAgICAgY29uc3RydWN0b3IobGVuZ3RoKSB7CiAgICAgICAgICAgIHRoaXMubGVuZ3RoID0gbGVuZ3RoOwogICAgICAgICAgICB0aGlzLmZpZWxkID0gbmV3IFVpbnQ4QXJyYXkoTWF0aC5jZWlsKGxlbmd0aCAvIDgpKTsKICAgICAgICB9CgogICAgICAgIGdldCBzaXplKCkgewogICAgICAgICAgICByZXR1cm4gdGhpcy5sZW5ndGg7CiAgICAgICAgfQoKICAgICAgICBnZXQoaW5kZXgpIHsKICAgICAgICAgICAgaWYgKGluZGV4ID49IHRoaXMubGVuZ3RoKSByZXR1cm4gZmFsc2U7CiAgICAgICAgICAgIGNvbnN0IGlCeXRlID0gaW5kZXggPj4gMzsKICAgICAgICAgICAgY29uc3QgaUJpdCA9IGluZGV4ICUgODsKICAgICAgICAgICAgY29uc3QgYnl0ZSA9IHRoaXMuZmllbGRbaUJ5dGVdOwogICAgICAgICAgICByZXR1cm4gKGJ5dGUgJiAoMSA8PCBpQml0KSkgIT09IDA7CiAgICAgICAgfQoKICAgICAgICBzZXQoaW5kZXgsIHZhbHVlKSB7CiAgICAgICAgICAgIGlmIChpbmRleCA+PSB0aGlzLmxlbmd0aCkgcmV0dXJuOwogICAgICAgICAgICBjb25zdCBpQnl0ZSA9IGluZGV4ID4+IDM7CiAgICAgICAgICAgIGNvbnN0IGlCaXQgPSBpbmRleCAlIDg7CiAgICAgICAgICAgIGNvbnN0IGJ5dGUgPSB0aGlzLmZpZWxkW2lCeXRlXTsKICAgICAgICAgICAgdGhpcy5maWVsZFtpQnl0ZV0gPSAoCiAgICAgICAgICAgICAgICB2YWx1ZQogICAgICAgICAgICAgICAgICAgID8gKGJ5dGUgfCAoMSA8PCBpQml0KSkKICAgICAgICAgICAgICAgICAgICA6IChieXRlICYgKCh+KDEgPDwgaUJpdCkpICYgMHhmZikpCiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgfQoKICAgIGNvbnN0IG5laWdoYm91cnMgPSBbCiAgICAgICAgWy0xLCAtMV0sIFswLCAtMV0sIFsxLCAtMV0sCiAgICAgICAgWy0xLCAwXSwgICAgICAgICAgWzEsIDBdLAogICAgICAgIFstMSwgMV0sICBbMCwgMV0sICBbMSwgMV0KICAgIF07CgogICAgY2xhc3MgQ2VsbEZpZWxkIHsKICAgICAgICBjb25zdHJ1Y3Rvcih3aWR0aCwgaGVpZ2h0KSB7CiAgICAgICAgICAgIHRoaXMud2lkdGggPSB3aWR0aDsKICAgICAgICAgICAgdGhpcy5oZWlnaHQgPSBoZWlnaHQ7CiAgICAgICAgICAgIHRoaXMuY2VsbHMgPSBuZXcgQml0RmllbGQod2lkdGggKiBoZWlnaHQpOwogICAgICAgIH0KCiAgICAgICAgc2V0Q2VsbEF0KHgsIHksIHZhbHVlKSB7CiAgICAgICAgICAgIHRoaXMuY2VsbHMuc2V0KHRoaXMuZ2V0SW5kZXgoeCwgeSksIHZhbHVlKTsKICAgICAgICB9CgogICAgICAgIGdldENlbGxBdCh4LCB5KSB7CiAgICAgICAgICAgIHJldHVybiB0aGlzLmNlbGxzLmdldCh0aGlzLmdldEluZGV4KHgsIHkpKTsKICAgICAgICB9CgogICAgICAgIGdldEFsaXZlTmVpZ2hib3Vycyh4LCB5KSB7CiAgICAgICAgICAgIHJldHVybiBuZWlnaGJvdXJzLnJlZHVjZSgoYWxpdmUsIFtkeCwgZHldKSA9PgogICAgICAgICAgICAgICAgYWxpdmUgKyAodGhpcy5nZXRDZWxsQXQoeCArIGR4LCB5ICsgZHkpID8gMSA6IDApLCAwKTsKICAgICAgICB9CgogICAgICAgIG5leHRHZW4oKSB7CiAgICAgICAgICAgIGNvbnN0IG5leHRDZWxscyA9IG5ldyBCaXRGaWVsZCh0aGlzLndpZHRoICogdGhpcy5oZWlnaHQpOwogICAgICAgICAgICBmb3IgKGxldCB5ID0gMDsgeSA8IHRoaXMuaGVpZ2h0OyB5ICs9IDEpIHsKICAgICAgICAgICAgICAgIGZvciAobGV0IHggPSAwOyB4IDwgdGhpcy53aWR0aDsgeCArPSAxKSB7CiAgICAgICAgICAgICAgICAgICAgbmV4dENlbGxzLnNldCh0aGlzLmdldEluZGV4KHgsIHkpLCB0aGlzLmRvZXNDZWxsU3Vydml2ZSh4LCB5KSk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgICAgdGhpcy5jZWxscyA9IG5leHRDZWxsczsKICAgICAgICB9CgogICAgICAgIGRvZXNDZWxsU3Vydml2ZSh4LCB5KSB7CiAgICAgICAgICAgIGNvbnN0IGlzQWxpdmUgPSB0aGlzLmdldENlbGxBdCh4LCB5KTsKICAgICAgICAgICAgY29uc3QgYW4gPSB0aGlzLmdldEFsaXZlTmVpZ2hib3Vycyh4LCB5KTsKICAgICAgICAgICAgaWYgKGlzQWxpdmUpIHsKICAgICAgICAgICAgICAgIGlmIChhbiA8IDIpIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgICAgIGlmIChhbiA8PSAzKSByZXR1cm4gdHJ1ZTsKICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgfQogICAgICAgICAgICByZXR1cm4gYW4gPT09IDM7CiAgICAgICAgfQoKICAgICAgICBnZXRJbmRleCh4LCB5KSB7CiAgICAgICAgICAgIGNvbnN0IHgxID0gKHggKyB0aGlzLndpZHRoKSAlIHRoaXMud2lkdGg7CiAgICAgICAgICAgIGNvbnN0IHkxID0gKHkgKyB0aGlzLmhlaWdodCkgJSB0aGlzLmhlaWdodDsKICAgICAgICAgICAgcmV0dXJuIHkxICogdGhpcy53aWR0aCArIHgxOwogICAgICAgIH0KICAgIH0KCiAgICBjb25zdCBTID0gNjsgICAvLyBDZWxsIHNpemUKICAgIGNvbnN0IENXID0gNjAwOyAvLyBXaWR0aCBpbiBjZWxscwogICAgY29uc3QgQ0ggPSAyODA7IC8vIEhlaWdodCBpbiBjZWxscwogICAgY29uc3QgbGRjZWxscyA9IFsKICAgICAgICBbMCwgMF0sCiAgICAgICAgWzEsIDBdLAogICAgICAgIFswLCAxXSwKICAgICAgICBbMSwgMV0KICAgIF07CgogICAgZnVuY3Rpb24gZHJhd0dlbihjdHgsIGNlbGxzLCBjZWxsU2l6ZSkgewogICAgICAgIGZvciAobGV0IHkgPSAwOyB5IDwgY2VsbHMuaGVpZ2h0OyB5ICs9IDEpIHsKICAgICAgICAgICAgZm9yIChsZXQgeCA9IDA7IHggPCBjZWxscy53aWR0aDsgeCArPSAxKSB7CiAgICAgICAgICAgICAgICBpZiAoIWNlbGxzLmdldENlbGxBdCh4LCB5KSkgY29udGludWU7CiAgICAgICAgICAgICAgICBjdHguZmlsbFJlY3QoeCAqIGNlbGxTaXplLCB5ICogY2VsbFNpemUsIGNlbGxTaXplIC0gMSwgY2VsbFNpemUgLSAxKTsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICBmdW5jdGlvbiBkcmF3TG9nbyhjZWxscywgeCwgeSwgbGVuKSB7CiAgICAgICAgY29uc3QgbGluZXMgPSBsb2dvU3RyLnNwbGl0KCdcbicpLnNsaWNlKDEsIC0xKTsKICAgICAgICBmb3IgKGxldCBpeSA9IDA7IGl5IDwgbGluZXMubGVuZ3RoOyBpeSArPSAxKSB7CiAgICAgICAgICAgIGNvbnN0IGxpbmUgPSBsaW5lc1tpeV07CiAgICAgICAgICAgIGZvciAobGV0IGl4ID0gMDsgaXggPCBsaW5lLmxlbmd0aDsgaXggKz0gMSkgewogICAgICAgICAgICAgICAgY29uc3QgZHJhdyA9IGxpbmVbaXhdID09PSAnKic7CiAgICAgICAgICAgICAgICBmb3IgKGNvbnN0IFtkeCwgZHldIG9mIGxkY2VsbHMpIHsKICAgICAgICAgICAgICAgICAgICBjb25zdCBjeCA9IHggKyBpeCAqIDIgKyBkeDsKICAgICAgICAgICAgICAgICAgICBjb25zdCBjeSA9IHkgKyBpeSAqIDIgKyBkeTsKICAgICAgICAgICAgICAgICAgICBjb25zdCB2YWx1ZSA9IGRyYXcgPyAoY2VsbHMuZ2V0Q2VsbEF0KGN4LCBjeSkgfHwgKE1hdGgucmFuZG9tKCkgPCBsZW4pKSA6IGZhbHNlOwogICAgICAgICAgICAgICAgICAgIGNlbGxzLnNldENlbGxBdChjeCwgY3ksIHZhbHVlKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICBmdW5jdGlvbiBjcmVhdGVBbmltYXRpb24ocGFyYW1zKSB7CiAgICAgICAgY29uc3QgeyBjYW52YXMsIHdpZHRoLCBoZWlnaHQsIGNlbGxTaXplLCBnZW5lcmF0aW9uVGltZW91dCwgbG9nb1RpbWVvdXQsIGNvbG9yIH0gPSBwYXJhbXM7CiAgICAgICAgY29uc3QgY2VsbHMgPSBuZXcgQ2VsbEZpZWxkKHdpZHRoLCBoZWlnaHQpOwogICAgICAgIGNvbnN0IHZwQ3R4ID0gY2FudmFzLmdldENvbnRleHQoJzJkJyk7CiAgICAgICAgY29uc3QgYnVmZmVyID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnY2FudmFzJyk7CiAgICAgICAgY29uc3QgYmZDdHggPSBidWZmZXIuZ2V0Q29udGV4dCgnMmQnKTsKCiAgICAgICAgbGV0IGxhc3RHZW4gPSBEYXRlLm5vdygpOwogICAgICAgIGxldCBsYXN0TG9nbyA9IERhdGUubm93KCkgLSBsb2dvVGltZW91dDsKICAgICAgICBsZXQgaXNMb2dvVmlzaWJsZSA9IGZhbHNlOwoKICAgICAgICBidWZmZXIud2lkdGggPSB3aWR0aCAqIGNlbGxTaXplOwogICAgICAgIGJ1ZmZlci5oZWlnaHQgPSBoZWlnaHQgKiBjZWxsU2l6ZTsKICAgICAgICBjYW52YXMud2lkdGggPSB3aWR0aCAqIGNlbGxTaXplOwogICAgICAgIGNhbnZhcy5oZWlnaHQgPSBoZWlnaHQgKiBjZWxsU2l6ZTsKCiAgICAgICAgLy8gQWRqdXN0ZWQgbW91c2UgcG9zaXRpb24gaGFuZGxpbmcgdG8gY29uc2lkZXIgc2NhbGluZwogICAgICAgIHdpbmRvdy5hZGRFdmVudExpc3RlbmVyKCdtb3VzZW1vdmUnLCAoZXZlbnQpID0+IHsKICAgICAgICAgICAgY29uc3QgcmVjdCA9IGNhbnZhcy5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKTsKICAgICAgICAgICAgY29uc3Qgc2NhbGVYID0gY2FudmFzLndpZHRoIC8gcmVjdC53aWR0aDsKICAgICAgICAgICAgY29uc3Qgc2NhbGVZID0gY2FudmFzLmhlaWdodCAvIHJlY3QuaGVpZ2h0OwogICAgICAgICAgICBjb25zdCB4ID0gTWF0aC5mbG9vcigoZXZlbnQuY2xpZW50WCAtIHJlY3QubGVmdCkgKiBzY2FsZVggLyBjZWxsU2l6ZSk7CiAgICAgICAgICAgIGNvbnN0IHkgPSBNYXRoLmZsb29yKChldmVudC5jbGllbnRZIC0gcmVjdC50b3ApICogc2NhbGVZIC8gY2VsbFNpemUpOwogICAgICAgICAgICBjZWxscy5zZXRDZWxsQXQoeCwgeSwgdHJ1ZSk7IC8vIE1hcmsgdGhlIGNlbGwgYXMgYWxpdmUKICAgICAgICB9KTsKCiAgICAgICAgZnVuY3Rpb24gbG9vcCgpIHsKICAgICAgICAgICAgaWYgKCFiZkN0eCB8fCAhdnBDdHgpIHJldHVybjsKICAgICAgICAgICAgY29uc3Qgbm93ID0gRGF0ZS5ub3coKTsKICAgICAgICAgICAgY29uc3QgZ2VuRHQgPSBub3cgLSBsYXN0R2VuOwogICAgICAgICAgICBjb25zdCBsb2dvRHQgPSBub3cgLSBsYXN0TG9nbzsKCiAgICAgICAgICAgIGlmIChnZW5EdCA+IGdlbmVyYXRpb25UaW1lb3V0KSB7CiAgICAgICAgICAgICAgICBsYXN0R2VuID0gbm93OwogICAgICAgICAgICAgICAgY2VsbHMubmV4dEdlbigpOwogICAgICAgICAgICB9CiAgICAgICAgICAgIGlmIChsb2dvRHQgPiBsb2dvVGltZW91dCkgewogICAgICAgICAgICAgICAgbGFzdExvZ28gPSBub3c7CiAgICAgICAgICAgICAgICBpc0xvZ29WaXNpYmxlID0gIWlzTG9nb1Zpc2libGU7CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIGJmQ3R4LmNsZWFyUmVjdCgwLCAwLCBidWZmZXIud2lkdGgsIGJ1ZmZlci5oZWlnaHQpOwogICAgICAgICAgICBiZkN0eC5maWxsU3R5bGUgPSBjb2xvcjsKICAgICAgICAgICAgZHJhd0dlbihiZkN0eCwgY2VsbHMsIGNlbGxTaXplKTsKICAgICAgICAgICAgdnBDdHgucHV0SW1hZ2VEYXRhKGJmQ3R4LmdldEltYWdlRGF0YSgwLCAwLCBidWZmZXIud2lkdGgsIGJ1ZmZlci5oZWlnaHQpLCAwLCAwKTsKICAgICAgICAgICAgcmVxdWVzdEFuaW1hdGlvbkZyYW1lKGxvb3ApOwogICAgICAgIH0KCiAgICAgICAgbG9vcCgpOwogICAgfQoKICAgIGNvbnN0IGNhbnZhcyA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdnYW1lQ2FudmFzJyk7CiAgICBjb25zdCBjb2xvciA9ICd3aGl0ZSc7ICAvLyBZb3UgY2FuIGNoYW5nZSB0aGUgY2VsbCBjb2xvciBoZXJlCiAgICBjcmVhdGVBbmltYXRpb24oewogICAgICAgIGNhbnZhczogY2FudmFzLAogICAgICAgIHdpZHRoOiBDVywKICAgICAgICBoZWlnaHQ6IENILAogICAgICAgIGNlbGxTaXplOiBTLAogICAgICAgIGdlbmVyYXRpb25UaW1lb3V0OiAzMDAsCiAgICAgICAgbG9nb1RpbWVvdXQ6IDQwNTAsCiAgICAgICAgY29sb3I6IGNvbG9yCiAgICB9KTsKPC9zY3JpcHQ+
Github
Building A Web Design Tool
Control
Building A Web Design Tool
Control
Crafting sites and apps at
MOM DESIGN
Crafting sites and apps at
MOM DESIGN
Software Developer
Vienna, Austria
<canvas id="gameCanvas"></canvas>
<script>
    class BitField {
        constructor(length) {
            this.length = length;
            this.field = new Uint8Array(Math.ceil(length / 8));
        }

        get size() {
            return this.length;
        }

        get(index) {
            if (index >= this.length) return false;
            const iByte = index >> 3;
            const iBit = index % 8;
            const byte = this.field[iByte];
            return (byte & (1 << iBit)) !== 0;
        }

        set(index, value) {
            if (index >= this.length) return;
            const iByte = index >> 3;
            const iBit = index % 8;
            const byte = this.field[iByte];
            this.field[iByte] = (
                value
                    ? (byte | (1 << iBit))
                    : (byte & ((~(1 << iBit)) & 0xff))
            );
        }
    }

    const neighbours = [
        [-1, -1], [0, -1], [1, -1],
        [-1, 0],          [1, 0],
        [-1, 1],  [0, 1],  [1, 1]
    ];

    class CellField {
        constructor(width, height) {
            this.width = width;
            this.height = height;
            this.cells = new BitField(width * height);
        }

        setCellAt(x, y, value) {
            this.cells.set(this.getIndex(x, y), value);
        }

        getCellAt(x, y) {
            return this.cells.get(this.getIndex(x, y));
        }

        getAliveNeighbours(x, y) {
            return neighbours.reduce((alive, [dx, dy]) =>
                alive + (this.getCellAt(x + dx, y + dy) ? 1 : 0), 0);
        }

        nextGen() {
            const nextCells = new BitField(this.width * this.height);
            for (let y = 0; y < this.height; y += 1) {
                for (let x = 0; x < this.width; x += 1) {
                    nextCells.set(this.getIndex(x, y), this.doesCellSurvive(x, y));
                }
            }
            this.cells = nextCells;
        }

        doesCellSurvive(x, y) {
            const isAlive = this.getCellAt(x, y);
            const an = this.getAliveNeighbours(x, y);
            if (isAlive) {
                if (an < 2) return false;
                if (an <= 3) return true;
                return false;
            }
            return an === 3;
        }

        getIndex(x, y) {
            const x1 = (x + this.width) % this.width;
            const y1 = (y + this.height) % this.height;
            return y1 * this.width + x1;
        }
    }

    const S = 6;   // Cell size
    const CW = 600; // Width in cells
    const CH = 280; // Height in cells

    const logoStr = `
******  ******   ******  ******  *    *    ******   ******  ******  *    *  *     *   ****   *    *   ****   *  *
*    *  *     *  *       *       **   *    *     *  *    *    *     *    *  *     *  *    *  *    *  *    *  * *
******  *****    ******  ****    * *  *    ******   ******    *      *  *   *     *  *       ******  *    *  **
*    *  *    *        *  *       *  * *    *     *  *    *     *      **    *     *  *    *  *    *  *    *  * *
*    *  *     *  ******  ******  *   **    ******   *    *     *      **     *****    ****   *    *   ****   *  *
`;

    const ldcells = [
        [0, 0],
        [1, 0],
        [0, 1],
        [1, 1]
    ];

    function drawGen(ctx, cells, cellSize) {
        for (let y = 0; y < cells.height; y += 1) {
            for (let x = 0; x < cells.width; x += 1) {
                if (!cells.getCellAt(x, y)) continue;
                ctx.fillRect(x * cellSize, y * cellSize, cellSize - 1, cellSize - 1);
            }
        }
    }

    function drawLogo(cells, x, y, len) {
        const lines = logoStr.split('\n').slice(1, -1);
        for (let iy = 0; iy < lines.length; iy += 1) {
            const line = lines[iy];
            for (let ix = 0; ix < line.length; ix += 1) {
                const draw = line[ix] === '*';
                for (const [dx, dy] of ldcells) {
                    const cx = x + ix * 2 + dx;
                    const cy = y + iy * 2 + dy;
                    const value = draw ? (cells.getCellAt(cx, cy) || (Math.random() < len)) : false;
                    cells.setCellAt(cx, cy, value);
                }
            }
        }
    }

    function createAnimation(params) {
        const { canvas, width, height, cellSize, generationTimeout, logoTimeout, color } = params;
        const cells = new CellField(width, height);
        const vpCtx = canvas.getContext('2d');
        const buffer = document.createElement('canvas');
        const bfCtx = buffer.getContext('2d');

        let lastGen = Date.now();
        let lastLogo = Date.now() - logoTimeout;
        let isLogoVisible = false;

        buffer.width = width * cellSize;
        buffer.height = height * cellSize;
        canvas.width = width * cellSize;
        canvas.height = height * cellSize;

        // Adjusted mouse position handling to consider scaling
        window.addEventListener('mousemove', (event) => {
            const rect = canvas.getBoundingClientRect();
            const scaleX = canvas.width / rect.width;
            const scaleY = canvas.height / rect.height;
            const x = Math.floor((event.clientX - rect.left) * scaleX / cellSize);
            const y = Math.floor((event.clientY - rect.top) * scaleY / cellSize);
            cells.setCellAt(x, y, true); // Mark the cell as alive
        });

        function loop() {
            if (!bfCtx || !vpCtx) return;
            const now = Date.now();
            const genDt = now - lastGen;
            const logoDt = now - lastLogo;

            if (genDt > generationTimeout) {
                lastGen = now;
                cells.nextGen();
            }
            if (logoDt > logoTimeout) {
                lastLogo = now;
                isLogoVisible = !isLogoVisible;
            } else if (isLogoVisible) {
                drawLogo(cells, 100, 12, Math.min((logoDt * 1.5) / logoTimeout, 1));
            }

            bfCtx.clearRect(0, 0, buffer.width, buffer.height);
            bfCtx.fillStyle = color;
            drawGen(bfCtx, cells, cellSize);
            vpCtx.putImageData(bfCtx.getImageData(0, 0, buffer.width, buffer.height), 0, 0);
            requestAnimationFrame(loop);
        }

        loop();
    }

    const canvas = document.getElementById('gameCanvas');
    const color = 'white';  // You can change the cell color here
    createAnimation({
        canvas: canvas,
        width: CW,
        height: CH,
        cellSize: S,
        generationTimeout: 300,
        logoTimeout: 4050,
        color: color
    });
</script>