Zoom hack, game speed change hack, and instant movement hack.
// ==UserScript== // @name Paper.IO Enhanced // @namespace - // @version 1.1.3 // @description Zoom hack, game speed change hack, and instant movement hack. // @author NotYou // @match *://paper-io.com/* // @match *://*.paper-io.com/* // @run-at document-start // @license GPL-3.0-or-later // @grant none // @icon  // ==/UserScript== (function() { const MIN_ZOOM = 0.5 const MAX_ZOOM = 3.5 const MIN_GAME_SPEED = 10 const MAX_GAME_SPEED = 1000 const DEBUG_LEVEL = 0 /* 0 - disabled; 1 - info; 2 - errors */ const API_PROPS = ['paperio2api', 'paper2'] class Utils { static minmax(n, min, max) { return Math.min(Math.max(n, min), max) } static debug(level, ...args) { let _level switch (level) { case 'log': _level = 0 break case 'error': _level = 1 break } if(DEBUG_LEVEL > _level) { console[level].apply(this, args) } } static waitForProperty(target, props) { let tries = 0 try { return new Promise((res, rej) => { let intervalId = setInterval(() => { tries++ let validProp = props.find(prop => target[prop]) if(validProp) { clearInterval(intervalId) return res(target[validProp]) } if(tries > 6) { rej('Timeout error, cannot find properties: "' + props + '" in target object.') } }, 1e3) }) } catch(e) { Utils.debug('error', 'waitForProperty', e) } } } class Hack { static init(paper_api) { const GAME = paper_api.game const CONFIG = GAME.config const SCENE = getScene() Utils.debug('log', this.name + ' initialized!') class GameSpeedHack { static init(maxGameSpeed, minGameSpeed) { window.addEventListener('wheel', e => { if(e.ctrlKey) { e.preventDefault() const isPositive = e.deltaY > 0 CONFIG.unitSpeed += isPositive ? -2.5 : 2.5 const current = CONFIG.unitSpeed CONFIG.unitSpeed = Utils.minmax(current, minGameSpeed, maxGameSpeed) } }) } } class ZoomHack { static init(maxZoom, minZoom) { window.addEventListener('wheel', e => { if(!e.ctrlKey) { const isPositive = e.deltaY > 0 SCENE.maxScale += isPositive ? -0.5 : 0.5 const current = SCENE.maxScale SCENE.maxScale = Utils.minmax(current, minZoom, maxZoom) } }) } } class InstantMovement { static init(listenersData) { listenersData.forEach(data => { try { window.addEventListener('keydown', e => { try { if(e.code === data.code) { const PLAYER = GAME.player PLAYER.position[data.direction] += 10 * (data.isNegative ? -1 : 1) } } catch(err) { Utils.debug('error', data, err) } }) } catch(e) { Utils.debug('error', this.name, data, e) } }) } } GameSpeedHack.init( MAX_GAME_SPEED, MIN_GAME_SPEED ) ZoomHack.init( MAX_ZOOM, MIN_ZOOM ) InstantMovement.init([{ code: 'KeyW', direction: 'y', isNegative: true, }, { code: 'KeyS', direction: 'y', isNegative: false, }, { code: 'KeyA', direction: 'x', isNegative: true, }, { code: 'KeyD', direction: 'x', isNegative: false, }]) function getScene() { if(paper_api.currentConfig) { return paper_api.currentConfig } if(paper_api.configs) { return paper_api.configs.paper2_classic } return paper_api.config } } } class Main { static init() { Utils.waitForProperty(window, API_PROPS).then(paperapi => { Utils.waitForProperty(paperapi, ['game']).then(() => { Hack.init(paperapi) }) }) } } Main.init() })()