// ==UserScript==
// @name Io Record
// @namespace http://tampermonkey.net/
// @version 0.0.8
// @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: no death screen
// FIXIT: gets disconnected when close window
(function() {
'use strict';
if (unsafeWindow.top !== unsafeWindow.self || document.querySelector('title')?.textContent?.includes('Just a moment'))
return;
const settings = {
toggleKey: 'm',
toggleMouseLock: '"',
recordAnimations: false,
recordLeaderboard: false,
recordMovingBorders: true,
autoStopAfter: 0, // seconds
fetchChangeLog: true
};
function serializeArray(recordData) {
return recordData.map(frame => Math.round(frame.offset) + ':' + btoa(String.fromCharCode(...new Uint8Array(frame.data)))).join("|");
}
function deserializeString(recordData) {
return recordData.split("|").map(frame => {
const [offset, base64Data] = frame.split(":");
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 };
});
}
const savedRecords = GM_getValue('records', []),
toggleMouseCode = JSON.parse(localStorage.hotkeys).C.c;
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 };
savedRecords.forEach(record => record.data = deserializeString(record.data))
let currentRecord = null,
clipPlayer = null,
start,
recordButton,
isPaused = true,
viewedRecord = null,
isMouseLocked = false,
wsOnmessage,
frameIndex = 0,
menuDom = document.createElement('html');
menuDom.innerHTML = `
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Io Record</title>
<style>
body {
width: 100vh;
height: 100vh;
background-color: #373737;
overflow: hidden;
}
svg {
cursor: pointer;
}
span {
background: #6f6f6f;
padding: 6px 12px;
text-align: center;
font-weight: bold;
border-radius: 10px;
margin: 5px;
cursor: pointer;
}
</style>
</head>
<body>
<div style="max-height: calc(100% - 70px); overflow: hidden visible; display: flex; width: 100%; position: absolute; left: 0; justify-content: center; flex-direction: column;">
${savedRecords.map(record => '<span>' + record.name + '</span>').join('')}
</div>
<div style="display: flex; position: absolute; bottom: 12px; left: 50%; transform: translate(-50%, 0);">
<svg title="Previous frame" width="50" height="50" viewBox="0 0 24 24">
<polygon points="19,5 9,12 19,19"/>
<rect x="5" y="5" width="2" height="14"/>
</svg>
<svg title="Play" width="50" height="50" viewBox="0 0 24 24">
<polygon points="8,5 19,12 8,19"/>
</svg>
<svg title="Pause" style="display:none;" width="50" height="50" viewBox="0 0 24 24">
<rect x="6" y="5" width="4" height="14"/>
<rect x="14" y="5" width="4" height="14"/>
</svg>
<svg title="Next frame" width="50" height="50" viewBox="0 0 24 24">
<polygon points="5,5 15,12 5,19"/>
<rect x="17" y="5" width="2" height="14"/>
</svg>
<svg title="Download clip" width="50" height="50" viewBox="0 0 24 24" fill="currentColor">
<path d="M12 16L12 4M8 12L12 16L16 12" stroke="currentColor" stroke-width="2" fill="none"/>
<line x1="4" y1="20" x2="20" y2="20" stroke="currentColor" stroke-width="2" />
</svg>
</div>
<input type="range" value="0" min="0" max="20000" style="width: calc(100% - 50px); position: absolute; bottom: 0; left: 0;">
<p style="font-size: 18px; margin: 0; width: 40px; position: absolute; bottom: 0; right: 0;">0s</p>
</body>
`;
const clipsList = menuDom.lastElementChild.firstElementChild;
const [previousFrameButton, playButton, pauseButton, nextFrameButton, downloadButton, timeControlInput, timeText] = menuDom.querySelectorAll('svg, input, p');
playButton.onclick = pauseButton.onclick = () => {
if (viewedRecord == null) return;
if (isPaused) {
isPaused = false;
goto();
} else {
isPaused = true;
}
updateButtonsStyle();
};
previousFrameButton.onclick = () => {
// This feature is glitchy
/*
if (viewedRecord == null || !frameIndex) return;
isPaused = true;
const recordData = savedRecords[viewedRecord].data;
let current;
do {
wsOnmessage(current = recordData[frameIndex--]);
} while (frameIndex && current.id != 10);
updateButtonsStyle();
timeText.innerText = Math.round((timeControlInput.max - (timeControlInput.value = recordData[frameIndex].offset)) / 1000) + 's';
*/
};
nextFrameButton.onclick = () => {
const recordData = savedRecords[viewedRecord].data;
if (viewedRecord == null || frameIndex >= recordData.length) return;
isPaused = true;
let current;
do {
wsOnmessage(current = recordData[frameIndex++]);
} while (frameIndex <= recordData.length && current.id != 10);
updateButtonsStyle();
timeText.innerText = Math.round((timeControlInput.max - (timeControlInput.value = recordData[frameIndex].offset)) / 1000) + 's';
};
timeControlInput.onchange = () => {
if (viewedRecord == null) return;
timeText.innerText = Math.round((timeControlInput.max - timeControlInput.value) / 1000) + 's';
const recordData = savedRecords[viewedRecord].data;
isPaused = true;
frameIndex = 0;
wsOnmessage(CLEAR_ALL);
let current;
do {
wsOnmessage(current = recordData[frameIndex++]);
} while (current.offset < timeControlInput.value);
isPaused = true;
updateButtonsStyle();
};
clipsList.onclick = ({ target }) => {
const clips = Array.from(clipsList.children);
const i = clips.indexOf(target);
isPaused = true;
if (viewedRecord != null)
clips[viewedRecord].style.border = '';
if (target.tagName != 'SPAN' || viewedRecord == i) {
viewedRecord = null;
return;
}
frameIndex = 0;
wsOnmessage(CLEAR_ALL);
target.style.border = '1px solid white';
const recordData = savedRecords[viewedRecord = i].data;
timeControlInput.max = recordData[recordData.length - 1].offset;
timeControlInput.value = 0;
timeText.innerText = Math.round(timeControlInput.max / 1000) + 's';
};
const dropZone = menuDom.lastElementChild;
dropZone.ondragenter = dropZone.ondragover = dropZone.ondragleave = event => event.preventDefault();
dropZone.addEventListener('drop', event => {
event.preventDefault();
const file = event.dataTransfer.files[0];
if (file) {
const reader = new FileReader();
reader.onload = e => {
const d = new Date();
savedRecords.push({ name: file.name, data: deserializeString(e.target.result) });
const span = document.createElement('span');
span.innerText = file.name;
clipsList.appendChild(span);
};
reader.readAsText(file);
}
});
downloadButton.addEventListener('click', () => viewedRecord != null && unsafeWindow.open(
URL.createObjectURL(new Blob([serializeArray(savedRecords[viewedRecord].data)], { type: 'text/plain' }))));
unsafeWindow.addEventListener('keyup', event => {
event.stopPropagation();
event.preventDefault();
if (event.key == settings.toggleKey && event.ctrlKey) {
if (currentRecord) {
endCurrentRecord();
} else {
const d = new Date();
savedRecords.push(currentRecord = {
name: `Clip ${d.getFullYear()}-${d.getMonth()}-${d.getDate()}_${d.getHours()}-${d.getMinutes()}-${d.getSeconds()}`,
data: []
/* , game: 'agma.io' */
});
settings.autoStopAfter && setTimeout(endCurrentRecord, settings.autoStopAfter * 1000);
start = performance.now();
recordButton.firstElementChild.style.color = 'red';
}
} else if (event.key == settings.toggleMouseLock) {
isMouseLocked = !isMouseLocked;
unsafeWindow[isMouseLocked ? 'onkeydown' : 'onkeyup']({ keyCode: toggleMouseCode });
}
});
function endCurrentRecord() {
if (!currentRecord) return;
recordButton.firstElementChild.style.color = '';
const span = document.createElement('span');
span.innerText = currentRecord.name;
clipsList.appendChild(span);
currentRecord = null;
}
function updateButtonsStyle() {
if (isPaused) {
recordButton.firstElementChild.style.color = '';
pauseButton.style.display = 'none';
playButton.style.display = 'block';
} else {
recordButton.firstElementChild.style.color = 'green';
playButton.style.display = 'none';
pauseButton.style.display = 'block';
}
}
function recordPacket(message) {
const data = new DataView(message.data);
const id = data.getUint8(0);
if (
id == 11 && data.byteLength == 3
|| id == 12 && data.byteLength == 5
)
return;
if (viewedRecord != null) {
if (id == 10) return isPaused && wsOnmessage(FAKE_CELL_UPDATE);
else if (id == 11 || id == 12) return;
}
wsOnmessage(message);
if (
currentRecord
&& viewedRecord == null
&& (settings.recordAnimations || id != 33)
&& (settings.recordLeaderboard || id < 48 && id > 50)
&& (settings.recordMovingBorders || id != 65 && id != 66)
&& MANDATORY_PACKETS.includes(id)
)
currentRecord.data.push({ id, data: message.data, offset: message.timeStamp - start });
}
function goto() {
if (viewedRecord == null) return;
if (!frameIndex)
wsOnmessage(CLEAR_ALL);
const recordData = savedRecords[viewedRecord].data;
const current = recordData[frameIndex++];
timeText.innerText = Math.round((timeControlInput.max - (timeControlInput.value = current.offset)) / 1000) + 's';
MANDATORY_PACKETS.includes(current.id) && wsOnmessage(current); // prevents XSS -> forbids unallowed packets
if (isPaused)
return;
if (recordData[frameIndex])
setTimeout(goto, recordData[frameIndex].offset - current.offset);
else setTimeout(() => {
frameIndex = 0;
wsOnmessage(CLEAR_ALL);
isPaused = true;
updateButtonsStyle();
}, 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);
}
originalDefineProperty(unsafeWindow, 'onblur', { set: () => 0 }); // prevents mouse lock to be reset
const originalSend = unsafeWindow.WebSocket.prototype.send;
unsafeWindow.WebSocket.prototype.send = function() {
if (viewedRecord != null) return;
return originalSend.apply(this, arguments);
}
let loaded = false;
unsafeWindow.addEventListener('load', () => {
if (loaded || typeof swal == 'undefined') return;
loaded = true;
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);
}
}
recordButton = document.createElement('div');
recordButton.id = 'recordButton';
recordButton.className = 'btn';
recordButton.style = 'right: 220px;';
recordButton.title = `Press <${settings.toggleKey.toUpperCase()}> to record.`;
recordButton.innerHTML = '<i class="fa fa-video-camera fa-lg" style=" margin-top: 8px;"></i>';
recordButton.onclick = function() {
if (clipPlayer && !clipPlayer.closed)
return clipPlayer.focus();
clipPlayer = unsafeWindow.open('', 'subWindow', 'width=300,height=400');
if (!clipPlayer)
return;
clipPlayer.document.open();
clipPlayer.document.appendChild(clipPlayer.document.adoptNode(menuDom));
clipPlayer.document.close();
clipPlayer.addEventListener('unload', () => {
viewedRecord = null;
isPaused = true;
updateButtonsStyle();
menuDom = clipPlayer.document.documentElement;
clipPlayer = null;
});
clipPlayer.focus();
}
document.body.appendChild(recordButton);
const style = document.createElement('style');
style.innerHTML = `
#recordButton {
transition: color .2s;
position: absolute;
top: 10px;
background: #373737;
padding: 7px;
z-index: 135;
opacity: .7;
}
#recordButton:hover {
color: #eee;
opacity: 1;
}
`;
document.head.appendChild(style);
const contextSpectate = document.getElementById("contextSpectate")
const contextHidePlayer = document.createElement('li');
contextHidePlayer.id = 'contextHidePlayer';
contextHidePlayer.className = 'contextmenu-item enabled';
contextHidePlayer.style = 'display: none;';
contextHidePlayer.innerHTML = '<div class="fa fa-eye-slash fa-2x context-icon"></div><p>Hide player</p>';
contextSpectate.parentElement.insertBefore(contextHidePlayer, contextSpectate);
const originalCss = $.prototype.css;
$.prototype.css = function() {
if (this.selector == '#btnFriends')
recordButton.style.right = parseInt(arguments[1]) + 46 + 'px';
return originalCss.apply(this, arguments)
};
setInterval(() => document.body.onmousemove({}), 250000); // anti afk to prevent FIXIT N°2
});
unsafeWindow.onbeforeunload = () => {
clipPlayer?.close();
// GM_setValue('settings', settings);
savedRecords.forEach(record => record.data = serializeArray(record.data))
GM_setValue('records', savedRecords);
};
console.log(`%cIo %cRecord %c- ${GM_info?.script?.version} Loaded`,
'font-weight: bold; font-size: 14pt; color: red;',
'font-weight: bold; font-size: 14pt; color: white;',
'font-weight: bold; font-size: 14pt; color: black;'
);
})();