Io Record

This script is in beta testing !! Record any io game (agma.io only for now)

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         Io Record
// @namespace    http://tampermonkey.net/
// @version      0.0.10
// @author       Big watermelon
// @description  This script is in beta testing !! Record any io game (agma.io only for now)
// @match        *://agma.io/*
// @license      All Rights Reserved
// @icon         
// @grant        unsafeWindow
// @grant        GM_setValue
// @grant        GM_getValue
// @run-at       document-start
// ==/UserScript==

/*

Copyright © 2024 Big watermelon. All Rights Reserved.
This work is proprietary and may not be copied, distributed, or modified without explicit permission.

*/

// FIXIT: on death doesnt show death screen cuz packet 12 is useless and sora is dumb
// FIXIT: make changelog fully text dependable otherwise people scared I'll hack them via XSS
// FIXIT: why shit drop works 1/4 times (on my OS)
// FIXIT: allow moving if spec mode while watching record
// TODO: next frame -> to keydown to allow fastforward
/* TODO: maybe some shits to show
            Interval: 29794 ms
            Packets:       965
            Bytes:      370 Ko

console.log(`Interval: ${Math.round(record[record.length - 1].timeStamp - record[0].timeStamp)}ms\nPackets: ${record.length\nBytes: ${Math.round(record.map(x=>x.data.byteLength).reduce((a, b)=>a+b)/1000)}Ko`);`

*/
// TODO: visual indicator when viewing record ?
/* TODO: optimize files deflate

pako.deflate(new Uint8Array(clip.map(({offset, data}) => new Uint8Array([
    ...new Uint8Array(new Uint32Array([offset]).buffer),
    ...new Uint8Array(data)
])).reduce((a, b) => [...a, ...b], [])))

*/
// TODO: make files into binaries
// FIXIT: depending server some clips may do shit ?
/* FIXIT: I know why cells disapear and it does weird shit
          some packets are missing like 10, 11, 12, 32
          so game doesnt really understand the state its in
*/
// FIXIT: change extension name and clip to binaries [Uint32 "offset", ... "packet"]
// TODO: allow multi file drop -> means add keybind for next/prev clip

(function() {
    'use strict';

    if (unsafeWindow.top !== unsafeWindow.self || document.querySelector('title')?.textContent?.includes('Just a moment'))
        return;

    const settings = Object.assign({
        saveRecordKey: 'o',
        pauseKey: ' ',
        nextFrameKey: 'ArrowRight',
        escapeViewKey: 'Escape',
        recordAnimations: false,
        recordLeaderboard: false,
        recordMovingBorders: true,
        recordFor: 10000,
        fetchChangeLog: true
    }, GM_getValue('settings', {}));

    const SCRIPT_VERSION = GM_info?.script?.version;
    let clientVersion = 'Unknown';

    function versionAlert(clipVersion, errorDetail) {
        swal({
            title: 'Io Record',
            text: `<span style="color:red;font-weight:bold;">An error occured while loading this clip.</span>
            <br>
            <span>Clip Version: ${clipVersion}</span>
            <br>
            <span>Io Record Version: ${SCRIPT_VERSION},${clientVersion}</span>
            <br>
            <span>If your the script isn't up to date you may not be able to view some clips.</span>
            <br>
            <span style="font-size:12pt;">${errorDetail}</span>`,
            html: true,
            type: 'error'
        });
    }
    function versionDiff(a, b) { // a < b
        a = a.split('.');
        b = b.split('.');
        while (a.length && b.length) {
            if (Number(a.shift()) < Number(b.shift()))
                return true;
        }
        return false;
    }
    function serializeArray(frames) {
        return SCRIPT_VERSION + ',' + clientVersion + '\n' + frames.map(frame => Math.round(frame.timeStamp - frames[0].timeStamp) + ': ' + btoa(String.fromCharCode(...new Uint8Array(frame.data)))).join('\n');
    }
    function deserializeString(frames) {
        const [framesep, offsetsep] = frames.includes('\n') ? ['\n', ': '] : ['|', ':'];
        frames = frames.split(framesep);
        const [s, c] = (frames[0].includes(offsetsep) ? '0.0.8,ag255' : frames.shift()).split(',');
        try {            
            return frames.map(frame => {
                const [offset, base64Data] = frame.split(offsetsep);
                const binaryString = atob(base64Data);
                const buffer = new Uint8Array(binaryString.length);
                for (let i = 0; i < binaryString.length; i++)
                    buffer[i] = binaryString.charCodeAt(i);
                return { id: buffer[0], offset: Number(offset), data: buffer.buffer };
            });
        } catch (error) {
            if (versionDiff(s, SCRIPT_VERSION) || c < clientVersion)
                versionAlert(s + ',' + c, error.name + ': ' + error.message);
            throw error;
        }
    }

    const MANDATORY_PACKETS = [10, 11, 12, 20, 32, 33, 48, 49, 50, 65, 66],
          CLEAR_ALL = { data: new Uint8Array([20]).buffer },
          FAKE_CELL_UPDATE = { data: new Uint8Array([10, 0, 0, 0, 0, 0, 0]).buffer };

    const record = [];
    let isPaused = true,
        viewedRecord = null,
        wsOnmessage,
        frameIndex = 0;

    unsafeWindow.addEventListener('keydown', event => viewedRecord && Object.values(settings).includes(event.key) && event.stopImmediatePropagation());
    unsafeWindow.addEventListener('keyup', event => {
        if ($('input, textarea').is(':focus')) return;
        if (viewedRecord) {
            if (event.key == settings.nextFrameKey) {
                if (frameIndex >= viewedRecord.length) return;
                isPaused = true;
                let current;
                do {
                    wsOnmessage(current = viewedRecord[frameIndex++]);
                } while (frameIndex <= viewedRecord.length && current.id != 10);
            } else if (event.key == settings.pauseKey) {
                if (isPaused) {
                    isPaused = false;
                    goto();
                } else {
                    isPaused = true;
                }
            } else if (event.key == settings.escapeViewKey) {
                frameIndex = 0;
                wsOnmessage(CLEAR_ALL);
                viewedRecord = null;
            }
        } else if (event.key == settings.saveRecordKey) {
            const link = document.createElement("a");
            link.href = URL.createObjectURL(new Blob([serializeArray(record)], { type: 'text/plain' }));
            const d = new Date();
            link.download = `agma-clip.bin`;
            document.body.appendChild(link);
            link.click();
            document.body.removeChild(link);
        } else {
            return;
        }
        event.stopImmediatePropagation();
        event.preventDefault();
    });

    function recordPacket(message) {
        const { timeStamp, data } = message;
        const id = new DataView(data).getUint8(0);
        if (
            id == 11 && data.byteLength == 3
            || id == 12 && data.byteLength == 5
        )
            return;
        if (viewedRecord) {
            if (id == 10) return isPaused && wsOnmessage(FAKE_CELL_UPDATE);
            else if (id == 11 || id == 12 || id == 17) return; // prevents spectator movement
        }
        wsOnmessage(message);
        while (timeStamp - record[0]?.timeStamp > settings.recordFor)
            record.shift();
        if (
            viewedRecord
            || !settings.recordAnimations && id == 33
            || !settings.recordLeaderboard && [48, 49, 50].includes(id)
            || !settings.recordMovingBorders && [65, 66].includes(id)
            || !MANDATORY_PACKETS.includes(id)
        )
            return;
        record.push({ timeStamp, data });
    }

    function goto() {
        if (!viewedRecord) return;
        if (!frameIndex)
            wsOnmessage(CLEAR_ALL);
        const current = viewedRecord[frameIndex++];
        MANDATORY_PACKETS.includes(current.id) && wsOnmessage(current);
        if (isPaused)
            return;
        if (viewedRecord[frameIndex])
            setTimeout(goto, viewedRecord[frameIndex].offset - current.offset);
        else setTimeout(() => {
            frameIndex = 0;
            wsOnmessage(CLEAR_ALL);
            isPaused = true;
        }, 10);
    }

    const originalDefineProperty = unsafeWindow.Object.defineProperty;
    unsafeWindow.Object.defineProperty = function(obj, prop, descriptor) {
        if (obj instanceof WebSocket && obj.url.includes('.agma.io')) {
            obj.addEventListener('message', recordPacket);
            originalDefineProperty(obj, 'onmessage', {
                set: function(onmessage) { wsOnmessage = onmessage; },
                get: function() { return wsOnmessage; }
            });
        }
        return originalDefineProperty(obj, prop, descriptor);
    }

    /* Temporarily disabled

    const originalSend = unsafeWindow.WebSocket.prototype.send;
    unsafeWindow.WebSocket.prototype.send = function() {
        // this just prevents mouse position
        if (!arguments[0]?.getUint8(0) && viewedRecord != null) return;
        return originalSend.apply(this, arguments);
    }

    */

    Object.defineProperties(HTMLBodyElement.prototype, { // risky but mandatory
        ondragstart: { get: () => null, set: () => null, configurable: true },
        ondrop: { get: () => null, set: () => null, configurable: true },
        ondragenter: { get: () => null, set: () => null, configurable: true },
        ondragover: { get: () => null, set: () => null, configurable: true }
    });

    let loaded = false;
    unsafeWindow.addEventListener('load', () => {
        if (loaded || typeof swal == 'undefined') return;
        loaded = true;
        for (const { src } of document.scripts) {
            if (clientVersion = src.match("ag[0-9]+")?.[0])
                break;
        }
        if (settings.fetchChangeLog) {
            try {
                const xhr = new XMLHttpRequest();
                xhr.open('GET', 'https://api.github.com/repos/Grosse-pasteque/io-record/contents/CHANGELOG', true);
                xhr.setRequestHeader('Accept', 'application/json');
                xhr.setRequestHeader('Authorization', 'Bearer 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');
                xhr.setRequestHeader('X-GitHub-Api-Version', '2022-11-28');
                xhr.onreadystatechange = function () {
                    if (xhr.readyState === XMLHttpRequest.DONE) {
                        if (xhr.status === 200) {
                            const changelog = atob(JSON.parse(xhr.responseText).content);
                            if (changelog[0] != GM_getValue('changelog', '0')) {
                                GM_setValue('changelog', changelog[0])
                                swal({ title: '', text: changelog.slice(1), html: true })
                            }
                        } else {
                            console.error('Error:', xhr.status, xhr.statusText);
                        }
                    }
                };
                xhr.send();
            } catch (e) {
                console.error("IO-Record, couldn't fetch CHANGELOG:", e);
            }
        }
        const canvas = document.getElementById('canvas');
        canvas.addEventListener("dragenter", event => {
            event.preventDefault();
            canvas.style.border = '5px solid green';
        });
        canvas.addEventListener("dragover", event => event.preventDefault());
        canvas.addEventListener("dragleave", event => {
            if (event.target != canvas) return;
            event.preventDefault();
            canvas.style.border = '';
        });
        canvas.addEventListener('drop', event => {
            canvas.style.border = '';
            const file = event.dataTransfer.files[0];
            if (!file) return;
            const reader = new FileReader();
            reader.onload = e => {
                frameIndex = 0;
                isPaused = false;
                viewedRecord = deserializeString(e.target.result);
                goto();
            };
            reader.readAsText(file);
            event.preventDefault();
        });
        const settingPageId = Math.random() * 10e17;

        const settingTab = document.createElement('button');
        settingTab.id = 'settingTab' + settingPageId;
        settingTab.className = 'setting-tablink';
        settingTab.onclick = openSettingPage.bind(null, settingPageId);
        settingTab.innerText = 'Io Record';

        const settingPage = document.createElement('div');
        settingPage.id = 'settingPage' + settingPageId;
        settingPage.className = 'setting-tabcontent';
        settingPage.innerHTML = `
            <div class="col-md-10 col-md-offset-1 stng" style="padding-left:20px;padding-right:10px;max-height:550px;overflow:hidden auto;margin:0;width:calc(100% - 5px);">
                <span class="hotkey-paragraph"> Keybinds</span>
                <div class="row stng-row">
                    Save Record
                    <div class="hotkey-input" data-name="saveRecordKey">${settings.saveRecordKey}</div>
                    <br>
                    Pause View
                    <div class="hotkey-input" data-name="pauseKey">${settings.pauseKey}</div>
                    <br>
                    Next Frame
                    <div class="hotkey-input" data-name="nextFrameKey">${settings.nextFrameKey}</div>
                    <br>
                    Escape Viewed Record
                    <div class="hotkey-input" data-name="escapeViewKey">${settings.escapeViewKey}</div>
                </div>
                <span class="hotkey-paragraph"> Record</span>
                <div class="row stng-row">
                    Animations
                    <input type="checkbox" data-name="recordAnimations" ${settings.recordAnimations ? "checked" : ''}>
                    <br>
                    Leaderboard
                    <input type="checkbox" data-name="recordLeaderboard" ${settings.recordLeaderboard ? "checked" : ''}>
                    <br>
                    Moving borders
                    <input type="checkbox" data-name="recordMovingBorders" ${settings.recordMovingBorders ? "checked" : ''}>
                </div>
                <span class="hotkey-paragraph"> Other</span>
                <div class="row stng-row">
                    Records Length (seconds)
                    <input type="number" min="0" class="hotkey-input" style="outline:none;border:none;" value="${~~(settings.recordFor / 1000)}">
                    <br>
                    Fetch Change Log
                    <input type="checkbox" data-name="fetchChangeLog" ${settings.fetchChangeLog ? "checked" : ''}>
                </div>
            </div>
        `;

        function onchange() {
            settings[this.dataset.name] = this.checked;
        }
        function onclick() {
            this.classList.add('selected');
            const handle = event => {
                if (event.type === 'keyup')
                    settings[this.dataset.name] = this.innerText = event.key;
                this.classList.remove('selected');
                unsafeWindow.removeEventListener('mousedown', handle);
                unsafeWindow.removeEventListener('keyup', handle);
                event.preventDefault();
                event.stopPropagation()
            };
            unsafeWindow.addEventListener('mousedown', handle);
            unsafeWindow.addEventListener('keyup', handle)
        }
        function oncontextmenu(event) {
            this.innerText = '';
            settings[this.dataset.name] = null;
            event.preventDefault()
        }
        settingPage.querySelectorAll('input[type=checkbox]').forEach(input => input.onchange = onchange.bind(input));
        settingPage.querySelector('input[type=number]').onchange = event => settings.recordFor = 1000 * event.target.value;
        settingPage.querySelectorAll('div.hotkey-input').forEach(hotkey => {
            hotkey.oncontextmenu = oncontextmenu.bind(hotkey);
            hotkey.onclick = onclick.bind(hotkey);
        });

        const style = document.createElement('style');
        style.innerHTML = `
            #settingPage${settingPageId} input {
                position: absolute;
                display: inline-block;
                right: 0;
            }
            #settingPage${settingPageId} .hotkey-input {
                max-width: 90px;
            }
            #settingPage${settingPageId} > div::-webkit-scrollbar {
                width: 8px;
                height: 8px;
            }
            #settingPage${settingPageId} > div::-webkit-scrollbar-track {
                background: #282934;
                border-radius: 10px;
            }
            #settingPage${settingPageId} > div::-webkit-scrollbar-thumb {
                background-color: #df8500;
                border-radius: 10px;
                border: 2px solid #282934;
            }
        `;

        const setting = document.getElementById('setting');
        setting.firstElementChild.appendChild(settingTab);
        setting.appendChild(settingPage);
        document.body.appendChild(style);
    });
    unsafeWindow.onbeforeunload = () => void GM_setValue('settings', settings);
    console.log(`🎥 Io Record - ${SCRIPT_VERSION} loaded!`);
})();