// ==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 // @downloadURL none // ==/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 = ` Io Record
${savedRecords.map(record => '' + record.name + '').join('')}

0s

`; 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 = ''; 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 = '

Hide player

'; 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;' ); })();