bonk.io Mobile Mod

This script makes bonk.io playable on mobile.

// ==UserScript==
// @name         bonk.io Mobile Mod
// @namespace    http://tampermonkey.net/
// @version      1.2
// @description  This script makes bonk.io playable on mobile.
// @author       kitaesq
// @match        https://bonk.io/gameframe-release.html
// @icon         https://www.google.com/s2/favicons?sz=64&domain=bonk.io
// @grant        none
// ==/UserScript==
if (!window.kitaes) window.kitaes = {}
if (!window.kitaes.requestIntercept) {
    window.kitaes.requestIntercept = async () => {
        const send = XMLHttpRequest.prototype.send
        XMLHttpRequest.prototype.send = function(body){
            this.addEventListener("load", () => {
                const event = new Event("kitaes-request")
                event.body = body
                event.request = this
                window.dispatchEvent(event)
            })
            return send.apply(this, [body]);
        };
        console.log("Request interceptor loaded")
    }
    window.kitaes.requestIntercept()
}
if (!window.kitaes.fullscreen) {window.kitaes.fullscreen = async () => {
    console.log("[Fullscreen mod] Loading fullscreen mod...")
    const styleElem = document.createElement("style")
    styleElem.textContent =
`body{
    overflow: hidden;
    visibility: hidden;
}
#maingameframe{
    margin: 0 !important;
    position: fixed;
    top: 0 !important;
    left: 0 !important;
    right: 0 !important;
    top: 0 !important;
    visibility: visible;
}
#theme_container{
    top: 36px;
    visibility: visible;
}`
    top.document.head.append(styleElem)
    const fstyleElem = document.createElement("style")
    fstyleElem.className = "kitaes-fullscreen-style"
    fstyleElem.textContent =
`#bonkiocontainer{
    width: 100% !important;
    height: 100% !important;
    border: none !important;
}
canvas{
    position: fixed;
    top: 0;
    bottom: 0;
    left: 0;
    right: 0;
    margin: auto;
    pointer-events: none;
}
#xpbarcontainer{
    top: -4px !important
}
#fullscreen_button{
    background-image: url();
    background-repeat: no-repeat;
    background-position: center;
    position: absolute;
    top: 35px;
    left: 0;
    margin: 10px;
    width: 40px;
    height: 40px;
    pointer-events: auto;
}
*{
    touch-action: manipulation;
    user-select: none;
}
#newbonklobby_chat_content *{
    user-select: text;
}`
    document.head.append(fstyleElem)
    console.log("[Fullscreen mod] Style was injected")
    const fullscreenBtn = document.createElement("div")
    fullscreenBtn.id = "fullscreen_button"
    fullscreenBtn.className = "brownButton brownButton_classic buttonShadow"
    let isFullScreen = false
    fullscreenBtn.onclick = () => {
        if (!isFullScreen) {
            document.body.requestFullscreen()
        }
        else{
            document.exitFullscreen()
        }
        isFullScreen = !isFullScreen
    }
    prettymenu.append(fullscreenBtn)
    console.log("[Fullscreen mod] Fullscreen button was added")
    const interval = setInterval(() => {
        if (document.body.lastElementChild.tagName !== "DIV" || document.body.lastElementChild.className) return
        document.body.lastElementChild.style.top = "85px"
        clearInterval(interval)
        console.log("[Fullscreen mod] FPS counter was moved to the bottom")
        console.log("[Fullscreen mod] Fullscreen mod was loaded")
    }, 500)
}
window.kitaes.fullscreen()
}
if (!window.kitaes.mobile) {window.kitaes.mobile = async () => {
    console.log("loading mobile mod...")
    top.document.head.innerHTML += '<meta name="viewport" content="height=600px, initial-scale=0.5, maximum-scale=0.5">'
    if ("virtualKeyboard" in navigator) {
        navigator.virtualKeyboard.overlaysContent = true;
    }
    const fstyle = document.createElement("style")
    fstyle.textContent =
`#ingamechatbox{
bottom: 210px !important;
}`
    document.head.append(fstyle)
    const buttonContainer = document.createElement("div")
    Object.assign(buttonContainer.style, {
        position: "fixed",
        bottom: "25px",
        right: "25px",
        width: "225px",
        height: "225px",
        display: "flex",
        flexWrap: "wrap",
        lineHeight: "70px",
        fontSize: "40px"
    })
    const keybinds = {enter: 13}
    async function parseControls(base64){
        const x = "data:image/png;base64," + base64
        const arrayBuffer = await (await fetch(x)).arrayBuffer()
        const view = new DataView(arrayBuffer);
        keybinds.up = view.getUint16(2)
        keybinds.down = view.getUint16(6)
        keybinds.left = view.getUint16(10)
        keybinds.right = view.getUint16(14)
        keybinds.heavy = view.getUint16(18)
        keybinds.special = view.getUint16(22)
    }
    window.addEventListener("kitaes-request", async (e) => {
        if (e.request.responseURL === "https://bonk2.io/scripts/account_savecontrols.php"){
            const {controls} = Object.fromEntries(new URLSearchParams(e.body))
            await parseControls(controls)
        }
        if (!(e.request.responseURL === "https://bonk2.io/scripts/login_legacy.php" || e.request.responseURL === "https://bonk2.io/scripts/login_auto.php")) return
        const data = JSON.parse(e.request.responseText)
        await parseControls(data.controls)
    })
    const buttons = [["↖", "↑", "↗"],
                     ["←", " ", "→"],
                     ["↙", "↓", "↘"]]
    const activeDirKeys = {up: false, down: false, left: false, right: false}
    const activeSpecialKeys = {heavy: false, special: false}
    let isActive = false
    function simulateKey(key, down, element) {
        if (down === activeDirKeys[key]) return
        if (!element) element = document
        if (Object.keys(activeDirKeys).indexOf(key) !== -1) activeDirKeys[key] = down
        const event = document.createEvent("HTMLEvents");
	    event.initEvent("key" + (down ? "down" : "up"), true, false);
        event.keyCode = keybinds[key]
        element.dispatchEvent(event);
    }
    function simulateKeys(keys, type) {
        for (const key of keys){
            simulateKey(key, type)
        }
    }
    function releaseAllKeys(){
        for (const a of Object.keys(activeDirKeys)){
            if (!activeDirKeys[a]) continue
            simulateKey(a, false)
        }
    }
    buttonContainer.ontouchstart = (e) => {
        e = e.targetTouches[0]
        const ContainerPos = buttonContainer.getBoundingClientRect()
        const x = e.clientX - ContainerPos.x
        const y = e.clientY - ContainerPos.y
        if      (x < 74)  simulateKey("left", true)
        else if (x > 147) simulateKey("right", true)
        if      (y < 74)  simulateKey("up", true)
        else if (y > 147) simulateKey("down", true)
        isActive = true
        console.log("down")
    }
    oncontextmenu = e => e.preventDefault()
    window.ontouchend = (e) => {
        if (!isActive) return
        isActive = false
        console.log("up")
        if (e.changedTouches[0].target === buttonContainer){
            releaseAllKeys()
            isActive = false
        }
        else if (e.changedTouches[0].target.key){
            simulateKey(e.changedTouches[0].target.key, false)
        }
    }
    const gameOverlay = document.createElement("div")
    gameOverlay.addEventListener('contextmenu', e => e.preventDefault());
    gameOverlay.style.zIndex = 99999
    gameOverlay.style.position = "fixed"
    gamerenderer.append(gameOverlay)
    gameOverlay.append(buttonContainer)
    const bs = {height: "70px", width: "70px", pointerEvents: "none", marginBottom: "5px", marginRight: "5px"}
    for (let i = 0; i < 3; i++){
        for (let j = 0; j < 3; j++){
            if (j === 1 && i === 1) {
                const div = document.createElement("div")
                Object.assign(div.style, bs)
                buttonContainer.append(div)
                continue
            }
            const keys = []
            if      (j === 0) keys.push("left")
            else if (j === 2) keys.push("right")
            if      (i === 0) keys.push("up")
            else if (i === 2) keys.push("down")
            const button = document.createElement("div")
            Object.assign(button.style, bs)
            button.keys = keys
            button.textContent = buttons[i][j]
            button.className = "brownButton brownButton_classic buttonShadow"
            buttonContainer.append(button)
        }
    }
    buttonContainer.ontouchmove = (e) => {
        e = e.targetTouches[0]
        const ContainerPos = buttonContainer.getBoundingClientRect()
        const x = e.clientX - ContainerPos.x
        const y = e.clientY - ContainerPos.y
        releaseAllKeys()
        if      (x < 74)  simulateKey("left", true)
        else if (x > 147) simulateKey("right", true)
        if      (y < 74)  simulateKey("up", true)
        else if (y > 147) simulateKey("down", true)
    }
    const list = ["special", "heavy", "enter"]
    const leftStyle = {height: "70px", width: "200px", fontSize: "30px", lineHeight: "70px", textTransform: "capitalize", position: "fixed", left: "30px"}
    for (let i = 0; i < 3; i++){
        const button = document.createElement("div")
        button.className = "brownButton brownButton_classic buttonShadow"
        button.key = list[i]
        button.textContent = button.key
        Object.assign(button.style, leftStyle)
        if (list[i] === "enter"){
            button.textContent = "💬"
            Object.assign(button.style, {left: "unset", right: "260px", bottom: "30px", width: "70px", visibility: "visible"})
            button.onpointerdown = (e) => {
                e.preventDefault()
            }
            button.onclick = () => {
                simulateKey("enter", true, fdocument.activeElement)
            }
            gameOverlay.append(button)
            return
        }
        button.style.bottom = (30 + (i * 100)) + "px"
        button.ontouchstart = (e) => {
            isActive = true
            simulateKey(button.key, true)
        }
        gameOverlay.append(button)
    }
}
window.kitaes.mobile()}