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