Io Record

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

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 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!`);
})();