Embedded diep lobby selector

A lobby selector and lobby API info formatter + other useful overlays

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         Embedded diep lobby selector
// @namespace    http://tampermonkey.net/
// @version      4.0 beta
// @description  A lobby selector and lobby API info formatter + other useful overlays
// @author       <Eclipsia discord@cz_eclipsia reddit@Ok-Wing4342>
// @icon         
// @match        *://*.diep.io/*
// @license      MIT
// @grant        none
// @run-at       document-start
// ==/UserScript==

/* 
TODO: 
make the inner div contained in a space (done)
stop execution if display is none (done)
add a way to choose a team in teams and 4teams (done)
edit css of buttons (done)
test future events and beta data (done)
game version button switcher mostly done ready to test (done)
add options to enable and disable render latency and net predict (done)

add special to sandboxes
highlighted mode now matches the colour of your team
same with score and background

// https://stackoverflow.com/questions/79614720/how-do-i-add-hover-and-active-events-to-my-tampermonkey-injected-script
*/

(async function() {
    'use strict';
    // region Top
    const TOGGLE_LOBBY_KEYBIND = 'q';
    const TOGGLE_PRESETS_KEYBIND = 'r';
    const UPDATE_DELAY = 1000; // miliseconds
    const BASE_API = "https://lb.diep.io/api/lb/"
    const BETA_API = "https://master-dev.diep.io/api/lb/"
    const storageKey = "diepioLobbyselect_"
    //const re_isBeta = /https?:\/\/(?<beta>beta\.)?diep\.io\/?.*/

    /**
     * Return the number of appearances of sub in orig.
     * @param {string} orig 
     * @param {string} sub 
     */
    const count = (orig, sub) => orig.split(sub).length - 1;

    /** @param {string} timeAlive */
    function deformatTimeAlive(timeAlive) {
        let totalMins = 0;
        for (const part of timeAlive.split(" ")) {
            let [,value, suffix] = part.match(/(\d+)(h|m|s)/);
            value = parseInt(value);
            switch (suffix) {
                case "h": totalMins += value * 3600
                case "m": totalMins += value 
                case "s": totalMins += value / 60
            }
        }
        return totalMins;
    }

    // region default builds
    class Builder {
        constructor() {
            this.build = "";
        }

        get build2s3() {
            return this.add("32323")
        }

        repeat(parts, count) {
            return this.add(parts.repeat(count));
        }

        max(parts) {
            return this.repeat(parts, 7)
        }

        add(parts) {
            this.build += parts;
            return this;
        }

        finish() {
            return this.toString();
        }

        toString() {
            return this.build.slice(0, 33);
        }
    }

    const defaultBuilds = {
        overlord: new Builder().max("4568").build2s3.finish(),
        factory: new Builder().max("4568").max("7").finish(),
        triplet: new Builder().max("5678").max("4").finish(),
        fighter: new Builder().max("5678").build2s3.finish(),
        annihilator: new Builder().repeat("4568", 6).add("48").max("7").finish(),
        sniper: new Builder().repeat("4568", 6).add("56").max("7").finish()
    }

    const tankToIndex = {
        overlord: 12,
        factory: 52,
        fighter: 24,
        triplet: 2,
        annihilator: 49,
        sniper: 6,
    }

    function getTankIconLinkFor(tank) {
        return `https://diep.io/old-assets/assets/diep/tanks/tank_${tankToIndex[tank.toLowerCase()]}.png`;
    }
    
    // region local storage management
    const storage = {
        get(key) {
            return localStorage.getItem(storageKey + key)
        },
        
        async set(key, value) {
            localStorage.setItem(storageKey + key, value);
            //console.log('set', key, 'to', value);
        },

        getBool(key, defaultValue) {
            let value = this.get(key, null);
            return value === null ? defaultValue : JSON.parse(value)
        },
        
        /**
         * @returns {number}
         */
        getNumber(key, defaultValue)  {
            let value = this.get(key, null);
            return value === null ? defaultValue : parseFloat(value);
        },

    }

    // region unfuck diep
    const consoleLog = console.log;
    const PREDICT_MOVEMENT = "net_predict_movement";
    const REN_LATENCY = "ren_latency";
    const REN_FPS = "ren_fps";

    let lobbyid;
    let [
        backgroundSuccess, 
        windowSuccess, 
        scoreSuccess, 
        checkedLobbyLink
    ] = [false, false, false, true];

    let renderLatency = storage.getBool("renLatency") ?? true;
    let predictMovement = storage.getBool("predictMovement") ?? false;
    let renderFPS = storage.getBool("renderFPS") ?? true;
    storage.set("predictMovement", predictMovement);
    storage.set("renLatency", renderLatency);
    storage.set("renFPS", renderFPS);

    const clipboardWrite = navigator.clipboard.writeText;
    navigator.clipboard.writeText = async (text) => {
        lobbyid = text.split("_")[3];
        navigator.clipboard.writeText = clipboardWrite;
    }

    // region async startup()
    async function startup() {
        if (!windowSuccess && window.input) {
            // force predict_movement off, bring back latency statistic 
            window.input.set_convar(REN_LATENCY, renderLatency);
            window.input.set_convar(PREDICT_MOVEMENT, predictMovement);
            window.input.set_convar(REN_FPS, renderFPS);
            windowSuccess = true;
        }

        let backgroundElem;
        if (!backgroundSuccess && (backgroundElem = document.getElementById("game-over-screen"))) {
            // remove blurry deathscreen background
            backgroundElem.style.setProperty("backdrop-filter", "none", "important");
            backgroundSuccess = true;

        }
        let scoreTextElem;
        if (!scoreSuccess && (scoreTextElem = document.getElementById("game-over-stats-player-score"))) {
            const timeAliveElem = document.getElementById("game-over-stats-time-alive");
            // unfuck the score
            let oldValue = scoreTextElem.textContent;
            const callback = () => {
                let newValue = scoreTextElem.textContent;
                if (newValue == oldValue) return;

                scoreTextElem.textContent = scoreTextElem.textContent.replace(/\B(?=(\d{3})+(?!\d))/g, ",")
                const score = parseInt(newValue.replaceAll(/[,.]/g, ""));
                const timeAliveMins = deformatTimeAlive(timeAliveElem.textContent);
                timeAliveElem.textContent += ` ${(score/timeAliveMins).toFixed().replace(/\B(?=(\d{3})+(?!\d))/g, ",")}/m`
                oldValue = newValue;
            }

            new MutationObserver(callback).observe(scoreTextElem, {
                childList: true,    
                subtree: true,
                characterData: true
            })

            scoreSuccess = true
        }

        //let copyLinkButton;
        //if (!checkedLobbyLink && (copyLinkButton = document.querySelector("#copy-party-link"))) {
        //    copyLinkButton.click();
        //    checkedLobbyLink = true;
        //}

        if (backgroundSuccess && windowSuccess && scoreSuccess && checkedLobbyLink) {
            consoleLog(`Finished unfucking job.`)
            return;
        }

        await Promise.resolve(new Promise(r => setTimeout(r, 500)));
        startup();
    };
    startup();

    const setBuild = build => input.execute(`game_stats_build ${build}`);
    
    // region Colours
    const intToTeam = [
        "blue",
        "red",
        "purple",
        "green",
    ];
    
    const intToColour = [
        "rgba(33, 143, 221, 1)", // blue
        "rgba(232, 27, 27, 1)",  // red
        "rgba(184, 13, 207, 1)", // purple
        "rgba(81, 220, 34, 1)",  // green
    ];

    const GREEN = "rgba(0, 255, 0, 1)"
    const RED = "rgba(255, 0, 0, 1)"

    let colours = {
        fra: "rgba(255, 0, 0, 0.5)",
        atl: "rgba(255, 255, 0, 0.5)",
        sgp: "rgba(50, 50, 200, 0.5)",
        syd: "rgba(0, 255, 0, 0.5)"
    }

    const defaultColour = "rgba(255, 255, 255, 0.5)"

    // region CSS
    const globalCSS = document.createElement("style");
    globalCSS.textContent = `

.lobby-button {
    padding: 5px;
    border: 1px solid black;
    background-color: rgba(255, 255, 255, 0.5);
    cursor: pointer;
    transition: background-color 0.2s ease;
    margin-left: 5px;
    margin-bottom: 2px;
    border-radius: 5px;
}

.button-row-div {
    z-index: 999999;
    border: 1px solid black;
    border-radius: 5px;
    display: flex;
    height: 20px;
}

.button-label {
    margin-top: 0px;
    margin-left: 3px;
}

.presets-header-button {
    margin: 10px;
    background-color: rgba(255, 255, 255, 1);
    cursor: pointer;
}

.presets-header-button:hover {
    background-color: rgba(160, 160, 160, 0.5);
}

.presets-header-button:active {
    background-color: rgb(0, 0, 0);
}

`;
    document.head.append(globalCSS);

    // region functions

    function offsetRGBA(rgba = "rgba(0, 0, 0, 0)", offset = 0, alphaOffset = 0.0) {
        const regex = /rgba\((?<r>\d{1,3})\D*,\D*(?<g>\d{1,3})\D*,\D*(?<b>\d{1,3})\D*,\D*(?<a>[0-1](\.\d)?)\)/
        const res = regex.exec(rgba);
        if (!res?.groups) throw Error(`failed regex: '${rgba}'`);

        const grps = res.groups;
        const r = parseInt(grps.r) + offset;
        const g = parseInt(grps.g) + offset;
        const b = parseInt(grps.b) + offset;
        const a = parseFloat(grps.a) + alphaOffset;
        return `rgba(${r}, ${g}, ${b}, ${a})`
    }

    const commonlySharedButtonCSS = `
        padding: 5px;
        border: 1px solid black;
        background-color: rgba(255, 255, 255, 0.5);
        cursor: pointer;
        transition: background-color 0.2s ease;
        margin-left: 5px;
        margin-bottom: 2px;
        border-radius: 5px;
    `;

    // TODO: sandboxes
    const thisURL = new URL(document.URL)
    let currentLobby = thisURL.searchParams.get("lobby")

    // region storage get


    /** @type {"verbose" | "concise"} */
    let layoutStyle = storage.get("layoutStyle") ?? "verbose"
    let buttonIndex = storage.getNumber('teamIndex', 0);
    let apiDevice = storage.get('device') || (thisURL.host.startsWith("mobile") ? "mobile" : "pc");
    /** @type {Object.<string, string>} */
    let presetTankBuilds = JSON.parse(storage.get("presetTankBuilds")) || defaultBuilds;
    let showUI = true;

    storage.set("layoutStyle", layoutStyle);
    storage.set("teamIndex", buttonIndex);
    storage.set("device", apiDevice);

    /** @type {"beta" | "main"} */
    let APItarget = thisURL.host.startsWith("beta") ? "beta" : "main"
    if (APItarget == "main") {
        APItarget ??= storage.get('APItarget');
    }

    // region helper funcs

    function getAPIEndpoint(device) {
        return (APItarget == "beta" ? BETA_API : BASE_API) + (device ?? apiDevice)
    }

    function getTextForNumPlayers(numPlayers) {
        // this function was vibe coded
        if (numPlayers < 850) return "Very Low";
        if (numPlayers < 900) return "Low";
        if (numPlayers < 950) return "Below Average";
        if (numPlayers < 1050) return "Average";
        if (numPlayers < 1100) return "Above Average";
        if (numPlayers < 1200) return "High";
        return "Very High";
    }

    function formatSeconds(seconds) {
        const days = Math.floor(seconds / (3600 * 24));
        const hours = Math.floor(seconds / 3600 % 24);
        const minutes = Math.floor(seconds / 60 % 60);
        seconds = Math.floor(seconds % 60);

        let output = "";
        if (days) output += `${days}d `
        if (hours) output += `${hours}h `
        if (minutes) output += `${minutes}m `
        if (seconds) output += `${seconds}s`
        return output.trim();
    }

    // region createLobbyButton
    function createLobbyButton(region, lobby) {
        const button = document.createElement('button');
        button.classList.add("lobby-button")
;
        let gamemodeName;
        if (layoutStyle == "verbose") {
            gamemodeName = lobby.gamemodeName || lobby.gamemode;
        } else {
            gamemodeName = lobby.gamemode;
            if (
                lobby.gamemodeName.toLowerCase().includes("maze") 
            ) gamemodeName = lobby.gamemodeName.toLowerCase();
        }
            
        button.textContent = `${gamemodeName} ${lobby.numPlayers}p`;

        let futureEventButton;
        if (lobby.nextGamemodeName) {
            futureEventButton = document.createElement("button")
            futureEventButton.classList.add("lobby-button");
            const secondsUntilText = formatSeconds(lobby.secondsUntilNextGamemode);
            futureEventButton.textContent = 
            `${lobby.gamemodeName ?? lobby.gamemode} => ${lobby.nextGamemodeName} in ${secondsUntilText}`
        }

        const buttonProxyHandler = {
            set(target, p, newValue, receiver) {
                button.style[p] = newValue
                if (futureEventButton) futureEventButton.style[p] = newValue
                return true
            },
        }

        const bothButtonStyle = new Proxy({}, buttonProxyHandler);

        const bgColor = colours[region.region] || defaultColour
        button.style.backgroundColor = bgColor;
        
        if (lobby.ip === currentLobby) {
            bothButtonStyle.border = "5px solid black";

            if (lobby.gamemode.includes("teams") || currentTeamIndex > 0) {
                bothButtonStyle.backgroundColor = intToColour[currentTeamIndex];
                button.textContent += ` (${intToTeam[currentTeamIndex]})`;
            } else {
                bothButtonStyle.backgroundColor = offsetRGBA(bgColor, -40)
            }

        } else {
            bothButtonStyle.border = "1px solid black";
        }

        if (futureEventButton) {
            futureEventButton.style.backgroundColor = offsetRGBA(bgColor, 120)
        }

        return {button, futureEventButton};
    }

    const empty = document.createElement("h2");
    empty.textContent = "< No Lobbies >";
    empty.style.cssText = `
        margin: 20px;
    `;
    // region async update()

    async function update() {
        const response = await fetch(getAPIEndpoint());
        const value = await response.json();
        /** @type {object[]} */
        const data = value.regions;

        const count = data.map(region => region.numPlayers).reduce((prv, cur) => prv + cur, 0);
        numPlayers.textContent = `Players Active: ${count} (${getTextForNumPlayers(count)})`        

        lobbySelectorDiv.textContent = "";
        if (!count) {
            lobbySelectorDiv.appendChild(empty)
            return
        }

        for (const region of data) {
            const countryCode = region.countryCode && layoutStyle == "verbose" ? ` (${region.countryCode})` : ""
            
            const playerCount = document.createElement('p');
            playerCount.textContent = `${region.regionName}${countryCode} ${region.numPlayers} Players`
            playerCount.style.cssText = `
                margin-right: 0px;
                background-color: ${colours[region.region] ?? defaultColour};
            `;
            playerCount.style.color = offsetRGBA(playerCount.style.backgroundColor, 200)
            lobbySelectorDiv.appendChild(playerCount);
            
            const futureEventButtons = [];

            for (let index = 0; index < region.lobbies.length; index++) {
                // put every sixth lobby on a new row
                if (index % 5 == 1 && index > 5) {
                    lobbySelectorDiv.appendChild(document.createElement('br'));
                }

                const lobby = region.lobbies[index];
                const {button, futureEventButton} = createLobbyButton(region, lobby);

                button.addEventListener('click', () => {
                    let teamIndex = buttonIndex;

                    if (lobby.gamemode === "teams" 
                        && teamIndex > 1 
                        || !lobby.gamemode.includes("teams")
                        && lobby.gamemode !== "event"
                    ) {
                        teamIndex = 0;
                    }
                    // https://diep.io/?lobby=fra_teams_fra-43ce658e19c456d0.diep.io:2001_55544_0
                    const searchParams = new URLSearchParams();
                    searchParams.set("lobby", `${region.region}_${lobby.gamemode}_${lobby.ip}_${"x" || "lobbyid"}_${teamIndex}`)

                    window.open(`https://${apiDevice == 'mobile' ? 'mobile.' : ''}diep.io/?${searchParams.toString()}`)
                });

                lobbySelectorDiv.appendChild(button);
                if (futureEventButton) futureEventButtons.push(futureEventButton);
            }
            if (layoutStyle == "verbose") {
                lobbySelectorDiv.appendChild(document.createElement('br'));
                futureEventButtons.forEach(button => lobbySelectorDiv.appendChild(button))
            }
            lobbySelectorDiv.appendChild(document.createElement('br'));
        }
        
    }
    update()
    let IntervalID = setInterval(update, UPDATE_DELAY);
    
    // toggle display visibility
    async function toggleLobby() {
        const display = lobby_container.style.display;
        lobby_container.style.display = display == 'block' ? 'none' : 'block';
        showUI = !showUI;
        if (showUI) {
            update();
            IntervalID = setInterval(update, UPDATE_DELAY);
        } else {
            clearInterval(IntervalID);
        }
    }

    // region HTML








    // region lobby selector






    // seperate each element with 2 empty lines
    // region text

    const lobby_container = document.createElement("div");
    lobby_container.style.cssText = `
        position: absolute;
        left: 0%;
        top: 0%;
        z-index: 999999;
        background-color: rgba(255, 255, 255, 0.5);
        padding: 5px;
        border: 1px solid black;
        border-radius: 5px;
        display: block;
    `;
    document.body.append(lobby_container);


    const title = document.createElement("p");
    title.textContent = 'Diep lobby selector by Eclipsia';
    title.style.cssText = `
        position: relative;
        z-index: 999999;
        padding: 0px;
        margin: 0px;
        color: black;
    `;
    lobby_container.appendChild(title);


    const numPlayers = document.createElement('p');
    numPlayers.textContent = "Players Active: ?";
    numPlayers.style.cssText = `
        position: relative;
        z-index: 999999;
        padding: 0px;
        margin: 0px;
        margin-bottom: 3px;
        color: black;
    `;
    lobby_container.appendChild(numPlayers);




    // region button row
    const buttonDiv = document.createElement('div');
    buttonDiv.style.cssText = `
        right: 0%;
        display: flex;
    `;
    lobby_container.appendChild(buttonDiv);


    const apiDeviceBtn = document.createElement("button");
    apiDeviceBtn.textContent = "device: " + apiDevice;
    apiDeviceBtn.style.cssText = `
        border: 3px solid black;
        background-color: rgba(255, 255, 255, 0.8);
        margin-right: 3px;
        padding: 2px;
    `;
    apiDeviceBtn.addEventListener("click", () => {
        apiDevice = apiDevice === "mobile" ? "pc" : "mobile";
        apiDeviceBtn.textContent = "device: " + apiDevice;
        storage.set('device', apiDevice);
    });
    buttonDiv.appendChild(apiDeviceBtn);


    const gameReleaseBtn = document.createElement('button');
    gameReleaseBtn.textContent = "API: " + APItarget;
    gameReleaseBtn.style.cssText = `
        border: 3px solid black;
        background-color: rgba(255, 255, 255, 0.8);
        display: flex;
        padding: 2px;
        margin-right: 3px;
    `;
    gameReleaseBtn.addEventListener('click', () => {
        APItarget = APItarget == "beta" ? "main" : "beta";
        gameReleaseBtn.textContent = "API: " + APItarget;
        storage.set('APItarget', APItarget);
    });
    buttonDiv.appendChild(gameReleaseBtn);


    const layoutStyleBtn = document.createElement("button");
    layoutStyleBtn.textContent = "style: " + layoutStyle;
    layoutStyleBtn.style.cssText = `
        border: 3px solid black;
        background-color: rgba(255, 255, 255, 0.8);
        display: flex;
        padding: 2px;
    `;
    layoutStyleBtn.addEventListener("click", () => {
        layoutStyle = layoutStyle == "verbose" ? "concise" : "verbose";
        layoutStyleBtn.textContent = "style: " + layoutStyle;
        storage.set("layoutStyle", layoutStyle);
    })
    buttonDiv.appendChild(layoutStyleBtn)


    // region target team
    const targetTeamDiv = document.createElement('div');
    targetTeamDiv.style.cssText = `
        z-index: 999999;
        background-color: rgba(25, 160, 205, 1);
        border: 1px solid black;
        border-radius: 5px;
        display: flex;
        height: 20px;
    `;
    lobby_container.appendChild(targetTeamDiv);

    const teamTitle = document.createElement('p');
    teamTitle.textContent = "target team: ";
    teamTitle.classList.add("button-label")
    targetTeamDiv.appendChild(teamTitle);

    const teamButton = document.createElement('button');
    teamButton.textContent = intToTeam[buttonIndex];
    teamButton.style.cssText = `
        margin-left: 5px;
        background-color: ${intToColour[buttonIndex]};
    `;
    teamButton.addEventListener('click', () => {
        buttonIndex = (buttonIndex + 1) % intToTeam.length;
        teamButton.textContent = intToTeam[buttonIndex];
        teamButton.style.backgroundColor = intToColour[buttonIndex];
        storage.set('teamIndex', buttonIndex);
    });
    targetTeamDiv.appendChild(teamButton);





    // region net_predict
    const divPredictMovement = document.createElement("div");
    divPredictMovement.classList.add("button-row-div")
    divPredictMovement.style.cssText = "background-color: rgba(222, 52, 248, 0.55);";
    lobby_container.appendChild(divPredictMovement)

    
    const netPredictLabel = document.createElement("p")
    netPredictLabel.textContent = "net_predict_movement: "
    netPredictLabel.classList.add("button-label")
    divPredictMovement.appendChild(netPredictLabel)

    const netPredictBtn = document.createElement("button");
    netPredictBtn.textContent = predictMovement
    netPredictBtn.style.cssText = `
        background-color: ${predictMovement ? GREEN : RED};
        margin-left: 5px;
    `;
    netPredictBtn.addEventListener('click', () => {
        predictMovement = !predictMovement;
        netPredictBtn.textContent = predictMovement;
        netPredictBtn.style.backgroundColor = predictMovement ? GREEN : RED;
        input.set_convar(PREDICT_MOVEMENT, predictMovement);
        storage.set("predictMovement", predictMovement);
    });
    divPredictMovement.appendChild(netPredictBtn);




    // region render latency
    const divRenderLatency = document.createElement("div");
    divRenderLatency.classList.add("button-row-div");
    divRenderLatency.style.cssText = "background-color: rgba(200, 200, 40, 0.55);";
    lobby_container.appendChild(divRenderLatency);

    const renLatencyLabel = document.createElement("p")
    renLatencyLabel.textContent = "render_latency: "
    renLatencyLabel.classList.add("button-label")
    divRenderLatency.appendChild(renLatencyLabel)

    const renderLatencyBtn = document.createElement("button");
    renderLatencyBtn.textContent = renderLatency
    renderLatencyBtn.style.cssText = `
        background-color: ${renderLatency ? GREEN : RED};
        margin-left: 5px;
    `;
    renderLatencyBtn.addEventListener('click', () => {
        renderLatency = !renderLatency;
        renderLatencyBtn.textContent = renderLatency;
        renderLatencyBtn.style.backgroundColor = renderLatency ? GREEN : RED;
        input.set_convar(REN_LATENCY, renderLatency);
        storage.set("renLatency", renderLatency);
    });
    divRenderLatency.appendChild(renderLatencyBtn);




    // region render FPS
    const divRenderFPS = document.createElement("div");
    divRenderFPS.classList.add("button-row-div");
    divRenderFPS.style.cssText = "background-color: rgba(201, 28, 224, 0.55);";
    lobby_container.appendChild(divRenderFPS);

    const renFPSLabel = document.createElement("p")
    renFPSLabel.textContent = "render_FPS: "
    renFPSLabel.classList.add("button-label")
    divRenderFPS.appendChild(renFPSLabel)

    const renFPSBtn = document.createElement("button");
    renFPSBtn.textContent = renderFPS
    renFPSBtn.style.cssText = `
        background-color: ${renderFPS ? GREEN : RED};
        margin-left: 5px;
    `;
    renFPSBtn.addEventListener('click', () => {
        renderFPS = !renderFPS;
        renFPSBtn.textContent = renderFPS;
        renFPSBtn.style.backgroundColor = renderFPS ? GREEN : RED;
        input.set_convar(REN_FPS, renderFPS);
        storage.set("renFPS", renderFPS);
    });
    divRenderFPS.appendChild(renFPSBtn);




    // region selector div


    const lobbySelectorDiv = document.createElement('div');
    lobbySelectorDiv.style.cssText = `
        right: 0%;
        top: 0%;
        z-index: 999999;
        background-color: rgba(141, 24, 187, 1);
        padding: 5px;
        border: 1px solid black;
        border-radius: 5px;
        display: block;
    `;
    lobby_container.appendChild(lobbySelectorDiv);





    // region saves upgrades

    const presetUpgrades_container = document.createElement("div");
    presetUpgrades_container.style.cssText = `
        position: absolute;
        right: 0%;
        top: 0%;
        z-index: 999999;
        background-color: rgba(255, 255, 255, 0.5);
        padding: 5px;
        border: 1px solid black;
        border-radius: 5px;
        display: none;
    `;
    document.body.append(presetUpgrades_container);

    const resetBtn = document.createElement("button");
    resetBtn.textContent = "reset"
    resetBtn.classList.add("presets-header-button");
    resetBtn.addEventListener("click", () => setBuild(0))
    presetUpgrades_container.appendChild(resetBtn);

    const newPresetBtn = document.createElement("button");
    newPresetBtn.textContent = "add preset"
    newPresetBtn.classList.add("presets-header-button");

    function addPresetBuild(tankname, build) {
        const upgradeDiv = document.createElement("div");
        upgradeDiv.style.cssText = `
            border: 3px solid black;
            display: flex;
            margin-top: 10px;
            margin-bottom: 10px;
        ;`
        presetUpgrades_container.appendChild(upgradeDiv);

        const tankIconImage = document.createElement("img");
        tankIconImage.style.cssText = `
            width: 50px;
            height: 50px;
        `;
        tankIconImage.src = getTankIconLinkFor(tankname);
        upgradeDiv.appendChild(tankIconImage);

        const tankNameLabel = document.createElement("p");
        tankNameLabel.style.cssText = `
            margin-bottom: 15px;
            margin-top: 15px;
            margin-right: 10px;
            margin-left: 10px;
            color: black;
        `;
        tankNameLabel.textContent = tankname;
        upgradeDiv.appendChild(tankNameLabel);

        const slashBuildArr = [];
        for (let i = 1; i < 9; i++) {
            slashBuildArr[i-1] = count(build, i);
        }

        const buildDisplay = document.createElement("input");
        buildDisplay.value = slashBuildArr.join("/")
        buildDisplay.style.cssText = `
            margin-bottom: 15px;
            margin-top: 15px;
            margin-right: 10px;
            margin-left: 10px;
            width: ${buildDisplay.value.length * 7}px;
        `;
        buildDisplay.disabled = true;
        upgradeDiv.appendChild(buildDisplay);

        const setButton = document.createElement("button");
        setButton.style.cssText = `
            background-color: rgba(255, 255, 255, 1);
            left: 0%;
            cursor: pointer;
        `;
        setButton.textContent = "set";
        setButton.addEventListener("click", () => {
            setBuild(build)
            togglePresets();
        });
        upgradeDiv.appendChild(setButton)
    }

    for (const [tankname, build] of Object.entries(presetTankBuilds)) {
        addPresetBuild(tankname, build);
    }
    
    function togglePresets() {
        presetUpgrades_container.style.display =  
        presetUpgrades_container.style.display == "block" ? "none" : "block";
    }

    document.addEventListener('keydown', (ev) => {
        if (ev.key === TOGGLE_LOBBY_KEYBIND) {
            toggleLobby();
        } else if (ev.key == TOGGLE_PRESETS_KEYBIND) {
            togglePresets();
        }
    })

    // kill yourself
    window.console.log = consoleLog
})();