cursors.io 300 bots hack

try to take over the world!

// ==UserScript==
// @name         cursors.io 300 bots hack
// @namespace    http://tampermonkey.net/
// @version      0.7
// @description  try to take over the world!
// @author       You
// @match        http://cursors.io/
// @match        http://kursors.io/
// @match        http://cursors.io/?editor
// @grant        none
// ==/UserScript==

(function() {
    'use strict';

"use strict"

// Beta client for the Zursor Master project
// Please do not paste this script itself into tampermonkey or cursors.io, instead please use the tampermonkey extension that I have provided here: https://pastebin.com/krbGjJ9v
// Especially since this script is still in development, the tampermonkey extension provided is automatically updated
// update as of oct. 13, 2020: Script has been updated to avoid any CORB/CORS errors. Change the __ip variable to the new server ip when it updates. Get the server ip by going to the 'Network' tab in inspect element, go to the 'WS' tab, and refresh the page. DM me on discord at 8y8x#5342 if the script should be updated
window.zM = {};
window.zMDK = {};
async function wait(ms = 1000) {
    setTimeout(()=>{return new Promise(async resolve => resolve());},ms);
}
window.D = document;
let __ip = '128.199.12.58';
(async function(styles, init, cfg, dependencies, elements) {
    zM.dependenciesLoaded = 0b000000;
    zM.version = '0.2';

    zM.cfg = cfg.raw;
    zM.createConfig = cfg.createConfig;
    zM.saveConfig = cfg.saveConfig;
    zM.getConfig = cfg.getConfig;

    zM.returnAfterUndeployed = false;
    zM.antiIdle = true;

    // load developer kit for plugins
    // it isn't exactly done yet, I plan to only do it once I'm mostly done the whole script- and it's stable and everything
    /**
     * Tools for modifying the localStorage configuration
     *
     * .generate() - creates a default config, returns the result. Optionally, you can put the object pointer as a parameter.
     * .save() - saves the parameter provided. Returns true/false dependant if errors occur.
     * .get() - returns the saved config. Optionally, you can put the object pointer as a  parameter.
     */
    zMDK.config = {
        generate: cfg.createConfig,
        save: cfg.saveConfig,
        get: cfg.getConfig
    }

    // some useful shortcuts
    window.head = D.head,
    window.body = D.body;

    // disable all elements on page
    var temp = [];
    temp = body.getElementsByTagName('*');
    for (var i = 0; i < temp.length; ++i) temp[i].style.display = 'none';

    // delete all header elements
    head.innerHTML = '<title>Loading...</title>';

    // add a container
    temp = D.createElement('div');
    temp.style.cssText = 'background:linear-gradient(45deg, rgba(33,33,33,1) 0%, rgba(66,66,66,1) 100%);position:fixed;top:0px;left:0px;width:100vw;height:100vh;';
    temp.id = 'zM-container'
    body.appendChild(temp);

    zM.el = zMDK.elements = {};
    zM.el.container = temp;

    zM.fleet = [];
    zM.fSel = NaN; // NaN for news

    temp = styles();

    zM.el.style = {
        fleetTopbarContainer: temp[0]
    }

    zM.pos = {
        mouseX: 400,
        mouseY: 300,
        lastMouseX: 400,
        lastMouseY: 300
    }

    zM.useWallhack = true;

    zM.ease = function(n, o, upd) {
        var dist = n - o,
            prog = (Date.now() - upd) / 100,
            maxMin = Math.min(Math.max(0, prog), 1);

        return o + (maxMin ** 2 * (3 - 2 * maxMin)) * dist;
    }

    zM.isLoading = false;

    zM.game = {
        m28nCursorsIP: `${__ip}:2828`
    }

    zM.asyncWaitUntil = async function(toEval) {
        return new Promise(resolve => {
            var temp = setInterval(()=>{if (eval(toEval)) resolve(), clearInterval(temp);},1);
        });
    }

    zM.origin = (location.hostname === 'cursors.io' || location.href.startsWith('file:///')) ? 'cursors' : location.hostname === 'kursors.io' ? 'kursors' : 'localhost';

    zM.movement = 1;

    // I decided to load the loading bars before the loading API therefore it works for when the script is starting first
    var temp = [D.createElement('div'), D.createElement('div'), D.createElement('div'), D.createElement('div')];

    temp[0].className = 'zM-loading-outer';
    zM.el.container.appendChild(temp[0]);

    temp[1].className = 'zM-loading-title';
    temp[0].appendChild(temp[1]);
    temp[1].textContent = 'Loading...'

    temp[2].className = 'zM-loading-bar-outer';
    temp[0].appendChild(temp[2]);

    temp[3].className = 'zM-loading-bar-fill';
    temp[2].appendChild(temp[3]);

    zM.el.loadingOuter = temp.shift();
    zM.el.loadingTitle = temp.shift();
    zM.el.loadingBarFill = temp.pop();

    // loading bar API
    zM.setLoadingWidth = function(width = 50) {
        zM.el.loadingBarFill.style.width = 'calc(' + width + '%' + ' - 10px)';
    }

    zM.showLoading = function() {
        zM.el.loadingOuter.style.display = 'block';
        D.getElementsByTagName('title')[0].textContent = 'Loading...';
    }

    zM.hideLoading = function() {
        zM.el.loadingOuter.style.display = 'none';
        D.getElementsByTagName('title')[0].textContent = 'zM / No fleets yet';
    }

    zM.setLoadingTitle = function(title) {
        zM.el.loadingTitle.textContent = title;
    }

    zM.clickQueue = 0;

    zM.updateTabs = function() { // updates fleet tabs
        zM.el.fleetTopbarOuter.innerHTML = '';
        zM.el.fleetTopbarContainer.innerHTML = '';
        zM.el.fleetTopbarOuter.appendChild(zM.el.fleetTopbarContainer);
        zM.el.fleetTopbarOuter.appendChild(zM.el.fleetTopbarAdd);
        zM.el.fleetTopbarAdd.appendChild(zM.el.fleetTopbarAddImg);
        zM.el.fleetTopbarOuter.appendChild(zM.el.fleetTopbarHint);

        for (var i = 0; i < zM.fleet.length; ++i) {
            var f = zM.fleet[i];
            zM.fleet[i].el.container.style.display = 'none';
            zM.el.fleetTopbarContainer.innerHTML += `<div style="position: absolute;    top: 0px;    left: ${100/zM.fleet.length*i}%;    width: ${100/zM.fleet.length}%;    height: 35px;    border-right: 1px solid #fffc;    background-color: ${i === zM.fSel ? '#fff3' : '#fff0'};transition: 0.3s;"><div style="position: absolute;    top: 0px;    left: 0px;    width: 100%;    height: 35px;    background-color: #0000;    transition: 0.3s;" onmouseenter="this.style.backgroundColor = '#0003';" onmouseout="this.style.backgroundColor = '#0000'" onclick="zM.fSel = ${i}; zM.updateTabs()"><div style="color:#fffc;font-family:\'Nova Flat\',Montserrat,sans-serif;font-weight:300;padding:8px;pointer-events:none;white-space:nowrap;overflow-x:hidden;user-select:none;">${zM.fSel === i ? '(Selected)' : ''} #${i} / (${f.botsOpened}/${f.maxBots}) / ${f.name}</div></div></div>`;
        }

        zM.fleet[zM.fSel].updateTabs();
    }

    zM.createFleet = function(ip, port, ipv6, startbots, maxbots, name) {
        var nF;
        var temp2 = function() {
            clearInterval(temp[2]);
            clearInterval(temp[3]);
            clearInterval(temp[4]);
            zM.setLoadingWidth(0);
            zM.setLoadingTitle('Server isn\'t a valid cursors.io server.');
            nF.bots.forEach(X => X.socket.close());
            setTimeout(zM.hideLoading, 2000);
        };
        var temp = [
            setInterval(() => {
                if (!zM.isLoading) clearInterval(temp[0]), zM.isLoading = true, temp[1]();
            }, 50),

            function() {
                zM.el.fleetTopbarHint.style.display = 'none';
                zM.el.fleetPromptOuter.style.display = 'none';
                zM.setLoadingWidth(0);
                zM.setLoadingTitle('Creating fleet object...');
                zM.showLoading();



                nF = new zM.Fleet(ip, port, ipv6, startbots, maxbots, name);

                zM.fSel = zM.fleet.push(nF) - 1;

                nF.createGUI(zM.el.container)
                temp[2] = setInterval(() => {
                    zM.setLoadingTitle('Connecting bots ('+nF.botsOpened+'/'+startbots+')');
                    zM.setLoadingWidth(100/startbots*nF.botsPinged);
                    if (nF.verified === false) temp2();
                    else if (startbots <= nF.botsPinged) {
                        clearInterval(temp[2]);
                        setTimeout(async ()=>{
                            zM.isLoading = false;
                            zM.hideLoading(),
                            zM.updateTabs(),
                            zM.el.game.style.display = 'block';
                            await zM.asyncWaitUntil('!(Date.now() % 50)');
                            nF.startTickLoop();
                        }, 800);
                    }
                }, 50);
            }
        ]
    }

    // get IP
    if (window.m28n) {
        m28n.findServerPreference('cursors', (err, end  ) => {
            if (err) {
                console.error(err);
            } else {
                zM.game.m28nCursorsIP = (end.ipv4 || ('[' + end.ipv6 + ']')) + ':2828';
            }
        });
    }







    zM.clog = [

        //
        //
        // Beta v0.2
        //
        // VVVVVVVVV

        {
            type: 'title',
            msg: 'Beta v0.2'
        },
        {
            type: 'desc',
            msg: 'I added several new improvements, mostly to the GUI, a full list is below:<br><br>'
        },
        {
            type: 'desc',
            msg: '• Added a changelog and a little information thingy at the bottom right. I plan to make it contain more information soon'
        },
        {
            type: 'desc',
            msg: '• Cursor movements are now synchronized, they should no longer split up'
        },
        {
            type: 'desc',
            msg: '• Script now loads the Nova Flat font used'
        },
    ];






    //dependencies.forEach(i => zM.el.container.appendChild(temp = D.createElement('script'), temp.src = 'http://pastebin.com/raw/'+i, temp));
    dependencies.forEach(eval);

    zM.setLoadingWidth(0);
    zM.showLoading();
    zM.setLoadingTitle('Loading dependencies...');
    temp = setInterval(() => {
        if (zM.dependenciesLoaded == 0b111111) {
            zM.setLoadingWidth(100);
            clearInterval(temp);
            elements.forEach(i => i(zM.el.container));
            head.getElementsByTagName('title')[0].textContent = 'zM / No fleets yet' // run all element functions
            setTimeout(zM.hideLoading,800);
        } else {
            var done = 0;
            for (var i = 0; i < temp.length; ++i) {
                done += (zM.dependenciesLoaded & 2**i) ? 1 : 0;
            }
            zM.setLoadingWidth((100 / 16 * done) | 0);
        }
    }, 50);

    init();
})(
    function() { // add <style>
        var out = [];
        var temp = [ // fleet topbar
            '.zM-' + 'fleet-topbar-outer {',
            '   position: fixed;    top: 0px;    left: 0px;',
            '   width: 100vw;    height: 35px;',
            '   background-color: rgba(0,0,0,.2);',
            '}',
            '',
            '.zM-' + 'fleet-topbar-container {',
            '   position: fixed;    top: 0px;    left: 0px;',
            '   width: calc(100vw - 45px);    height: 35px;',
            '}',
            '',
            '.zM-' + 'fleet-topbar-tab {',
            '   display: inline-block;',
            '}',
            '',
            '.zM-' + 'fleet-topbar-add {',
            '   position: fixed;    top: 0px;    right: 0px;',
            '   width: 45px;    height: 35px;',
            '   background-color: rgba(0,0,0,.0);',
            '   transition: 0.3s;',
            '   cursor: pointer;    user-select: none;',
            '}',
            '',
            '.zM-' + 'fleet-topbar-add:hover {',
            '   background-color: rgba(0,0,0,.2)',
            '}',
            '',
            '.zM-' + 'fleet-topbar-hint {',
            '   color: rgba(255, 255, 255, .3);',
            '   text-align: right;    font-family: Montserrat, sans-serif;',
            '   font-weight: 200;    font-size: 14px;',
            '   padding-right: 55px;',
            '   padding-top: 8px;    user-select: none;',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        // --------------------------------------------------------

        var temp = [ // fleet prompt
            '.zM-' + 'fleet-prompt-outer {',
            '   position: fixed;    top: 85px;    left: 50px;',
            '   width: calc( 100vw - 100px );    height: calc( 100vh - 135px );',
            '   background-color: #0003;   display: none;    border-radius: 20px;    z-index: 5000;',
            '}',
            '',
            '.zM-' + 'fleet-prompt-container {',
            '   position: fixed;    top: 85px;    left: calc( 50vw - 300px );',
            '   width: 600px; height: calc( 100vh - 135px );',
            '}',
            '',
            '.zM-' + 'fleet-prompt-title {',
            '   text-align: center;    font-family: \'Nova Flat\', Montserrat, sans-serif;    font-weight: 300;',
            '   color: rgba(255, 255, 255, .8);    font-size: 24px;    padding-top: 10px;    user-select: none;',
            '}',
            '',
            '.zM-' + 'fleet-prompt-entry {',
            '   width: 580px;    height: 50px;    ',
            '   background-color: rgba(0, 0, 0, .1);',
            '   border-radius: 10px;',
            '   margin-left: 10px;    margin-top: 10px;',
            '}',
            '',
            '.zM-' + 'fleet-prompt-opt {',
            '   text-align: right;    color: rgba(255, 255, 255, .5);    font-family: \'Nova Flat\', Montserrat, sans-serif;',
            '   font-weight: 300;    font-size: 14px;',
            '   padding-right: 20px;    padding-top: 16px;    width: 280px;    display: inline-block;    user-select: none;',
            '}',
            '',
            '.zM-' + 'fleet-prompt-create {',
                'position: absolute;    bottom: 10px;    left: 95px;',
                'width: 200px;    height: 20px;',
                'background-color: #0c6;    font-family: \'Nova Flat\', Montserrat, sans-serif;',
                'font-size: 16px;    font-weight: 300;    color: #0008;    text-align: center;',
                'border-top-left-radius: 10px;    border-bottom-left-radius: 10px;    user-select: none;    cursor: pointer;',
            '}',
            '',
            '.zM-' + 'fleet-prompt-cancel {',
                'position: absolute;    bottom: 10px;    left: 305px;',
                'width: 200px;    height: 20px;',
                'background-color: #666;    font-family: \'Nova Flat\', Montserrat, sans-serif;',
                'font-size: 16px;    font-weight: 300;    color: #0008;    text-align: center;',
                'border-top-right-radius: 10px;    border-bottom-right-radius: 10px;    user-select: none;    cursor: pointer;',
            '}',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        // --------------------------------------------------------

        var temp = [ // loading bar stuff
            '.zM-' + 'loading-outer {',
                'position: fixed;    top: 0px;    left: 0px;',
                'width: 100vw;    height: 100vh;    background: linear-gradient(45deg, rgba(33,33,33,1) 0%, rgba(66,66,66,1) 100%);',
                'z-index: 50000;    display: none;',
            '}',
            '',
            '.zM-' + 'loading-title {',
                'position: fixed;    top: calc(50vh - 32px);    left: 50vw;',
                'color: #fffb;    font-family: Montserrat, sans-serif;    font-weight: 300;    font-size: 24px;',
                'transform: translateX(-50%);    z-index: 50001;    user-select: none;',
            '}',
            '',
            '.zM-' + 'loading-bar-outer {',
                'position: fixed;    top: 50vh;    left: calc(50vw - 200px);',
                'width: 400px;    height: 30px;    background-color: #0003;    border-radius: 15px;',
            '}',
            '',
            '.zM-' + 'loading-bar-fill {',
                'width: 0%;    height: 20px;    background-color: #3c5;    border-radius: 10px;    margin: 5px;    transition: 0.5s;',
            '}',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        // --------------------------------------------------------

        var temp = [ // fleet page
            '.zM-' + 'fleet-gui-container {',
                'position: fixed;    top: 35px;    left: 0px;    width: 100vw;    height: calc(100vh - 35px);    z-index:1000;',
            '}',
            '',
            '.zM-' + 'fleet-gui-topbarOuter {',
                'position: fixed;    top: 35px;    left: 0px;    width: 100vw;    height: 35px;    background-color: #0002;    z-index:1001;',
            '}',
            '',
            '.zM-' + 'fleet-gui-topbar {',
                'position: fixed;    top: 35px;    left: 0px;    width: calc(100vw - 45px);    height: 35px;    z-index:1002;',
            '}',
            '',
            '.zM-' + 'fleet-gui-tabNew {',
                'position: fixed;    top: 35px;    right: 0px;    width: 45px;    height: 35px;    background-color: #0000;    transition: 0.3s;    z-index:1003;    cursor: pointer;',
            '}',
            '',
            '.zM-' + 'fleet-gui-tabNew:hover {',
            '   background-color: rgba(0,0,0,.2)',
            '}',
            '',
            '.zM-' + 'fleet-gui-tabNewImg {',
                'position: fixed;    top: 35px;    right: 0px;    width: 45px;    height: 35px;    pointer-events: none;    z-index:1004;    user-select:none;',
            '}',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        // --------------------------------------------------------

        var temp = [ // news page
            '.zM-' + 'news-outer {',
                'position: fixed;    top: 35px;    left: 0px;    width: 100vw;    height: calc(100vh - 35px);    z-index: 9000;',
            '}',
            '',
            '.zM-' + 'news-container {',
                'position: absolute;    top: calc(50vh - 190px);    left: calc(50vw - 500px);    width: 1000px;    height: 380px;    z-index 9001;',
            '}',
            '',
            '.zM-' + 'news-page-top {',
                'z-index: 9002;',
                'width: 1000px; margin-left: 0px; margin-bottom: 10px;    height: 120px;    background-color: #0003;    border-top-left-radius: 75px;    border-top-right-radius: 75px;',
            '}',
            '',
            '.zM-' + 'news-page-mid {',
                'z-index: 9002;',
                'width: 1000px; margin-left: 0px; margin-bottom: 10px;    height: 120px;    background-color: #0003',
            '}',
            '',
            '.zM-' + 'news-page-bottom {',
                'z-index: 9002;',
                'width: 1000px; margin-left: 0px; margin-bottom: 10px;    height: 120px;    background-color: #0003;    border-bottom-left-radius: 75px;    border-bottom-right-radius: 75px;',
            '}',
            '',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        // --------------------------------------------------------

        var temp = [ // game
            '.zM-' + 'game {',
                'position: fixed;    top: 80px;    left: 10px;    width: 800px;    height: 600px;    background-color: #fff;    z-index: 9000;    display: none;    cursor: none;',
            '}',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        // --------------------------------------------------------

        var temp = [ // bottom-right info area
            '.zM-' + 'info-outer {',
                'position: fixed; bottom: 10px; right: 10px; width: 200px; height: 40px; background-color: #0003; border-radius: 20px; z-index: 20000;',
            '}',
            '',
            '.zM-' + 'info-ver {',
                'position: absolute; top: 10px; left: 10px; font-family: Montserrat, sans-serif; color: #fff8; user-select: none; z-index: 20001;',
            '}',
            '',
            '.zM-' + 'info-clog-button-outer {',
                'position: absolute; top: 5px; right: 5px; width: 120px; height: 30px; background-color: #0003; user-select: none; z-index: 20002; border-radius: 15px; cursor: pointer;',
            '}',
            '',
            '.zM-' + 'info-clog-button {',
                'position: absolute; top: 8px; left: 12px; font-size: 12px; color: #fff8; user-select: none; z-index: 20003; pointer-events: none; font-family: Montserrat;',
            '}',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        // --------------------------------------------------------

        var temp = [ // changelog area
            '.zM-' + 'clog-outer {',
                'position: fixed; top: 0px; left: 0px; width: 100vw; height: 100vh; background:linear-gradient(45deg, rgba(33,33,33,1) 0%, rgba(66,66,66,1) 100%); z-index: 15000; display: none;',
            '}',
            '',
            '.zM-' + 'clog-container {',
                'position: fixed; top: 50px; left: calc(50vw - 400px); width: 800px; height: calc(100vh - 100px); background-color: #0003; border-radius: 50px; z-index: 15001;',
            '}',
            '',
            '.zM-' + 'clog-title {',
                'color: #fffc; font-family: Montserrat; font-weight: 300; font-size: 36px; z-index: 15002; margin-bottom: 20px; margin-left: 30px; margin-top: 30px; user-select: none;',
            '}',
            '',
            '.zM-' + 'clog-desc {',
                'color: #fff8; font-family: Montserrat; font-weight: 400; font-size: 14px; z-index: 15002; margin-bottom: 10px; margin-left: 30px; user-select: none;',
            '}',
            '',
            '.zM-' + 'clog-close {',
                'position: fixed; top: 0px; right: 0px; width: 45px; height: 35px; cursor: pointer; user-select: none;',
            '}',
            '',
            D.createElement('style')
        ];
        // adds style element to "out"
        out[out.push(temp.pop()) - 1].innerHTML = temp.join('\n');

        for (var i = 0; i < out.length; ++i) head.appendChild(out[i]);
        return out;
    },
    function() { // initialize, start different tasks
        zM.promptFleet = function() {
            if (isNaN(zM.fSel)) {
                zM.el.newsOuter.style.display = 'none';
            } else {
                var temp = parseInt(zM.fSel);
                var f = zM.fleet[temp];
                if (f) {
                    f.el.container.style.display = 'none';
                }
            }
            zM.el.game.style.display = 'none';
            zM.el.fleetPromptOuter.style.display = 'block';
        }

        zM.zWs = new WebSocket('ws://localhost:2882');

    },
    (function(cfg) { // loads configuration
        function createConfig(x) { // "x" is a pointer to the variable to change

            if (!x.keybinds) {
                x.keybinds = []
            }

            return x;
        }
        function saveConfig(x) {
            try {
                var out = JSON.stringify(x);
                localStorage.setItem('zM', btoa(out));

                return true;
            } catch(err) {
                switch (true) {
                    case err instanceof TypeError: {
                        console.error('[CFG Loader] Error with saving the configuration');
                    }


                    default: { // other unhandled errors
                        console.error('[CFG Loader] Error with loading the configuration: Unhandled error occured');
                    }
                }

                console.debug(err);

                return false;
            }
        }
        function getConfig(x) {
            try {
                var out = localStorage.getItem('zM'); // null
                if (!out) throw 'Undefined storage';
                out = atob(out); // DOMException errors
                out = JSON.parse(out); // SyntaxErrors
                if (x) x = out;
                return out;
            } catch(err) {
                switch (true) {
                    case err instanceof DOMException: { // atob error, invalid configuration
                        console.error('[CFG Loader] Error with loading the configuration: Configuration was incorrectly saved');
                    } break;


                    case err instanceof SyntaxError: { // JSON errors return syntax errors for some reason
                        console.error('[CFG Loader] Error with loading the configuration: Configuration is not an object');
                    } break;
                    default: { // other unhandled errors
                        console.error('[CFG Loader] Error with loading the configuration: Unhandled error occured');
                    }
                }

                console.debug(err);

                if (x) x = {};
                return null;
            }
        }
        if (cfg === null) { // if the localstorage has been erased or script has never been used before
            cfg = {};
            createConfig(cfg);
        } else {
            try {
                cfg = getConfig(cfg);
                if (!cfg) saveConfig((createConfig(cfg), cfg));
            } catch(err) {
                switch (true) {
                    case err instanceof DOMException: { // atob error, invalid configuration
                        console.error('[CFG Loader] Error with loading the configuration: Configuration was incorrectly saved');
                    } break;


                    case err instanceof SyntaxError: { // JSON errors return syntax errors for some reason
                        console.error('[CFG Loader] Error with loading the configuration: Configuration is not an object');
                    } break;
                    default: { // other unhandled errors
                        console.error('[CFG Loader] Error with loading the configuration: Unhandled error occured');
                    }
                }

                console.debug(err);

                // technically these functions don't need to be placed in each other, they use pointers anyway
                cfg = {};
                saveConfig(createConfig(cfg));
            }
        }

        return {
            raw: cfg,
            getConfig: getConfig,
            saveConfig: saveConfig,
            createConfig: createConfig
        }
    })(localStorage.getItem('zM')),

    // ---
    // list of dependencies (pastebin) below
    // edit as of oct. 13, 2020: pastebin now uses CORS and doesn't allow requests, so the scripts are now embedded here instead
    // ---
    [
        /* zM external images */
        `zM.dependenciesLoaded |= 0b000001;

        window.zMIMG = {
            add: {img:null,uri:""},
            close: {img:null,uri:""},
            cursor: {img:null,uri:""},
            public: {img:null,uri:''},
            movement: {img:null,uri:''},
            local: {img:null,uri:''},
            follow: {img:null,uri:''},
            admin: {img:null,uri:''}
        }

        for (var i in zMIMG) {
            if (zMIMG.hasOwnProperty(i)) {
                zMIMG[i].img = new Image();
                zMIMG[i].img.src = zMIMG[i].uri;
            }
        }`,
        /* zM fleet class */
        `zM.dependenciesLoaded |= 0b000010;

        zM.Fleet = class {
            constructor (ip, port, ipv6, maxbots, startbots, name) {
                console.log(startbots);
                this.ip = ip;
                this.port = port;
                this.maxBots = maxbots;
                this.startBots = startbots;
                this.name = name;
                this.ipv6 = ipv6;
                this.cSel = 0;

                this.wsIp = 'ws://' + (this.ipv6?'['+this.ip+']':this.ip) + ':' + this.port;

                this.bots = [];

                this.ids = [];

                this.botsPinged = 0;
                this.botsOpened = 0;
                this.botsIds = 0;
                this.botsLevels = 0;
                this.botsBusy = 0;
                this.botsDeployed = 0;
                this.botsUpdated = 0;
                this.verified = true;

                this.isHelping = false;

                this.el = {};

                this.helpers = [];

                var extF = this;

                this.helpIndex = 0;

                this.helpLoop = setInterval(()=>{
                    ++this.helpIndex;
                    if (this.isHelping && this.helpers.length > 0) {
                        for (var i = 0; i < this.helpers.length; ++i) {
                            this.helpers[i].toHelp = [];
                            this.helpers[i].helping = true;
                            if (this.helpers[i].socket.readyState != 1) this.helpers.splice(i--, 1);
                        }
                        var todo = [];
                        var obj = this.bots[this.cSel].obj;
                        for (var i = 0; i < obj.length; ++i) {
                            if (obj[i].type === 4) todo.push(obj[i]);
                        }
                        var amount = todo.length / this.helpers.length;
                        var j = 0;
                        var count = amount;
                        for (var i = 0; i < this.helpers.length; ++i) {
                            for (; j < count | 0; ++j) {
                                var obj = todo[j];
                                var moves = zM.path(this.helpers[i].realX, this.helpers[i].realY, obj.x + (obj.w>>1), obj.y + (obj.h>>1), this.helpers[i].obj, this.helpers[i].grid);
                                if ((this.helpers[i].realX === obj.x + (obj.w>>1) && this.helpers[i].realY === obj.y + (obj.h>>1)) ? true : moves.length > 1) this.helpers[i].toHelp.push(todo[j]);
                            }
                            count += amount;
                        }
                        var f = this;
                        if (this.isHelping) for (var j = 0; j < this.helpers.length; ++j) {
                            var index = this.helpIndex % this.helpers[j].toHelp.length;
                            if (isNaN(index)) index = this.helpIndex % this.helpers[0].toHelp.length; // has nothing to do, copy off other bots
                            if (isNaN(index)) index = 0; // still undefined, no buttons at all
                            console.log(j, this.helpers[j]);
                            console.log(index, this.helpers[j].toHelp[index]);
                            if (this.helpers[j].toHelp.length <= index) continue;
                            var moves = zM.path(this.helpers[j].realX, this.helpers[j].realY, this.helpers[j].toHelp[index].x + (this.helpers[j].toHelp[index].w>>1), this.helpers[j].toHelp[index].y + (this.helpers[j].toHelp[index].h>>1), this.helpers[j].obj, this.helpers[j].grid);
                            moves.forEach(Y => {
                                zM.packet.moveSocket([this.helpers[j].socket], Y[0], Y[1], [this.helpers[j]]);
                            });
                            for (var k = 0; k < 10; ++k) zM.packet.clickSocket([this.helpers[j].socket], this.helpers[j].toHelp[index].x + (this.helpers[j].toHelp[index].w>>1), this.helpers[j].toHelp[index].y + (this.helpers[j].toHelp[index].h>>1), [this.helpers[j]])
                        }
                    }
                },300);

                this.elTemplate = {
                    tab: '<div style="width: '
                }

                setTimeout(()=>{for (var i = 0; i < startbots; ++i) this.bots.push(new zM.Bot(this));},300);
            }

            createGUI(c) {
                var temp = this.el = {
                    container: D.createElement('div'),
                    topbarOuter: D.createElement('div'),
                    topbar: D.createElement('div'),
                    tabNew: D.createElement('div'),
                    tabNewImg: D.createElement('img')
                }

                temp.container.className = 'zM-fleet-gui-container';
                c.appendChild(temp.container);

                temp.topbarOuter.className = 'zM-fleet-gui-topbarOuter';
                temp.container.appendChild(temp.topbarOuter);

                temp.topbar.className = 'zM-fleet-gui-topbar';
                temp.topbarOuter.appendChild(temp.topbar);

                temp.tabNew.className = 'zM-fleet-gui-tabNew';
                temp.topbarOuter.appendChild(temp.tabNew);

                temp.tabNewImg.className = 'zM-fleet-gui-tabNewImg';
                temp.tabNewImg.src = zMIMG.add.uri;
                temp.tabNew.appendChild(temp.tabNewImg);

                temp.tabNew.addEventListener('click', ev => {
                    this.bots.push(new zM.Bot(this));
                });

                this.updateTabs();
            }

            updateTabs() {
                this.el.container.style.display = 'block';
                this.el.topbar.innerHTML = '';
                this.el.topbarOuter.appendChild(this.el.tabNew);
                this.el.tabNew.appendChild(this.el.tabNewImg);
                var nb = [];
                for (var i = 0; i < this.bots.length; ++i) {
                    if (this.bots[i].socket.readyState == 1) nb.push(this.bots[i]);
                }
                for (var i = 0; i < nb.length; ++i) {
                    var b = nb[i];
                    this.el.topbar.innerHTML += \`<div style="position: absolute;    top: 0px;    left: \${100/nb.length*i}%;    width: \${100/nb.length}%;    height: 35px;    border-right: 1px solid #fffc;    background-color: \${i === this.cSel ? '#fff3' : '#fff0'};transition: 0.3s;"><div style="position: absolute;    top: 0px;    left: 0px;    width: 100%;    height: 35px;    background-color: #0000;    transition: 0.3s;" onmouseenter="this.style.backgroundColor = '#0003';" onmouseout="this.style.backgroundColor = '#0000'" onclick="zM.fleet[zM.fSel].cSel = \${i}; zM.fleet[zM.fSel].updateTabs()"><div style="color:#fffc;font-family:\'Nova Flat\',Montserrat,sans-serif;font-weight:300;padding:8px;pointer-events:none;white-space:nowrap;overflow-x:hidden;user-select:none;">\${eval(\`zM.fleet[zM.fSel].cSel === \${i} ? '(Selected)' : ''\`)} #\${i} / ID \${b.id} / LVL \${b.level}</div></div></div>\`;
                }
            }

            /* depreciated, only solves for selected bot
            startTickLoop() {
                this.tickLoop = setInterval(()=>{
                    var b = this.bots[this.cSel];

                    var toMove = [];
                    for (var i = 0; i < this.bots.length; ++i) {
                        if (this.bots[i].level === b.level) if (zM.useWallhack) toMove.push(this.bots[i].socket);
                    }

                    if (zM.useWallhack) {
                        var moves = zM.path(b.realX, b.realY, zM.pos.mouseX>>1, zM.pos.mouseY>>1, b.obj);
                        moves.forEach(Y => {
                            zM.packet.moveSocket(toMove, Y[0], Y[1]);
                        });
                    } else {
                        zM.packet.moveSocket(toMove, zM.pos.mouseX>>1, zM.pos.mouseY>>1);
                    }

                    for (;-1<zM.clickQueue;--zM.clickQueue) zM.packet.clickSocket(toMove, zM.pos.mouseX>>1, zM.pos.mouseY>>1);

                    if (zM.drawing) zM.packet.drawSocket(toMove, zM.pos.lastMouseX>>1, zM.pos.lastMouseY>>1, zM.pos.mouseX>>1, zM.pos.mouseY>>1);

                    zM.pos.lastMouseX = zM.pos.mouseX;
                    zM.pos.lastMouseY = zM.pos.mouseY;
                }, 50);
            }*/

            // alternative, solves for each bot- causes lots of lag without grid
            startTickLoop() {
                this.tickLoop = setInterval(()=>{
                    for (var i = 0; i < zM.fleet.length; ++i) {
                        if (zM.fleet[i] === this) break;
                    }

                    if (i !== zM.fSel) return;
                    var b = this.bots[this.cSel];

                    var toMoveSocket = [];
                    var toMove = [];
                    for (var i = 0; i < this.bots.length; ++i) {
                        // disconnect after 90 min of AFK, should bypass any bans from AFKing
                        if (Date.now() - this.bots[i].packets.lastSent > 5400000) this.bots[i].socket.close();
                        if (
                            this.bots[i].level === b.level &&
                            !this.bots[i].helping && !this.bots[i].deployed
                        ) toMove.push(this.bots[i]), toMoveSocket.push(this.bots[i].socket);
                    }

                    if (zM.movement) {
                        if (zM.useWallhack) {
                            // solve first
                            toMove.forEach(X => {
                                // random here is anti-idle disconnect
                                if (!(X.realX === zM.pos.mouseX>>1 && X.realY === zM.pos.mouseY>>1) || (Math.random() >= 0.99 && zM.antiIdle)) {
                                    var moves = zM.path(X.realX, X.realY, zM.pos.mouseX>>1, zM.pos.mouseY>>1, b.obj, b.grid);
                                    X.toDo = moves;
                                    X.realX = zM.pos.mouseX>>1;
                                    X.realY = zM.pos.mouseY>>1;
                                }
                            });

                            // then send packets synchronized, all together
                            var count = 0;
                            for (var i = 0; i < toMove.length; ++i) count = Math.max(count, toMove[i].toDo.length);
                            for (var i = 0; i < count; ++i) {
                                toMove.forEach(X => {
                                    if (i >= X.toDo.length) return;
                                    zM.packet.moveSocket([X.socket], X.toDo[i][0], X.toDo[i][1], [X]);
                                });
                            }
                        } else {
                            zM.packet.moveSocket(toMoveSocket, zM.pos.mouseX>>1, zM.pos.mouseY>>1, toMove);
                        }
                    } else {
                        if (zM.clickQueue >= 1) {
                            toMove.forEach(X => {
                                var moves = zM.path(X.realX, X.realY, zM.pos.mouseX>>1, zM.pos.mouseY>>1, b.obj, b.grid);
                                moves.forEach(Y => {
                                    zM.packet.moveSocket([X.socket], Y[0], Y[1], [X]);
                                });
                            });
                        }
                        if (zM.antiIdle) {
                            toMove.forEach(X => {
                                // random here is anti-idle disconnect, if movement is disabled.
                                if (Date.now() - X.packets.lastSent > 10000) zM.packet.moveSocket([X.socket], X.realX, X.realY, [X]);
                            });
                        }
                    }

                    for (;0<zM.clickQueue;--zM.clickQueue) zM.packet.clickSocket(toMoveSocket, zM.pos.mouseX>>1, zM.pos.mouseY>>1, toMove);

                    if (zM.drawing && zM.movement) zM.packet.drawSocket(toMoveSocket, zM.pos.lastMouseX>>1, zM.pos.lastMouseY>>1, zM.pos.mouseX>>1, zM.pos.mouseY>>1, toMove);

                    zM.pos.lastMouseX = zM.pos.mouseX;
                    zM.pos.lastMouseY = zM.pos.mouseY;
                }, 50);
            }
        }`,
        /* zM bot class */
        `zM.dependenciesLoaded |= 0b000100

        zM.Bot = class {
            constructor (fleet) {
                this.fleet = fleet;

                this.socket = new WebSocket(this.fleet.wsIp);
                this.socket.binaryType = 'arraybuffer';

                this.opened = false;
                this.isOpen = false;

                this.level = -1;
                this.id = -1;

                this.obj = [];
                this.players = [];
                this.clicks = [];
                this.drawings = [];

                this.realX = 400;
                this.realY = 300;

                this.local = 0;

                this.packets = {
                    receivedTotal: 0,
                    sentTotal: 0,
                    receivedPS: 0,
                    sentPS: 0
                }

                this.toHelp = [];
                this.toDo = [];

                this.timestampStart = 0;

                this.helping = false;
                this.deployed = false;

                this.prevLevels = [];

                this.socket.addEventListener('open', () => {
                    ++this.fleet.botsOpened;
                    this.opened = true;
                    this.isOpen = true;
                    ++this.fleet.botsPinged;
                    zM.updateTabs();
                });

                this.socket.addEventListener('close', () => {
                    if (this.opened) {
                        --this.fleet.botsOpened;
                        var i = this.fleet.ids.indexOf(this.id);
                        if (i != -1) this.fleet.ids.splice(i, 1);
                    }
                    else ++this.fleet.botsPinged;
                    this.isOpen = false;

                    for (var i = 0; i < this.fleet.bots.length; ++i) if (this.fleet.bots[i] == this) { console.log('y'); this.fleet.bots.splice(i,1); break; }

                    zM.updateTabs();
                });

                this.socket.addEventListener('message', ev => {
                    var buf = ev.data,
                        dat = new DataView(buf);

                    ++this.packets.receivedTotal;
                    ++this.packets.receivedPS;
                    setTimeout(()=>{--this.packets.receivedPS},1000);

                    var type = dat.getUint8(0);
                    if (type === 0) this.id = dat.getUint32(1, 1), ++this.fleet.botsIds, this.fleet.ids.push(this.id);
                    else if (type === 1) {
                        var out = zM.parse.cursors(buf, 1);
                        this.local = out.shift();
                        var players = out.shift();
                        var idsHere = [];
                        // create players for all new ids
                        for (var i = 0; i < players.length; ++i) {
                            if (players[i].id === this.id) this.realX = players[i].x, this.realY = players[i].y;
                            var index = -1;
                            for (var j = 0; j < this.players.length; ++j) {
                                if (this.players[j].id === players[i].id) {index = j; break;}
                            }
                            if (index === -1) this.players.push({
                                x: players[i].x,
                                y: players[i].y,
                                lastUpdate: Date.now(),
                                ox: players[i].x,
                                oy: players[i].y,
                                id: players[i].id
                            });
                        }
                        for (var i = 0; i < players.length; ++i) {
                            for (var j = 0; j < this.players.length; ++j) {
                                if (this.players[j].id === players[i].id) {
                                    this.players[j].ox = zM.ease(this.players[j].x, this.players[j].ox, this.players[j].lastUpdate);
                                    this.players[j].oy = zM.ease(this.players[j].y, this.players[j].oy, this.players[j].lastUpdate);
                                    this.players[j].x = players[i].x;
                                    this.players[j].y = players[i].y;
                                    this.players[j].lastUpdate = Date.now();
                                    idsHere.push(players[i].id);
                                }
                            }
                        }
                        var nPl = [];
                        for (var i = 0; i < idsHere.length; ++i) {
                            for (var j = 0; j < this.players.length; ++j) {
                                if (idsHere[i] === this.players[j].id) nPl.push(this.players[j]);
                            }
                        }

                        this.players = nPl;
                        var off = out.shift();

                        out = zM.parse.clicks(buf, off);
                        for (;0 < out[0].length;) {
                            this.clicks.push(out[0].shift());
                        }

                        off = out.pop();

                        out = zM.parse.remove(buf, off);
                        for (var i = 0; i < out[0].length; ++i) {
                            for (var j = 0; j < this.obj.length; ++j) {
                                if (this.obj[j].id === out[0][i]) {this.obj.splice(j, 1); break;}
                            }
                        }

                        off = out.pop();

                        out = zM.parse.objects(buf, off);
                        var obj = zM.parse.objData(out.shift());

                        for (var i = 0; i < obj.length; ++i) {
                            var index = -1;
                            for (var j = 0; j < this.obj.length; ++j) {
                                if (this.obj[j].id === obj[i].id) {index = j; break;};
                            }
                            if (index === -1) this.obj.push(obj[i]);
                            else this.obj[j] = obj[i];
                        }

                        off = out.pop();

                        out = zM.parse.drawing(buf, off);

                        for (var i = 0; i < out[0].length; ++i) {
                            var todo = out[0][i];
                            this.drawings.push(todo);
                        }

                        this.online = dat.getUint32(dat.byteLength-4, true);

                    } else if (type === 4) {
                        if (this.level === -1) ++this.fleet.botsLevels;

                        this.drawings = [];

                        this.realX = dat.getUint16(1, true);
                        this.realY = dat.getUint16(3, true);

                        var objdata = zM.parse.objects(buf, 5);
                        this.obj = zM.parse.objData(objdata[0]);

                        var grid = 100;
                        for (var i = 0; i < this.obj.length; ++i) {
                            if (grid <= 1) {grid = 1; break;}
                            if (this.obj[i].type === 1) if (
                                (this.obj[i].x/grid|0) != (this.obj[i].x/grid) ||
                                (this.obj[i].y/grid|0) != (this.obj[i].y/grid) ||
                                (this.obj[i].w/grid|0) != (this.obj[i].w/grid) ||
                                (this.obj[i].h/grid|0) != (this.obj[i].h/grid)
                            ) --grid, i = -1;
                        }

                        this.grid = grid;

                        var compare = [];
                        for (var i = 0; i < this.obj.length; ++i) {
                            var o = this.obj[i];
                            if (o.type === 0) {
                                compare.push({
                                    x: o.x,
                                    y: o.y,
                                    size: o.size,
                                    content: o.content
                                });
                            } else if (o.type === 1) {
                                if (o.color === '#000000') compare.push({
                                    x: o.x,
                                    y: o.y,
                                    w: o.w,
                                    h: o.h
                                });
                            } else if (o.type === 2) {
                                compare.push({
                                    x: o.x,
                                    y: o.y,
                                    w: o.w,
                                    h: o.h,
                                    isBad: o.isBad
                                });
                            } else {
                                compare.push({
                                    x: o.x,
                                    y: o.y,
                                    w: o.w,
                                    h: o.h,
                                    color: o.color
                                });
                            }
                        }
                        compare = JSON.stringify(compare);
                        var i = this.prevLevels.indexOf(compare);
                        if (i != -1) this.level = i;
                        else ++this.level, this.prevLevels.push(compare);

                        this.fleet.updateTabs();
                    } else if (type === 5) {
                        this.realX = dat.getUint16(1, true);
                        this.realY = dat.getUint16(3, true);
                    }
                });
            }
        }`,

        `zM.dependenciesLoaded |= 0b001000;
        let ctx;

        zM.reqLoad = function() {
            zM.r = {
                fps: 0,
                hudText: function(text, x, y, subLength = 0, color = '#fff') {
                    ctx.font = '12px "Nova Flat"';
                    ctx.globalAlpha = .5;
                    ctx.lineWidth = 3;
                    ctx.strokeStyle = '#000';
                    ctx.strokeText(text, x-ctx.measureText(text).width*subLength, y);

                    ctx.globalAlpha = 1;
                    ctx.fillStyle = color;
                    ctx.fillText(text, x-ctx.measureText(text).width*subLength, y);
                },
                infLvl: 5
            }
            ctx = zM.el.game.getContext('2d');
        }

        zM.reqDo = function() {
            ctx.clearRect(0, 0, 800, 600);

            if (isNaN(zM.fSel)) {
                ctx.font = '60px "Nova Flat"';
                ctx.fillStyle = '#000';
                var t = 'No fleet selected';
                ctx.fillText(t, 400 - ctx.measureText(t).width/2, 330);
            } else if (!zM.fleet[zM.fSel]) {
                ctx.font = '60px "Nova Flat"';
                ctx.fillStyle = '#000';
                var t = 'Undefined fleet selected';
                ctx.fillText(t, 400 - ctx.measureText(t).width/2, 330);
            } else if (!zM.fleet[zM.fSel].bots[zM.fleet[zM.fSel].cSel]) { // you can get this if the bot no longer exists (or an invalid bot selected state)
                ctx.font = '60px "Nova Flat"';
                ctx.fillStyle = '#000';
                var t = 'Lost connection to server';
                ctx.fillText(t, 400 - ctx.measureText(t).width/2, 330);
            } else {
                var b = zM.fleet[zM.fSel].bots[zM.fleet[zM.fSel].cSel];

                if (b.socket.readyState == WebSocket.CONNECTING) {
                    ctx.font = '60px "Nova Flat"';
                    ctx.fillStyle = '#000';
                    var t = 'Connecting';
                    ctx.fillText(t, 400 - ctx.measureText(t).width/2, 330);
                } else { // we don't need to handle closed states, the bot will be gone
                    for (var i = 0; i < b.obj.length; ++i) {
                        var o = b.obj[i];
                        switch (o.type) {
                            case 0:
                                ctx.globalAlpha = 1;
                                ctx.font = o.size + 'px "Nova Flat"';
                                ctx.fillStyle = '#000';
                                var x = o.x*2;
                                if (o.isCentered) x -= ctx.measureText(o.content).width/2;
                                ctx.fillText(o.content, x, o.y*2);
                                break;
                            case 1:
                                ctx.globalAlpha = 1;
                                ctx.fillStyle = o.color;
                                ctx.fillRect(o.x*2, o.y*2, o.w*2, o.h*2);
                                ctx.globalAlpha = .2;
                                ctx.strokeStyle = '#000';
                                ctx.lineWidth = 1;
                                ctx.strokeRect(o.x*2+1, o.y*2+1, o.w*2-2, o.h*2-2);
                                break;
                            case 2:
                                ctx.globalAlpha = .2;
                                ctx.fillStyle = o.isBad?'#f00':'#0f0';
                                ctx.fillRect(o.x*2, o.y*2, o.w*2, o.h*2);
                                break;
                            case 3:
                                ctx.globalAlpha = .2;
                                ctx.fillStyle = o.color;
                                ctx.fillRect(o.x*2, o.y*2, o.w*2, o.h*2);
                                ctx.font = '24px "Nova Flat"';
                                ctx.fillStyle = '#000';
                                ctx.globalAlpha = .5;
                                ctx.fillText(o.count, o.x*2 + o.w - ctx.measureText(o.count).width/2, o.y*2+o.h+9);
                                break;
                            case 4:
                                ctx.globalAlpha = 1;
                                ctx.fillStyle = o.color;
                                ctx.fillRect(o.x*2, o.y*2, o.w*2, o.h*2);

                                ctx.globalAlpha = .2;
                                ctx.fillStyle = '#000';
                                ctx.fillRect(o.x*2, o.y*2, o.w*2, o.h*2);

                                ctx.globalAlpha = 1;
                                ctx.fillStyle = o.color;
                                ctx.fillRect(o.x*2+8, o.y*2+8, o.w*2-16, o.h*2-16);

                                var l = Date.now() - o.lastClickAt > 150 ? 8 : 16;

                                ctx.globalAlpha = .2;
                                ctx.lineWidth = 1;
                                ctx.strokeStyle = '#000';
                                ctx.beginPath(),

                                ctx.moveTo(o.x*2+1, o.y*2+1),
                                ctx.lineTo(o.x*2+o.w*2-2, o.y*2+1),
                                ctx.lineTo(o.x*2+o.w*2-2, o.y*2+o.h*2-2),
                                ctx.lineTo(o.x*2+1, o.y*2+o.h*2-2),
                                ctx.lineTo(o.x*2+1, o.y*2+1);

                                ctx.lineTo(o.x*2+1+l, o.y*2+1+l);

                                ctx.moveTo(o.x*2+o.w*2-2, o.y*2+1),
                                ctx.lineTo(o.x*2+o.w*2-2-l, o.y*2+1+l);

                                ctx.moveTo(o.x*2+o.w*2-2, o.y*2+o.h*2-2);
                                ctx.lineTo(o.x*2+o.w*2-2-l, o.y*2+o.h*2-2-l);

                                ctx.moveTo(o.x*2+1, o.y*2+o.h*2-2),
                                ctx.lineTo(o.x*2+1+l, o.y*2+o.h*2-2-l);

                                ctx.moveTo(o.x*2+l, o.y*2+l),
                                ctx.lineTo(o.x*2+o.w*2-l, o.y*2+l),
                                ctx.lineTo(o.x*2+o.w*2-l, o.y*2+o.h*2-l),
                                ctx.lineTo(o.x*2+l, o.y*2+o.h*2-l),
                                ctx.lineTo(o.x*2+l, o.y*2+l);

                                ctx.stroke();

                                ctx.fillStyle = '#000';
                                ctx.globalAlpha = .5;
                                ctx.font = '24px "Nova Flat"';
                                ctx.fillText(o.count, o.x*2 + o.w - ctx.measureText(o.count).width/2, o.y*2+o.h+9);
                        }
                    }

                    ctx.strokeStyle = '#000';
                    ctx.lineWidth = 1.5;
                    for (var i = 0; i < b.clicks.length; ++i) {
                        if (typeof b.clicks[i] === 'object') {
                            if (Date.now() - b.clicks[i].time > 500) b.clicks.splice(i--, 1);
                            else {
                                ctx.beginPath(),
                                ctx.globalAlpha = .5 - (Date.now() - b.clicks[i].time)/1000;
                                var radius = (Date.now() - b.clicks[i].time)/20;
                                if (radius < 0.1) radius = 0.1 // Anti-crash failsafe
                                ctx.arc(b.clicks[i].x*2, b.clicks[i].y*2, radius, 0, 2*Math.PI);
                                ctx.stroke();
                            }
                        }
                    }

                    ctx.strokeStyle = '#000';
                    ctx.lineWidth = 1;
                    ctx.globalAlpha = .3;
                    for (var i = 0; i < b.drawings.length; ++i) {
                        if (typeof b.drawings[i] === 'object') {
                            if (Date.now() - b.drawings[i].time > 10000) b.drawings.splice(i--, 1);
                            else {
                                ctx.beginPath(),
                                ctx.moveTo(b.drawings[i].x*2, b.drawings[i].y*2);
                                ctx.lineTo(b.drawings[i].x2*2, b.drawings[i].y2*2);
                                ctx.stroke();
                            }
                        }
                    }

                    ctx.globalAlpha = 1;
                    ctx.font = '12px "Nova Flat"';
                    for (var i = 0; i < b.players.length; ++i) {
                        var img;
                        if (zM.fleet[zM.fSel].ids.indexOf(b.players[i].id) != -1) {
                            for (var j = 0; i < zM.fleet[zM.fSel].bots.length; ++j) {
                                if (zM.fleet[zM.fSel].bots[j]) if (zM.fleet[zM.fSel].bots[j].id == b.players[i].id) break;
                            }
                            if (j < zM.fleet[zM.fSel].bots.length) {
                                if (zM.fleet[zM.fSel].bots[j].helping || zM.fleet[zM.fSel].bots[j].deployed) img = zMIMG.movement.img;
                                else img = zMIMG.local.img;
                            } else img = zMIMG.local.img;
                        } else img = zMIMG.cursor.img;
                        var x = zM.ease(b.players[i].x, b.players[i].ox, b.players[i].lastUpdate)*2;
                        var y = zM.ease(b.players[i].y, b.players[i].oy, b.players[i].lastUpdate)*2;
                        ctx.drawImage(img, x-4, y-5);
                        x += 5;
                        y-= 2;
                        if (x + ctx.measureText(\`\${b.players[i].id}\`).width > 790) x = 790 - ctx.measureText(\`\${b.players[i].id}\`).width;
                        if (y - 12 < 10) y = 22;
                        zM.r.hudText(\`\${b.players[i].id}\`, x, y);
                    }

                    switch (zM.r.infLvl) {
                        case 5:
                            zM.r.hudText(\`Received (tp/ps) \${b.packets.receivedTotal} / \${b.packets.receivedPS}\`, 790, 22, 1);
                            zM.r.hudText(\`Sent (tp/ps) \${b.packets.sentTotal} / \${b.packets.sentPS}\`, 790, 36, 1);
                        case 4:
                        case 3:
                        case 2:
                            zM.r.hudText(\`Level \${b.level}\`, 10, 36);
                        case 1:
                            zM.r.hudText(
                                b.local < 30 ? 'Use shift+click to draw ('+b.local+' > 0)' :
                                b.local < 100 ? 'Too many cursors, drawing is disabled ('+b.local+' > 30)' :
                                b.local < 1010 ? 'Too many cursors, not all cursors are shown ('+b.local+' > 100)' :
                                'Use shift+click to draw ('+b.local+' > 1010)'
                            , 10, 590);

                            zM.r.hudText(\`\${b.online} players online\`, 790, 590, 1);
                        case 0:
                    }
                }
            }

            if (zM.r.infLvl >= 2) zM.r.hudText('FPS ' + zM.r.fps,10,22);

            ctx.fillStyle = zM.movement?'#ff0':'#f00';
            ctx.globalAlpha = 0.3;
            ctx.beginPath();
            ctx.moveTo(zM.pos.mouseX+3, zM.pos.mouseY+6);
            ctx.arc(zM.pos.mouseX+3, zM.pos.mouseY+6, 20, 0, 2*Math.PI);
            ctx.fill();

            ctx.globalAlpha = 1;
            ctx.drawImage(zMIMG.cursor.img, zM.pos.mouseX-4, zM.pos.mouseY-5);


            zM.reqAfter();
        }

        zM.reqAfter = function() {
            ++zM.r.fps;
            requestAnimationFrame(zM.reqDo);
            setTimeout(()=>{--zM.r.fps},1000);
        }`,

        /* zM packet parsing */
        `zM.dependenciesLoaded |= 0b010000;

        zM.parse = {
            /**
             * Returns an array as [local, players, offset]
             */
            cursors: function(buffer, offset) {
                var dat = new DataView(buffer);
                var local = dat.getUint16(offset, true);

                offset += 2;

                var players = [];
                for (var i = 0; i < local; ++i) {
                    players.push({
                        id: dat.getUint32(offset, true),
                        x: dat.getUint16(offset + 4, true),
                        y: dat.getUint16(offset + 6, true),
                    });
                    offset += 8;
                }

                return [local, players, offset];
            },

            /**
             * Returns an array as [clicks, offset]
             */
            clicks: function(buffer, offset) {
                var dat = new DataView(buffer);

                var count = dat.getUint16(offset, true);
                var clicks = [];

                offset += 2;
                for (var i = 0; i < count; ++i) {
                    clicks.push({
                        x: dat.getUint16(offset, true),
                        y: dat.getUint16(offset + 2, true),
                        time: Date.now()
                    });
                    offset += 4;
                }

                return [clicks, offset];
            },

            /**
             * Returns an array as [clicks, offset]
             */
            drawing: function(buffer, offset) {
                var dat = new DataView(buffer);

                var count = dat.getUint16(offset, true);
                var drawings = [];

                offset += 2;
                for (var i = 0; i < count; ++i) {
                    drawings.push({
                        x: dat.getUint16(offset, true),
                        y: dat.getUint16(offset + 2, true),
                        x2: dat.getUint16(offset + 4, true),
                        y2: dat.getUint16(offset + 6, true),
                        time: Date.now()
                    });
                    offset += 8;
                }

                return [drawings, offset];
            },

            /**
             * Returns an array as [ids, offset]
             */
            remove: function(buffer, offset) {
                var dat = new DataView(buffer);

                var count = dat.getUint16(offset, true);
                var ids = [];

                offset += 2;
                for (var i = 0; i < count; ++i) {
                    ids.push(dat.getUint32(offset, true));
                    offset += 4;
                }

                return [ids, offset];
            },

            /**
             * Parses objdata (string) and outputs as objects.
             */
            objData: function(objdata) {
                var obj = objdata;
                var nObj = [];

                for (var i = 0; i < obj.length; ++i) {
                    var nO = {};
                    obj[i] = obj[i].split(/\\.+/g);
                    nO.id = parseInt(obj[i].shift());
                    var type = obj[i].shift();
                    switch (type) {
                        case '0':
                            nO.type = 0;
                            nO.x = parseInt(obj[i].shift());
                            nO.y = parseInt(obj[i].shift());
                            nO.size = parseInt(obj[i].shift());
                            nO.isCentered = obj[i].shift() === 'false'?false:true;

                            nO.content = obj[i].join('');
                            break;
                        case '1':
                            nO.type = 1;
                            nO.x = parseInt(obj[i].shift());
                            nO.y = parseInt(obj[i].shift());
                            nO.w = parseInt(obj[i].shift());
                            nO.h = parseInt(obj[i].shift());
                            nO.color = obj[i].shift();
                            break;
                        case '2':
                            nO.type = 2;
                            nO.x = parseInt(obj[i].shift());
                            nO.y = parseInt(obj[i].shift());
                            nO.w = parseInt(obj[i].shift());
                            nO.h = parseInt(obj[i].shift());
                            nO.isBad = obj[i].shift()==='false'?false:true;
                            break;
                        case '3':
                            nO.type = 3;
                            nO.x = parseInt(obj[i].shift());
                            nO.y = parseInt(obj[i].shift());
                            nO.w = parseInt(obj[i].shift());
                            nO.h = parseInt(obj[i].shift());
                            nO.count = parseInt(obj[i].shift());
                            nO.color = obj[i].shift();
                            break;
                        case '4':
                            nO.type = 4;
                            nO.x = parseInt(obj[i].shift());
                            nO.y = parseInt(obj[i].shift());
                            nO.w = parseInt(obj[i].shift());
                            nO.h = parseInt(obj[i].shift());
                            nO.count = parseInt(obj[i].shift());
                            nO.color = obj[i].shift();
                            nO.lastClickAt = 0;
                            break;
                    }

                    nObj.push(nO);
                }

                return nObj;
            },

            /**
             * Returns an array as [objdata, offset]
             * objdata is required to be further parsed with zM.parse.objData()
             */
            objects: function(buffer, offset) {
                var dat = new DataView(buffer);

                var count = dat.getUint16(offset, true);
                var objdata = [];

                offset += 2;
                for (var i = 0; i < count; ++i) {
                    var id = dat.getUint32(offset, true);
                    offset += 4;
                    var type = dat.getUint8(offset);
                    var objdat = id+'.';
                    ++offset;
                    switch (type) {
                        case 0:
                            objdat += '0.';
                            objdat += \`\${dat.getUint16(offset, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+2, true)}.\`;
                            objdat += \`\${dat.getUint8(offset+4)}.\`;
                            objdat += \`\${!!dat.getUint8(offset+5)}.\`;
                            offset += 5;
                            for (;1;) if (dat.getUint8(++offset) != 0) objdat += String.fromCharCode(dat.getUint8(offset));
                                      else break;

                            ++offset;
                            break;
                        case 1:
                            objdat += '1.';
                            objdat += \`\${dat.getUint16(offset, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+2, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+4, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+6, true)}.\`;
                            var color = dat.getUint32(offset+8, true).toString(16);
                            for (; color.length < 6;) color = '0' + color;
                            objdat += '#' + color + '.';

                            offset += 12;

                            break;

                        case 2:
                            objdat += '2.';
                            objdat += \`\${dat.getUint16(offset, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+2, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+4, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+6, true)}.\`;
                            objdat += \`\${!!dat.getUint8(offset+8)}.\`;
                            offset += 9;
                            break;

                        case 3:
                            objdat += '3.';
                            objdat += \`\${dat.getUint16(offset, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+2, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+4, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+6, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+8, true)}.\`;
                            var color = dat.getUint32(offset+10, true).toString(16);
                            for (; color.length < 6;) color = '0' + color;
                            objdat += '#' + color + '.';

                            offset += 14;
                            break;

                        case 4:
                            objdat += '4.';
                            objdat += \`\${dat.getUint16(offset, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+2, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+4, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+6, true)}.\`;
                            objdat += \`\${dat.getUint16(offset+8, true)}.\`;
                            var color = dat.getUint32(offset+10, true).toString(16);
                            for (; color.length < 6;) color = '0' + color;
                            objdat += '#' + color + '.';
                            offset += 14;
                            break;
                    }
                    objdata.push(objdat);
                }

                return [objdata, offset];
            }
        }

        zM.packet = {
            /**
             *
             * @param {array} sockets Array of sockets to move
             */
            moveSocket(sockets, x, y, bots) {
                var buf = new ArrayBuffer(9),
                    dat = new DataView(buf);

                dat.setUint8(0, 1);
                dat.setUint16(1, x, true);
                dat.setUint16(3, y, true);
                dat.setUint32(5, -1, true);
                sockets.forEach(X => typeof X.send === 'function' && X.send(buf));
                bots.forEach(X => {
                    ++X.packets.sentTotal;
                    ++X.packets.sentPS;
                    setTimeout(()=>{--X.packets.sentPS},1000);
                    X.packets.lastSent = Date.now();
                });
            },

            /**
             *
             * @param {array} sockets Array of sockets to click with
             */
            clickSocket(sockets, x, y, bots) {
                var buf = new ArrayBuffer(9),
                    dat = new DataView(buf);

                dat.setUint8(0, 2);
                dat.setUint16(1, x, true);
                dat.setUint16(3, y, true);
                dat.setUint32(5, -1, true);
                sockets.forEach(X => typeof X.send === 'function' && X.send(buf));
                bots.forEach(X => {
                    ++X.packets.sentTotal;
                    ++X.packets.sentPS;
                    setTimeout(()=>{--X.packets.sentPS},1000);
                    X.packets.lastSent = Date.now();
                });
            },

            /**
             *
             * @param {array} sockets Array of sockets to draw with
             */
            drawSocket(sockets, x, y, x2, y2, bots) {
                var buf = new ArrayBuffer(9),
                    dat = new DataView(buf);

                dat.setUint8(0, 3);
                dat.setUint16(1, x, true);
                dat.setUint16(3, y, true);
                dat.setUint16(5, x2, true);
                dat.setUint16(7, y2, true);
                sockets.forEach(X => typeof X.send === 'function' && (X.send(buf)));
                bots.forEach(X => {
                    ++X.packets.sentTotal;
                    ++X.packets.sentPS;
                    setTimeout(()=>{--X.packets.sentPS},1000);
                    X.packets.lastSent = Date.now();
                });
            }
        }`,
        /* zM Pathfinder */
        `zM.dependenciesLoaded |= 0b100000;

        var visit = [];

        zM.dos = function(dx, dy, items, gridSpace) {
            var gridX = 400/gridSpace,
                gridY = 300/gridSpace;
            var grid = [];
            visit = [];
            for (var i = 0; i < gridY; i++) {
                grid[i] = [];
                visit[i] = [];
                for (var j = 0; j < gridX; j++) grid[i][j] = 0, visit[i][j] = 0;
            }
            items.forEach(function(d) {
                if (d.type === 1) {
                    for (var i = 0; i < d.h; i+=gridSpace) {
                        for (var j = 0; j < d.w; j+=gridSpace) {
                            grid[(d.y+i)/gridSpace][(d.x+j)/gridSpace] = 3;
                        }
                    }
                }
            });
            var bfs = [[dx,dy]],
                bfs2 = [];
            while (bfs.length) {
                bfs.forEach(function(dat) {
                    var x = dat[0],
                        y = dat[1];
                    if (grid[y][x] == 3) return;
                    grid[y][x] = 3;
                    for (var X = x + 1; X < gridX && !(grid[y][X] & 1); X++) {
                        grid[y][X] |= 1;
                        if (!visit[y][X]) {
                            visit[y][X] = [x, y], bfs2.push([X, y]);
                        }
                    }
                    for (var X = x - 1; X >= 0 && !(grid[y][X] & 1); X--) {
                        grid[y][X] |= 1;
                        if (!visit[y][X]) {
                            visit[y][X] = [x, y], bfs2.push([X, y]);
                        }
                    }
                    for (var Y = y + 1; Y < gridY && !(grid[Y][x] & 2); Y++) {
                        grid[Y][x] |= 2;
                        if (!visit[Y][x]) {
                            visit[Y][x] = [x, y], bfs2.push([x, Y]);
                        }
                    }
                    for (var Y = y - 1; Y >= 0 && !(grid[Y][x] & 2); Y--) {
                        grid[Y][x] |= 2;
                        if (!visit[Y][x]) {
                            visit[Y][x] = [x, y], bfs2.push([x, Y]);
                        }
                    }
                });
                bfs = bfs2;
                bfs2 = [];
            }
        }

        zM.path = function(ox, oy, dx, dy, items, grid) {

            var rdx = dx, rdy = dy;
            ox /= grid;
            oy /= grid;
            dx /= grid;
            dy /= grid;

            ox |= 0;
            oy |= 0;
            dx |= 0;
            dy |= 0;

            var mov = [];
            if (!(ox == dx && oy == dy)) {
                zM.dos(ox,oy,items,grid);
                var xy2 = [dx,dy];
                while (visit[xy2[1]][xy2[0]]) {
                    mov.push(xy2);
                    xy2 = visit[xy2[1]][xy2[0]];
                }

                mov.reverse();
            }

            for (var i = 0; i < mov.length; ++i) {
                mov[i][0] *= grid;
                mov[i][0] += grid/2;
                mov[i][1] *= grid;
                mov[i][1] += grid/2;
            }
            mov.push([rdx, rdy]);
            return mov;
        }`
    ],

    // ---
    // add elements from functions below:
    // ---

[
    function(c) {
        var temp = [D.createElement('link')];
        temp[0].href = 'https://fonts.googleapis.com/css?family=Montserrat:100,200,300,400';
        temp[0].rel = 'stylesheet';
        head.appendChild(temp[0]);

        temp[1] = D.createElement('link');
        temp[1].href = 'https://fonts.googleapis.com/css?family=Nova+Flat&display=swap';
        temp[1].rel = 'stylesheet';
        head.appendChild(temp[1]);
    },
    function(c) { // news area (null fleet selected)
        var temp = [D.createElement('div')];
        temp[0].className = 'zM-news-outer';

        temp[1] = D.createElement('div');
        temp[1].className = 'zM-news-container';
        temp[0].appendChild(temp[1]);

        temp[2] = D.createElement('div');
        temp[2].className = 'zM-news-page-top';
        temp[1].appendChild(temp[2]);

        temp[3] = D.createElement('div');
        temp[3].className = 'zM-news-page-mid';
        temp[1].appendChild(temp[3]);

        temp[4] = D.createElement('div');
        temp[4].className = 'zM-news-page-bottom';
        temp[1].appendChild(temp[4]);

        zM.el.newsOuter = temp[0];
        zM.el.newsPageTop = temp[2];
        zM.el.newsPageMid = temp[3];
        zM.el.newsPageBottom = temp[4];
        c.appendChild(temp[0]);
    },
    function(c) { // fleet prompt
        var temp = [D.createElement('div')];
        temp[0].className = 'zM-fleet-prompt-outer';

        temp[1] = D.createElement('div');
        temp[1].className = 'zM-fleet-prompt-container';
        temp[0].appendChild(temp[1]);

        temp[2] = D.createElement('div');
        temp[2].className = 'zM-fleet-prompt-title';
        temp[2].textContent = 'Create new fleet';
        temp[1].appendChild(temp[2]);


        temp[3] = D.createElement('div');
        temp[3].className = 'zM-fleet-prompt-entry';
        temp[1].appendChild(temp[3]);

        temp[4] = D.createElement('div');
        temp[4].className = 'zM-fleet-prompt-opt';
        temp[4].textContent = 'Use premade IPs'
        temp[3].appendChild(temp[4]);

        temp[5] = D.createElement('input');
        temp[5].type = 'checkbox';
        temp[5].checked = true;
        temp[5].addEventListener('change', () => {
            if (temp[5].checked) {
                temp[6].style.display = 'block';
                temp[10].style.display = 'none';
                temp[13].style.display = 'none';
                temp[16].style.display = 'none';
                if (temp[12].value + ':' + temp[15].value === `${ip}:2828`) temp[8].value = "0";
                else if (temp[12].value + ':' + temp[15].value === 'kursors.io/ws:8080') temp[8].value = "1";
                else if (temp[12].value + ':' + temp[15].value === 'localhost:9004') temp[8].value = "2";
            } else {
                temp[6].style.display = 'none';
                temp[10].style.display = 'block';
                temp[13].style.display = 'block';
                temp[16].style.display = 'block';

                // set input values
                switch (temp[8].value) {
                    case "0": {
                        temp[12].value = __ip;
                        temp[15].value = '2828';
                        break;
                    }
                    case "1": {
                        temp[12].value = 'kursors.io/ws';
                        temp[15].value = '8080';
                        break;
                    }
                    case "2": {
                        temp[12].value = 'localhost';
                        temp[15].value = '9004';
                        break;
                    }
                }

                temp[18].checked = false;
            }
        })
        temp[3].appendChild(temp[5]);


        temp[6] = D.createElement('div');
        temp[6].className = 'zM-fleet-prompt-entry';
        temp[1].appendChild(temp[6]);

        temp[7] = D.createElement('div');
        temp[7].className = 'zM-fleet-prompt-opt';
        temp[7].textContent = 'Select an IP to connect to:'
        temp[6].appendChild(temp[7]);

        // dropdown
        temp[8] = D.createElement('select');
        temp[8].style.outline = 'none';
        temp[6].appendChild(temp[8]);

        temp[9] = [D.createElement('option'), D.createElement('option'), D.createElement('option')];
        temp[9][0].value = "0";
        temp[9][0].textContent = `cursors.io ${__ip}:2828)`;
        temp[9][1].value = "1";
        temp[9][1].textContent = 'kursors.io (kursors.io/ws:8080)';
        temp[9][2].value = "2";
        temp[9][2].textContent = 'local server (localhost:9004)';

        temp[9].forEach(X => temp[8].appendChild(X));

        temp[10] = D.createElement('div');
        temp[10].className = 'zM-fleet-prompt-entry';
        temp[10].style.display = 'none';
        temp[1].appendChild(temp[10]);

        temp[11] = D.createElement('div');
        temp[11].className = 'zM-fleet-prompt-opt';
        temp[11].textContent = 'IP address:'
        temp[10].appendChild(temp[11]);
        temp[12] = D.createElement('input');
        temp[12].type = 'text';
        temp[12].style.outline = 'none';
        temp[12].width = 200;
        temp[10].appendChild(temp[12]);

        temp[13] = D.createElement('div');
        temp[13].className = 'zM-fleet-prompt-entry';
        temp[13].style.display = 'none';
        temp[1].appendChild(temp[13]);

        temp[14] = D.createElement('div');
        temp[14].className = 'zM-fleet-prompt-opt';
        temp[14].textContent = 'Port:'
        temp[13].appendChild(temp[14]);
        temp[15] = D.createElement('input');
        temp[15].type = 'text';
        temp[15].style.outline = 'none';
        temp[15].width = 200;
        temp[13].appendChild(temp[15]);

        temp[16] = D.createElement('div');
        temp[16].className = 'zM-fleet-prompt-entry';
        temp[16].style.display = 'none';
        temp[1].appendChild(temp[16]);

        temp[17] = D.createElement('div');
        temp[17].className = 'zM-fleet-prompt-opt';
        temp[17].textContent = 'Using IPv6:'
        temp[16].appendChild(temp[17]);
        temp[18] = D.createElement('input');
        temp[18].type = 'checkbox';
        temp[18].style.outline = 'none';
        temp[16].appendChild(temp[18]);

        // max bots
        temp[19] = D.createElement('div');
        temp[19].className = 'zM-fleet-prompt-entry';
        temp[1].appendChild(temp[19]);

        temp[20] = D.createElement('div');
        temp[20].className = 'zM-fleet-prompt-opt';
        temp[20].textContent = 'Maximum bots to this fleet:';
        temp[19].appendChild(temp[20]);

        temp[21] = D.createElement('input');
        temp[21].type = 'number';
        temp[19].appendChild(temp[21]);
        temp[21].addEventListener('change', ev => {
            var num = Math.min(100, Math.max(parseInt(temp[21].value), 0));
            if (isNaN(num)) num = 3;
            temp[21].value = num;
        });



        if (zM.origin === 'cursors') temp[21].value = 3;
        else temp[21].value = 20;


        // start bots
        temp[22] = D.createElement('div');
        temp[22].className = 'zM-fleet-prompt-entry';
        temp[1].appendChild(temp[22]);

        temp[23] = D.createElement('div');
        temp[23].className = 'zM-fleet-prompt-opt';
        temp[23].textContent = 'Start with # bots:';
        temp[22].appendChild(temp[23]);

        temp[24] = D.createElement('input');
        temp[24].type = 'number';
        temp[22].appendChild(temp[24]);
        temp[24].addEventListener('change', ev => {
            var num = Math.min(100, Math.max(parseInt(temp[24].value), 0));
            if (isNaN(num)) num = 3;
            temp[24].value = num;
        });

        if (zM.origin === 'cursors') temp[24].value = 3;
        else temp[24].value = 20;


        // zursor username
        temp[25] = D.createElement('div');
        temp[25].className = 'zM-fleet-prompt-entry';
        temp[1].appendChild(temp[25]);

        temp[26] = D.createElement('div');
        temp[26].className = 'zM-fleet-prompt-opt';
        temp[26].textContent = 'Fleet name (public):';
        temp[25].appendChild(temp[26]);

        temp[27] = D.createElement('input');
        temp[27].type = 'text';
        temp[25].appendChild(temp[27]);


        // confirmation buttons
        temp[28] = D.createElement('div');
        temp[28].className = 'zM-fleet-prompt-create';
        temp[28].textContent = 'Create fleet';
        temp[1].appendChild(temp[28]);

        temp[29] = D.createElement('div');
        temp[29].className = 'zM-fleet-prompt-cancel';
        temp[29].textContent = 'Cancel';
        temp[1].appendChild(temp[29]);

        temp[29].addEventListener('click', ev => {
            temp[0].style.display = 'none';
            if (isNaN(zM.fSel)) zM.el.newsOuter.style.display = 'block';
            else zM.updateTabs();
        });

        temp[28].addEventListener('click', () => {
            var ip, port, ipv6 = false;

            if (!temp[5].checked) ip = temp[12].value, port = temp[15].value, ipv6 = temp[18].checked;
            else {
                switch (temp[8].value) {
                    case "0": ip = __ip; port = '2828'; break;
                    case "1": ip = 'kursors.io/ws'; port = '8080'; break;
                    default: ip = 'localhost'; port = '9004'; break;
                }
            }

            var maxBots = parseInt(temp[21].value),
                startBots = parseInt(temp[24].value);

            var name = temp[27].value;
            if (name.length === 0) name = 'Unnamed fleet';
            zM.createFleet(ip, port, ipv6, maxBots, startBots, name);
        });



        // give temp to zM.el
        zM.el.fleetPromptOuter = temp[0];
        zM.el.fleetPromptContainer = temp[1];
        c.appendChild(temp[0]);
    },
    function(c) { // fleet topbar
        var temp = [D.createElement('div')];
        temp[0].className = 'zM-fleet-topbar-outer';
        c.appendChild(temp[0]);
        temp[4] = D.createElement('div');
        temp[4].className = 'zM-fleet-topbar-container';
        temp[0].appendChild(temp[4]);


        temp[1] = D.createElement('div');
        temp[1].className = 'zM-fleet-topbar-add';
        temp[4].appendChild(temp[1]);
        temp[1].addEventListener('click', ev => {
            zM.promptFleet();
        });


        temp[2] = D.createElement('img');
        temp[2].src = zMIMG.add.uri;
        temp[2].width = 45, temp[2].height = 35, temp[2].style.cssText = 'position:absolute;top:0px;left:0px;opacity:0.8;'
        temp[1].appendChild(temp[2]);


        temp[3] = D.createElement('div');
        temp[3].className = 'zM-fleet-topbar-hint';
        temp[3].textContent = 'Add fleet here to start playing';
        temp[0].appendChild(temp[3]);


        // give temp to zM.el
        zM.el.fleetTopbarOuter = temp.shift();
        zM.el.fleetTopbarContainer = temp.pop();
        zM.el.fleetTopbarAdd = temp.shift();
        zM.el.fleetTopbarAddImg = temp.shift();
        zM.el.fleetTopbarHint = temp.shift();
    },
    function(c) {
        var temp = document.createElement('canvas');

        temp.className = 'zM-game';
        temp.width = 800;
        temp.height = 600;
        c.appendChild(temp);

        temp.addEventListener('mousemove', ev => {
            zM.pos.mouseX = ev.offsetX;
            zM.pos.mouseY = ev.offsetY;
        });

        temp.addEventListener('mousedown', ev => {
            if (ev.shiftKey || ev.ctrlKey) zM.drawing = true;
            else {
                ++zM.clickQueue;
            }
        });

        temp.addEventListener('mouseup', ev => {
            zM.drawing = false;
        });

        addEventListener('keydown', ev => {
            if (isNaN(zM.fSel)) return;
            console.log(ev.key);
            switch (ev.key) {
                // all possible keys that can be drawn
                case 'A':
                case 'a':case 'B':case 'b':case 'C':case 'c':
                case 'D':case 'd':case 'E':case 'e':case 'F':
                case 'f':case 'G':case 'g':case 'H':case 'h':
                case 'I':case 'i':case 'J':case 'j':case 'K':
                case 'k':case 'L':case 'l':case 'M':case 'm':
                case 'N':case 'n':case 'O':case 'o':case 'P':
                case 'p':case 'Q':case 'q':case 'R':case 'r':
                case 'S':case 's':case 'T':case 't':case 'U':
                case 'u':case 'V':case 'v':case 'W':case 'w':
                case 'X':case 'x':case 'Y':case 'y':case 'Z':
                case 'z':case '1':case '2':case '3':case '4':
                case '5':case '6':case '7':case '8':case '9':
                case '0':case '!':case '@':case '#':case '$':
                case '%':case '^':case '&':case '*':case '(':
                case ')':case '`':case '~':case '-':case '_':
                case '=':case '+':case '[':case ']':case '{':
                case '}':case '\\':case '|':case ';':case ':':
                case '\'':case '"':case '.':case ',':case '<':
                case '>':case '/':case '?':

                case '¡':
                case '¢':case '£':case '¤':case '¥':case '¦':
                case '§':case '¨':case '©':case 'ª':case '«':
                case '¬':case '®':case '¯':case '°':case '±':
                case '²':case '³':case '´':case 'µ':case '¶':
                case '·':case '¸':case '¹':case 'º':case '»':
                case '¼':case '½':case '¾':case '¿':

                case 'À':
                case 'Á':case 'Â':case 'Ã':case 'Ä':case 'Å':
                case 'Æ':case 'Ç':case 'È':case 'É':case 'Ê':
                case 'Ë':case 'Ì':case 'Í':case 'Î':case 'Ï':
                case 'Ð':case 'Ñ':case 'Ò':case 'Ó':case 'Ô':
                case 'Õ':case 'Ö':case 'Ø':case 'Ù':case 'Ú':
                case 'Û':case 'Ü':case 'Ý':case 'Þ':case 'ß':
                case 'à':case 'á':case 'â':case 'ã':case 'ä':
                case 'å':case 'æ':case 'ç':case 'è':case 'é':
                case 'ê':case 'ë':case 'ì':case 'í':case 'î':
                case 'ï':case 'ð':case 'ñ':case 'ò':case 'ó':
                case 'ô':case 'õ':case 'ö':case 'ø':case 'ù':
                case 'ú':case 'û':case 'ü':case 'ý':case 'þ':
                case 'ÿ':
                    //zM.fleet[zM.fSel].
                    break;

                case 'F1':
                    ev.preventDefault();
                    zM.movement ^= 1;
                    break;

                case 'F2':
                    var f = zM.fleet[zM.fSel];
                    ev.preventDefault();
                    var i = 0;
                    for (; i < f.bots.length; ++i) {
                        if (f.bots[i]) if (!f.bots[i].deployed && !f.bots[i].helping && i !== f.cSel && f.bots[i].level === f.bots[f.cSel].level) break;
                    }
                    if (f.bots[i]) {
                        var b = f.bots[i];
                        b.deployed = true;
                    }
                    break;

                case 'F3':
                    var f = zM.fleet[zM.fSel];
                    ev.preventDefault();
                    var i = 0;
                    for (; i < f.bots.length; ++i) {
                        if (f.bots[i]) if (f.bots[i].deployed && f.bots[i].level === f.bots[f.cSel].level) break;
                    }
                    if (i < f.bots.length) {
                        var b = f.bots[i];
                        var s = f.bots[f.cSel];
                        b.deployed = false;

                        if (zM.returnAfterUndeployed) {
                            var moves = zM.path(b.realX, b.realY, s.realX, s.realY, b.obj, b.grid);
                            moves.forEach(Y => {
                                zM.packet.moveSocket([b.socket], Y[0], Y[1], [b]);
                            });
                        }
                    }
                    break;

                case 'F4':
                    ev.preventDefault();
                    var f = zM.fleet[zM.fSel];
                    if (f.isHelping) break;
                    f.isHelping = true;
                    var helpers = [];
                    for (var i = 0; i < f.bots.length; ++i) {
                        console.log(!f.bots[i].helping && !f.bots[i].deployed && i !== f.cSel && f.bots[i].level === f.bots[f.cSel].level);
                        console.log('1 ' + !f.bots[i].helping);
                        console.log('2 ' + !f.bots[i].deployed);
                        console.log('3 ' + (i !== f.cSel));
                        console.log('4 ' + (f.bots[i].level === f.bots[f.cSel].level));
                        if (f.bots[i]) if (!f.bots[i].helping && !f.bots[i].deployed && i !== f.cSel && f.bots[i].level === f.bots[f.cSel].level) helpers.push(f.bots[i]);
                    }
                    f.helpers = helpers;
                    break;

                case 'F5':
                    ev.preventDefault();
                    for (var i = 0; i < zM.fleet[zM.fSel].helpers.length; ++i) {
                        zM.fleet[zM.fSel].helpers[i].helping = false;
                    }
                    zM.fleet[zM.fSel].helpers = [];
                    zM.fleet[zM.fSel].isHelping = false;
                    break;
            }
        });

        zM.el.game = temp;

        zM.reqLoad();

        zM.reqDo();
    },

    function(c) {
        var temp = [D.createElement('div')];

        temp[0].className = 'zM-info-outer';
        c.appendChild(temp[0]);

        temp[1] = D.createElement('div');
        temp[1].className = 'zM-info-ver';
        temp[1].textContent = 'v' + zM.version;
        temp[0].appendChild(temp[1]);

        temp[2] = D.createElement('div');
        temp[2].className = 'zM-info-clog-button-outer';
        temp[0].appendChild(temp[2]);

        temp[3] = D.createElement('div');
        temp[3].className = 'zM-info-clog-button';
        temp[3].textContent = 'view changelog';
        temp[2].appendChild(temp[3]);




        temp[4] = D.createElement('div');
        temp[4].className = 'zM-clog-outer';
        c.appendChild(temp[4]);

        temp[5] = D.createElement('div');
        temp[5].className = 'zM-clog-container';
        temp[4].appendChild(temp[5]);


        temp[6] = D.createElement('div');
        temp[6].className = 'zM-clog-close';
        temp[4].appendChild(temp[6]);

        temp[7] = D.createElement('img');
        temp[7].src = zMIMG.close.uri;
        temp[6].appendChild(temp[7]);


        zM.el.clogOuter = temp[4];
        zM.el.clogContainer = temp[5];

        temp[2].addEventListener('click', ev => {
            zM.el.clogOuter.style.display = 'block';
        });
        temp[6].addEventListener('click', ev => {
            zM.el.clogOuter.style.display = 'none';
        });


        var out = [];
        for (var i = 0; i < zM.clog.length; ++i) {
            var temp = D.createElement('div');
            if (zM.clog[i].type === 'title') {
                temp.className = 'zM-clog-title';
                temp.innerHTML = zM.clog[i].msg;
                out.push(temp);
            } else {
                temp.className = 'zM-clog-desc';
                temp.innerHTML = zM.clog[i].msg;
                out.push(temp);
            }
        }

        out.forEach(X => zM.el.clogContainer.appendChild(X));
    }
]
);
})();