Greasy Fork is available in English.

Gartic phone DRAW bot

Auto drawing bot (DO NOT OVERDO)

// ==UserScript==
// @name         Gartic phone DRAW bot
// @namespace    http://tampermonkey.net/
// @version      1.1.2
// @license      LIT
// @description  Auto drawing bot (DO NOT OVERDO)
// @author       StickySkull & DoctorDeathDDrac
// @source       https://t.me/doctordeathddracula
// @source       https://t.me/stickyskull
// @supportURL   https://discord.gg/sHj5UauJZ4
// @match        *://garticphone.com/*
// @icon         https://www.google.com/s2/favicons?domain=garticphone.com
// @grant        none
// @run-at       document-start
// ==/UserScript==



class UpdateUserScript {
    constructor() {}

    cc(tag, options = {}, parent = false, init = false) {
        const children = options.children || [];
        delete options.children;
        const element = Object.assign(document.createElement(tag), options);
        for (const child of children) element.appendChild(child);
        if (init) init(element);
        return parent ? parent.appendChild(element) : element;
    }

    check() {
        fetch('https://greasyfork.org/ru/scripts/436728-garticphone-draw-bot/versions.json').then(resp => resp.json().then(json => {
            if (json[0].version != GM.info.script.version) {
                this.base = this.cc('div', {
                    style: `width:100%;z-index:10;position:fixed;top:0;transform:scale(${((window.innerWidth - (window.innerWidth < 1920 ? 180 : 320)) / 1150) / 1.5});display:flex;flex-direction:column;align-items:center;transform-origin:top;`,
                    children: [
                        this.cc('div', {
                            style: 'margin-top:5px;display:flex;flex-direction:row;gap:5px;padding:5px 20px;border-radius:5px;border:2px solid white;background-color:#ffffff55;',
                            children: [
                                this.cc('div', {
                                    style:  `color:red;font-family:'Black';font-size:20px;text-align:center;`,
                                    textContent: GM.info.script.version
                                }),
                                this.cc('div', {
                                    style:  `color:white;font-family:'Black';font-size:20px;text-align:center;`,
                                    textContent: String.fromCharCode(10140)
                                }),
                                this.cc('div', {
                                    style:  `color:lime;font-family:'Black';font-size:20px;text-align:center;`,
                                    textContent: json[0].version
                                }),
                                this.cc('a', {
                                    href: 'https://greasyfork.org/scripts/436728-garticphone-draw-bot/code/Garticphone%20DRAW%20bot.user.js',
                                    style:  `margin-left:12px;color:aqua;font-family:'Black';font-size:20px;text-align:center;`,
                                    textContent: 'UPDATE',
                                    onclick: () => {this.base.remove()}
                                }),
                            ]
                        })
                    ]
                }, document.documentElement);
            }
        }));
    }
}

(new UpdateUserScript()).check();


class Log {
    constructor(parent=document.documentElement, maxCount=5, stayTime=3000, disappearanceTime=1000) {
        this.Id = Math.random();
        this.stayTime = stayTime;
        this.maxCount = maxCount;
        this.disappearanceTime = disappearanceTime;
        this.defaultStyle = '';
        this.element = this.cc('div', {
            style: `width:100%;z-index:10;position:fixed;top:0;transform:scale(${this.getScale()});display:flex;flex-direction:column;align-items:center;transform-origin:top;`
        }, parent);
        window.addEventListener('resize', this.update.bind(this));
    }

    cc(tag, options = {}, parent = false, init = false) {
        const children = options.children || [];
        delete options.children;
        const element = Object.assign(document.createElement(tag), options);
        for (const child of children) element.appendChild(child);
        if (init) init(element);
        return parent ? parent.appendChild(element) : element;
    }

    update() {
        this.element.style = `width:100%;z-index:2;position:fixed;top:0;transform:scale(${this.getScale()});display:flex;flex-direction:column;align-items:center;transform-origin:top;`;
    }

    getScale() {
        return (window.innerWidth - (window.innerWidth < 1920 ? 180 : 320)) / 1150;
    }

    remove() {
        window.removeEventListener('resize', this.update.bind(this));
        this.element.remove();
    }

    log(text, color='#FFFFFF', timer=this.stayTime, dtimer=this.disappearanceTime) {
        if (this.element.childElementCount == this.maxCount) {
            clearInterval(this.element.firstChild.__timeout);
            this.element.firstChild.remove();
        }
        this.cc('div', {
            style: `display:flex;flex-direction:column;border-radius:10px;color:${color};background-color:${color + '55'};border:solid;padding:10px 50px;margin-top:10px;transition:${dtimer}ms;`,
            children: [
                this.cc('div', {
                    style: `color:${color};font-family:'Black';font-size:20px;text-align:center;`,
                    textContent: text,
                })
            ]
        }, this.element, element => {
            element.__timeout = setTimeout(element.remove.bind(element), timer + dtimer);
            element.__dtimeout = setTimeout(() => {
                element.style.opacity = 0;
            }, timer);
        });
    }
}


class ImageWorker {
    constructor() {
        this.sx = 0;
        this.sy = 0;
        this.sWidth = 0;
        this.sHeight = 0;
        this.dx = 0;
        this.dy = 0;
        this.dWidth = 0;
        this.dHeight = 0;
        this.image = null;
        this.canvas = null;
        this.context2D = null;
    }

    setRect(sx=0, sy=0, sWidth=0, sHeight=0, dx=0, dy=0, dWidth=0, dHeight=0) {
        this.sx = sx;
        this.sy = sy;
        this.sWidth = sWidth;
        this.sHeight = sHeight;
        this.dx = dx;
        this.dy = dy;
        this.dWidth = dWidth;
        this.dHeight = dHeight;
    }

    setFitScale() {
        if (this.image.width / this.image.height > this.canvas.width / this.canvas.height) {
            this.setFitWidth();
        } else {
            this.setFitHeight();
        }
    }

    setFitHeight() {
        this.setRect();
        const ratio = this.canvas.height / this.image.height;
        this.sWidth = this.image.width;
        this.sHeight = this.image.height;
        this.dHeight = this.canvas.height;
        this.dWidth = Math.floor(this.image.width * ratio);
        this.dx = Math.floor(this.canvas.width / 2 - (this.image.width * ratio / 2));
    }

    setCover() {
        this.setRect();
        this.sWidth = this.image.width;
        this.sHeight = this.image.height;
        this.dWidth = this.canvas.width;
        this.dHeight = this.canvas.height;
    }

    setFitWidth() {
        this.setRect();
        const ratio = this.canvas.width / this.image.width;
        this.sWidth = this.image.width;
        this.sHeight = this.image.height;
        this.dHeight = Math.floor(this.image.height * ratio);
        this.dWidth = this.canvas.width;
        this.dy = Math.floor(this.canvas.height / 2 - (this.image.height * ratio / 2));
    }

    setImage(image) {
        this.image = image;
    }

    setCanvas(canvas) {
        this.context2D = (this.canvas = canvas).getContext('2d');
    }

    getCanvasData(x=0, y=0, width=this.canvas.width, height=this.canvas.height) {
        return this.context2D.getImageData(x, y, width, height).data;
    }

    getColor(flatdata, width, height, x, y) {
        let start = y * width * 4 + x * 4;
        return [flatdata[start], flatdata[start + 1], flatdata[start + 2], flatdata[start + 3]];
    }

    print() {
        this.context2D.clearRect(0, 0, this.canvas.width, this.canvas.height);
        this.context2D.drawImage(this.image, this.sx, this.sy, this.sWidth, this.sHeight, this.dx, this.dy, this.dWidth, this.dHeight);
    }
}


class CanvasWorker {
    constructor() {
        this.id = Array(16).fill().map(i => 'abcdefghijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz'[Math.floor(24 * Math.random())]).join('');
        this.sketch = document.createElement('canvas');
        this.sketch.context = this.sketch.getContext('2d');
        this.sketch.id = this.id;
        this.sketch.style = 'position:fixed;top:0;left:0;';
    }

    getColorMidDiffRGB([r1, g1, b1], [r2, g2, b2]) {
        return Math.abs((r1 + g1 + b1) / 3 - (r2 + g2 + b2) / 3);
    }

    getColorMidDiffRGBA([r1, g1, b1, a1], [r2, g2, b2, a2]) {
        return Math.abs((r1 + g1 + b1 + a1) / 4 - (r2 + g2 + b2 + a2) / 4);
    }

    getColorEachDiffRGB([r1, g1, b1], [r2, g2, b2]) {
        return Math.abs(r1 - r2) + Math.abs(g1 - g2) + Math.abs(b1 - b2);
    }

    getColorEachDiffRBGA([r1, g1, b1, a1], [r2, g2, b2, a2]) {
        return Math.abs(r1 - r2) + Math.abs(g1 - g2) + Math.abs(b1 - b2) + Math.abs(a1 - a2);
    }

    getColor(flatdata, width, height, x, y) {
        let start = y * width * 4 + x * 4;
        return [flatdata[start], flatdata[start + 1], flatdata[start + 2], flatdata[start + 3]];
    }

    findEdges(canvas, bound) {
        let data = canvas.getContext('2d').getImageData(0, 0, canvas.width, canvas.height).data;
        let cords = Array();
        for (let x=0; x < canvas.width - 1; x++) {
            for (let y=0; y < canvas.height - 1; y++) {
                if (this.getColorEachDiffRGB(
                    this.getColor(data, canvas.width, canvas.height, x, y),
                    this.getColor(data, canvas.width, canvas.height, x + 1, y + 1)
                ) > bound) {
                    cords.push([x, y]);
                }
            }
        }
        return cords;
    }

    out() {
        if (document.querySelector('#' + this.id)) return;
        document.documentElement.appendChild(this.sketch);
    }

}


class AutoDraw {
    constructor() {
        this.LOGS = new Log(document.documentElement, 3, 1000);
        this.constants = {
            maxRatio: 2,
            minRatio: 10
        };
        this.selectors = {
            drawingTable: '.draw',
            fillToolButton: '.tool.fil',
            buttom: '.bottom',
            header: '.book .header'
        };
        this.texts = {
            cantOpenItHereWarning: 'You can open menu only in game!',
            cantUseItInTHisGameMode: 'Autodrawing cannot be used in this game mode.',
            loadFile: 'LOAD IMAGE',
            inputOrInsert: 'drop, insert or CTRL + V',
            clearImg: 'CLEAR',
            print: 'PRINT',
            ratioDesc: 'less ratio less effective, but more quality',
            drewThisThisRound: "You can't draw more then one pic (or server will exploude!)",
            nothingTodraw: 'Nothing to draw.',
            cantPaseOnClosedMenu: 'Cannot paste on closed menu.',
            cannotUploadM20: 'You cannot upload image more than 20MB!',
            cannotPasteM20: 'You cannot paste image more than 20MB!',
            fileNotAttached: 'Error, no files attached.'
        };
        this.state = {
            drewThisRound: false,
            currentFile: null,
            canUpdateMenu: true,
            iCanvas: null,
            printRatio: 2,
            ws: null,
            turnNum: null
        };
    }

    init() {
        this.setTraps();
        this.setKeboardListener();
        this.setTriggerOnElement(this.selectors.drawingTable, this.onDrawingTable.bind(this));
        this.setTriggerOnElement(this.selectors.drawingTable, this.removeMenu.bind(this), 'removedNodes');
        this.addOnResize();
        this.addOnPaste();
    }

    addOnPaste() {
        window.addEventListener('paste', event => {
            let items = (event.clipboardData || event.originalEvent.clipboardData).items;
            for (let index in items) {
                let item = items[index];
                if (item.kind == 'file' && item.type.includes('image')) {
                    if (this.isMenuOpened()) {
                        this.imageDataInput(item.getAsFile(), 'paste');
                    } else {
                        this.LOGS.log(this.texts.cantPaseOnClosedMenu, '#FF8800');
                    }
                }
            }
        });
    }

    addOnResize() {
        window.addEventListener('resize', this.onWindowResize.bind(this));
    }

    onWindowResize() {
        this.updateMenu();
    }

    setTraps() {
        this.proxyWebSocket();
    }

    proxyWebSocket() {
        const t = this;
        window.WebSocket = new Proxy(WebSocket, {
            construct(target, args) {
                let ws = new target(...args);
                t.initWS(ws);
                return ws;
            }
        });
    }

    initWS(ws) {
        window._WS = this.state.ws = ws;
//         ws.send = new Proxy(ws.send, {
//             apply(target, thisArg, args) {
//                 console.log(args[0]);
//                 return Reflect.apply(...arguments);
//             }
//         });
        ws.addEventListener('message', this.onWebSocketMessage.bind(this));
    }

    onWebSocketMessage(event) {
        if (!event.data.includes('[')) return;
        const data = JSON.parse(event.data.replace(/^\d+/g, ''));
        switch (data[1]) {
            case 11: {
                this.state.drewThisRound = false;
                this.state.turnNum = data[2].turnNum;
                break;
            }
        }
    }

    getScale() {
        return (window.innerWidth - (window.innerWidth < 1920 ? 180 : 320)) / 1150;
    }

    setKeboardListener() {
        window.addEventListener('keydown', this.onKeyDown.bind(this));
    }

    onKeyDown(event) {
        this.keyDownSwitchTable(event);
    }

    keyDownSwitchTable(event) {
        switch (event.which || event.keyCode) {
            case 120: this.switchMenu(); break;
            case 27: this.removeMenu(); break;
        }
    }

    cc(tag, options = {}, parent = false, init = false) {
        const children = options.children || [];
        delete options.children;
        const element = Object.assign(document.createElement(tag), options);
        for (const child of children) element.appendChild(child);
        if (init) init(element);
        return parent ? parent.appendChild(element) : element;
    }

    onDrawingTable(element) {
        if (element.querySelector(this.selectors.fillToolButton)) this.generateDrawMenuButton(element);
    }

    generateDrawMenuButton(element) {
        const header = element.querySelector(this.selectors.header);
        this.cc('button', {
            textContent: String.fromCharCode(9998),
            style: 'position:absolute;color:white;appearance:none;outline:none;border:none;background-color:transparent;font-size:30px;left:50px;top:6px;cursor:pointer;',
            onclick: this.switchMenu.bind(this)
        }, header);
    }

    switchMenu() {
        const menuBG = document.body.querySelector('.my-bg');
        const drawTable = document.body.querySelector(this.selectors.drawingTable);
        const fillTool = document.body.querySelector(this.selectors.fillToolButton);
        if (menuBG) {
            menuBG.remove();
        } else {
            if (!drawTable) {
                this.LOGS.log(this.texts.cantOpenItHereWarning, '#FF6666');
            } else if (drawTable && !fillTool) {
                this.LOGS.log(this.texts.cantUseItInTHisGameMode, '#FF6666');
            } else {
                this.generateMenu();
            }
        }
    }

    removeMenu() {
        const menuBG = document.body.querySelector('.my-bg');
        if (menuBG) menuBG.remove();
    }

    isMenuOpened() {
        const menuBG = document.body.querySelector('.my-bg');
        const drawTable = document.body.querySelector(this.selectors.drawingTable);
        const fillTool = document.body.querySelector(this.selectors.fillToolButton);

        if (!drawTable) {
            this.removeMenu();
            return false;
        } else if (drawTable && !fillTool) {
            this.removeMenu();
            return false;
        }

        return Boolean(menuBG);
    }

    updateMenu() {
        if (this.isMenuOpened()) {
            const menuBG = document.body.querySelector('.my-bg');
            menuBG.remove();
            this.generateMenu();
        }
    }

    getBase64(file, callback) {
        const reader = new FileReader();
        reader.readAsDataURL(file);
        reader.onload = callback.bind(this, reader);
        reader.onerror = error => this.LOGS.log('Error in <AutoDraw.getBase64>: ' + error, '#000000');
    }

    loadImage(reader) {
        const image = new Image();
        image.src = reader.result;
        image.onload = this.onImageLoaded.bind(this);
    }

    async print() {
        if (!this.state.iCanvas) return this.LOGS.log(this.texts.nothingTodraw, '#FF5533');
        if (this.state.drewThisRound) return this.LOGS.log(this.texts.drewThisThisRound, '#FF5533');

        this.state.drewThisRound = true;

        const originalMap = {};
        const data = this.state.iCanvas.getCanvasData();
        const width = this.state.iCanvas.canvas.width;
        const height = this.state.iCanvas.canvas.height;
        const insensetiveAlpha = 20;
        const maxDifference = 50;

        function sleep(ms) {
            return new Promise(resolve => setTimeout(resolve, ms));
        };

        function componentToHex(c) {
            let hex = c.toString(16);
            return hex.length == 1 ? "0" + hex : hex;
        };

        function rgbaToHex(r, g, b, a) {
            return "#" + componentToHex(r) + componentToHex(g) + componentToHex(b) + componentToHex(a);
        };

        function hexToRgba(hex) {
            let bigint = parseInt(hex.split('#')[1], 16);
            return [(bigint >> 24) & 255, (bigint >> 16) & 255, (bigint >> 8) & 255, bigint & 255];
        }

        function getColorEachDiffRBGA([r1, g1, b1, a1], [r2, g2, b2, a2]) {
            return Math.abs(r1 - r2) + Math.abs(g1 - g2) + Math.abs(b1 - b2) + Math.abs(a1 - a2);
        };

        function getColorMidDiffRGBA([r1, g1, b1, a1], [r2, g2, b2, a2]) {
            return Math.abs((r1 + g1 + b1 + a1) / 4 - (r2 + g2 + b2 + a2) / 4);
        };

        function getMidColorFromHEX(hexArray) {
            if (hexArray.length > 1) {
                for (let i = 0; i < hexArray.length - 1; i++) {
                    let color1 = hexToRgba(hexArray[i]);
                    let color2 = hexToRgba(hexArray[i + 1]);
                    return rgbaToHex(
                        Math.floor((color1[0] + color2[0]) / 2),
                        Math.floor((color1[1] + color2[1]) / 2),
                        Math.floor((color1[2] + color2[2]) / 2),
                        Math.floor((color1[3] + color2[3]) / 2)
                    );
                }
            }
            return hexArray[0];
        }

        function getMidHex(hex1, hex2) {
            let color1 = hexToRgba(hex1);
            let color2 = hexToRgba(hex2);
            return rgbaToHex(
                Math.floor((color1[0] + color2[0]) / 2),
                Math.floor((color1[1] + color2[1]) / 2),
                Math.floor((color1[2] + color2[2]) / 2),
                Math.floor((color1[3] + color2[3]) / 2)
            );
        }

        for (let x = 0; x < width; x += this.state.printRatio) {
            for (let y = 0; y < height; y+=this.state.printRatio) {
                let start = y * width * 4 + x * 4;
                let color = [data[start], data[start + 1], data[start + 2], data[start + 3]];

                if (color[3] < insensetiveAlpha) continue;

                let hexColor = rgbaToHex(color[0], color[1], color[2], color[3]);
                let b = hexColor.split('');
                b[2] = '0';
                b[4] = '0';
                b[6] = '0';
                b[8] = '0';
                hexColor = b.join('');
                let place = `${x},${y},${this.state.printRatio},${this.state.printRatio}`;
                originalMap[hexColor] = originalMap[hexColor] ? originalMap[hexColor] + ',' + place : place;
            }
        }

        // extension
        const changedMap = {};
        const subMap = JSON.parse(JSON.stringify(originalMap));
        while (Object.keys(subMap).length !== 0) {
            const color = Object.keys(subMap)[0];
            const closeColors = [color];
            let closePath = subMap[color];
            delete subMap[color];
            for (let next in subMap) {
                let rgba1 = hexToRgba(color);
                let rgba2 = hexToRgba(next);
                if (getColorEachDiffRBGA(rgba1, rgba2) < maxDifference) {
                    closeColors.push(next);
                    closePath += "," + subMap[next];
                    delete subMap[next];
                }
            }
            changedMap[getMidColorFromHEX(closeColors)] = closePath;
        }
        // extension end

        let vId = 1;
        for (let color in changedMap) {
            let opacity = Math.round(Number("0x" + color.substr(7)) / 255 * 100) / 100;
            this.state.ws.send(`42[2,7,{"t":${this.state.turnNum},"d":1,"v":[8,${vId++},["${color.substr(0, 7)}",${opacity}],${changedMap[color]}]}]`);
            await sleep(150);
        }

        //         let vId = 1;
        //         for (let color in originalMap) {
        //             let opacity = Math.round(Number("0x" + color.substr(7)) / 255 * 100) / 100;
        //             this.state.ws.send(`42[2,7,{"t":${this.state.turnNum},"d":1,"v":[8,${vId++},["${color.substr(0, 7)}",${opacity}],${originalMap[color]}]}]`);
        //             await sleep(150);
        //         }

        this.removeMenu();
        setTimeout(() => this.executeReconnect(), 2e3);
    }

    executeReconnect() {
        this.state.ws.close();
    }

    onImageLoaded(event) {
        const image = event.target;
        if (!this.isMenuOpened()) return this.LOGS.log('Menu closed.', '#000000');
        this.state.iCanvas = new ImageWorker(0, 0, image.width, image.height, 0, 0, 500, 500);
        this.state.iCanvas.setImage(image);
        this.state.iCanvas.setCanvas(this.state.canvas);

        this.state.iCanvas.setFitScale();
        this.state.iCanvas.print();
        this.updateMenu();

        // this.state.cw = new CanvasWorker();

        // this.state.cw.sketch.width = this.state.canvas.width;
        // this.state.cw.sketch.height = this.state.canvas.height;

        // const a = this.state.cw.findEdges(this.state.canvas, 100);
        // this.state.cw.sketch.context.fillStyle = '#FFFFFF';
        // this.state.cw.sketch.context.fillRect(0, 0, this.state.cw.sketch.width, this.state.cw.sketch.height);
        // this.state.cw.sketch.context.fillStyle = '#000000';
        // for (let p of a) this.state.cw.sketch.context.fillRect(p[0], p[1], 1, 1);
        // this.state.cw.out();
    }

    async imageDataInput(data, type) {
        if (['fileinput-ondrop', 'fileinput-onchange'].includes(type)) {
            const file = data.target.files[0];
            if (file) {
                if (file.size > 20971520) {
                    this.LOGS.log(this.texts.cannotUploadM20, '#FF6666');
                } else {
                    this.state.currentFile = file;
                    this.getBase64(file, this.loadImage.bind(this));
                }
            } else {
                this.LOGS.log(this.texts.fileNotAttached, '#000000');
            }
        } else if (['paste'].includes(type)) {
            if (data) {
                if (data.size > 20971520) {
                    this.LOGS.log(this.texts.cannotPasteM20, '#FF6666');
                } else {
                    this.state.currentFile = data;
                    this.getBase64(data, this.loadImage.bind(this));
                }
            } else {
                this.LOGS.log(this.texts.fileNotAttached, '#000000');
            }
        }
    }


    generateMenu() {
        this.cc('div', {
            className: 'my-bg',
            style: 'width:100%;height:100%;background-color:rgba(0,0,0,0.8);position:absolute;left:0;top:0;z-index:2;-webkit-box-pack:center;justify-content:center;-webkit-box-align:center;align-items:center;display:flex;inset:0;',
            onclick: event => {
                if (event.currentTarget == event.target) this.removeMenu();
            },
            children: [
                this.cc('style', {
                    textContent: `.-dashed-line:hover {opacity:1 !important;}`
                    .concat(`#-reset-menu-preview:hover {background-color:white !important;border:4px solid black !important;color:black !important;}`)
                    .concat(`.control-button {flex:1;background-color:black;border-radius:7px;border:2px solid black;color:white;font-family:Black;font-size:20px;padding:0 10px;cursor:pointer;}`)
                    .concat(`.control-button:hover {background-color:white !important;border:2px solid black !important;color:black !important;}`)
                }),
                this.cc('div', {
                    style: `position:relative;display:flex;flex-direction:column;-webkit-box-align:center;align-items:center;background-color:rgb(255,255,255);padding:25px 30px;border-radius:12px;transform:scale(${this.getScale()});`,
                    children: [
                        this.cc('button', {
                            style: 'border:none;background:none;position:absolute;top:15px;right:15px;width:30px;height:30px;display:flex;-webkit-box-align:center;align-items:center;-webkit-box-pack:center;justify-content:center;cursor:pointer;',
                            children: [
                                this.cc('div', {
                                    style: 'font-family:ico;color:rgb(172,167,198);font-size:25px;',
                                    textContent: String.fromCharCode(59654),
                                })
                            ],
                            onclick: this.removeMenu.bind(this)
                        }),
                        this.cc('div', {
                            style: 'display:flex;flex-direction:column;align-items:center;margin-bottom:20px;',
                            children: [
                                this.cc('div', {
                                    style: 'color:black;font-family:Black;font-size:40px;',
                                    textContent: this.texts.loadFile
                                })
                            ]
                        }),
                        this.cc('div', {
                            style: 'display:flex;flex-direction:row;',
                            className: '-flex-settings',
                            children: [
                                this.cc('div', {
                                    className: '-screen-pad-with-bottom',
                                    style: 'display:flex;flex-direction:column',
                                    children: [
                                        this.cc('div', {
                                            className: '.m-screen',
                                            style: 'width:758px;height:424px;',
                                            children: [
                                                this.cc('div', {
                                                    style: `width:758px;height:424px;position:absolute;display:flex;flex-direction:column;align-items:center;justify-content:center;opacity:0.3;`.concat(this.state.iCanvas ? 'display:none;' : ''),
                                                    children: [
                                                        this.cc('div', {
                                                            style: 'display:flex;flex-direction:column;align-items:center;',
                                                            children: [
                                                                this.cc('div', {
                                                                    style: 'algin-text:center;font-family:Black;font-size:50px;',
                                                                    textContent: this.state.currentFile ? this.state.currentFile.name : this.texts.inputOrInsert
                                                                })
                                                            ]
                                                        })
                                                    ]
                                                }),
                                                this.cc('div', {
                                                    className: '-dashed-line',
                                                    style: `position:absolute;border:4px dashed black;border-radius:10px;opacity:0.3;`.concat(this.state.iCanvas ? 'display:none;' : ''),
                                                    children: [
                                                        this.cc('input', {
                                                            type: "file",
                                                            accept: "image/*",
                                                            style: `width:758px;height:424px;cursor:pointer;position:relative;opacity:0;`,
                                                            ondragenter: event => {
                                                                event.preventDefault();
                                                            },
                                                            ondrop: event => {
                                                                event.preventDefault();
                                                                this.imageDataInput(event, 'fileinput-ondrop');
                                                            },
                                                            onchange: event => {
                                                                this.imageDataInput(event, 'fileinput-onchange');
                                                            }
                                                        })
                                                    ]
                                                }),
                                                this.cc('div', {
                                                    style: 'position:absolute;background-repeat:repeat;background-size:22px;image-rendering:pixelated;background-image:url();width:758px;height:424px;border:2px solid;'.concat(this.state.iCanvas ? '' : 'display:none;'),
                                                    children: [
                                                        this.state.canvas || (this.state.canvas = this.cc('canvas', {
                                                            style: 'width:758px;height:424px;position:absolute;',
                                                            width: 758,
                                                            height: 424
                                                        }))
                                                    ]
                                                }),
                                            ]
                                        }),
                                        this.cc('div', {
                                            style: 'display:flex;flex-direction:row;width:100%;margin-top:25px;gap:10px;',
                                            children: [
                                                this.cc('button', {
                                                    id: '-reset-menu-preview',
                                                    style: 'background-color:black;border-radius:7px;border:4px solid black;color:white;font-family:Black;font-size:28px;padding:0 30px;cursor:pointer;height:40px;',
                                                    textContent: this.texts.print,
                                                    onclick: () => {
                                                        this.print();
                                                    }
                                                }),
                                                this.cc('div', {
                                                    style: 'flex:1;display:flex;flex-direction:column;',
                                                    children: [
                                                        this.generateRange('ratio', this.constants.maxRatio, this.constants.minRatio, this.state.printRatio, 1, event => {
                                                            this.state.printRatio = Number(event.target.value);
                                                        }),
                                                        this.cc('div', {
                                                            style: 'font-family:Black;font-size:15px;',
                                                            textContent: this.texts.ratioDesc,
                                                        })
                                                    ]
                                                }),
                                                this.cc('button', {
                                                    id: '-reset-menu-preview',
                                                    style: 'background-color:black;border-radius:7px;border:4px solid black;color:white;font-family:Black;font-size:28px;padding:0 30px;cursor:pointer;height:40px;',
                                                    textContent: this.texts.clearImg,
                                                    onclick: () => {
                                                        this.state.canvas = null;
                                                        this.state.currentFile = null;
                                                        this.state.iCanvas = null;
                                                        this.updateMenu();
                                                    }
                                                })
                                            ]
                                        })
                                    ]
                                })
                            ].concat(this.state.currentFile ? [
                                this.cc('div', {
                                    className: '-settings-plane',
                                    style: 'margin-left:30px;max-width:200px;width:200px;',
                                    children: [
                                        this.generateRange('dx', -this.state.iCanvas.canvas.width, this.state.iCanvas.canvas.width, this.state.iCanvas.dx, 1, event => {
                                            this.state.iCanvas.dx = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.generateRange('dy', -this.state.iCanvas.canvas.height, this.state.iCanvas.canvas.height, this.state.iCanvas.dy, 1, event => {
                                            this.state.iCanvas.dy = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.generateRange('dw', 10, this.state.iCanvas.canvas.width * 2, this.state.iCanvas.dWidth, 1, event => {
                                            this.state.iCanvas.dWidth = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.generateRange('dh', 10, this.state.iCanvas.canvas.height * 2, this.state.iCanvas.dHeight, 1, event => {
                                            this.state.iCanvas.dHeight = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.generateRange('sx', 0, this.state.iCanvas.image.width, this.state.iCanvas.sx, 1, event => {
                                            this.state.iCanvas.sx = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.generateRange('sy', 0, this.state.iCanvas.image.height, this.state.iCanvas.sy, 1, event => {
                                            this.state.iCanvas.sy = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.generateRange('sw', 10, this.state.iCanvas.image.height, this.state.iCanvas.sWidth, 1, event => {
                                            this.state.iCanvas.sWidth = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.generateRange('sh', 10, this.state.iCanvas.image.height, this.state.iCanvas.sHeight, 1, event => {
                                            this.state.iCanvas.sHeight = Number(event.target.value);
                                            this.state.iCanvas.print();
                                        }),
                                        this.cc('div', {
                                            style: 'display:flex;flex-direction:column;gap:3px;',
                                            children: [
                                                this.cc('button', {
                                                    className: 'control-button',
                                                    textContent: 'cover',
                                                    onclick: () => {
                                                        this.state.iCanvas.setCover();
                                                        this.state.iCanvas.print();
                                                        this.updateMenu();
                                                    }
                                                }),
                                                this.cc('button', {
                                                    className: 'control-button',
                                                    textContent: 'fit height',
                                                    onclick: () => {
                                                        this.state.iCanvas.setFitHeight();
                                                        this.state.iCanvas.print();
                                                        this.updateMenu();
                                                    }
                                                }),
                                                this.cc('button', {
                                                    className: 'control-button',
                                                    textContent: 'fit width',
                                                    onclick: () => {
                                                        this.state.iCanvas.setFitWidth();
                                                        this.state.iCanvas.print();
                                                        this.updateMenu();
                                                    }
                                                }),
                                                this.cc('button', {
                                                    className: 'control-button',
                                                    textContent: 'smart fit',
                                                    onclick: () => {
                                                        this.state.iCanvas.setFitScale();
                                                        this.state.iCanvas.print();
                                                        this.updateMenu();
                                                    }
                                                })
                                            ]
                                        })
                                    ]
                                })
                            ] : [])
                        })
                    ]
                })
            ]
        }, document.body);
    }

    generateRange(name, from, to, cur, step, inputCallback) {
        return this.cc('div', {
            style: 'display:flex;flex-direction:row;max-width:200px;gap:5px;width:200px;height:42px;',
            children: [
                this.cc('div', {
                    style: 'display:flex;align-items:center;justify-content:center;',
                    children: [
                        this.cc('div', {
                            style: 'color:black;font-family:Black;font-size:20px;',
                            textContent: name
                        })
                    ]
                }),
                this.cc('div', {
                    style: 'max-width:140px;min-width:140px;',
                    children: [
                        this.cc('div', {
                            style: 'display:flex;flex-direction:row;',
                            children: [
                                this.cc('div', {
                                    style: 'color:black;font-family:Black;font-size:14px;',
                                    textContent: from
                                }),
                                this.cc('div', {
                                    style: 'flex:1;display:flex;flex-direction:column;align-items:center;',
                                    children: [
                                        this.cc('div', {
                                            style: 'color:black;font-family:Black;font-size:14px;',
                                            textContent: Math.floor(to - (to - from) / 2)
                                        })
                                    ]
                                }),
                                this.cc('div', {
                                    style: 'color:black;font-family:Black;font-size:14px;',
                                    textContent: to
                                })
                            ]
                        }),
                        this.cc('input', {
                            style: 'width:100%;',
                            type: 'range',
                            min: from,
                            max: to,
                            step: step,
                            oninput: event => {
                                event.target.parentNode.parentNode.querySelector('#dispval').textContent = event.target.value;
                                inputCallback.call(this, event);
                            }
                        }, false, element => (element.value=cur)),
                    ]
                }),
                this.cc('div', {
                    style: 'display:flex;align-items:center;justify-content:center;',
                    children: [
                        this.cc('div', {
                            style: 'color:black;font-family:Black;font-size:20px;',
                            id: 'dispval',
                            textContent: cur
                        })
                    ]
                })
            ]
        });
    }

    setTriggerOnElement(selector, callback, action='addedNodes', once=false, searchIn=document) {
        const observer = new MutationObserver(mutations => {
            for (const mutation of mutations) {
                const nodes = mutation[action] || [];
                for (const node of nodes) {
                    const element = node.matches && node.matches(selector) ? node : (node.querySelector ? node.querySelector(selector) : null);
                    if (element) {
                        if (once) {
                            observer.disconnect();
                            return callback(element);
                        } else {
                            callback(element);
                        }
                    }
                }
            }
        });

        observer.observe(searchIn, {
            attributes: false,
            childList: true,
            subtree: true
        });

        return observer;
    }

}

const AD = new AutoDraw();
AD.init();