// ==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!`);
})();