Advanced Agma

clickable chat links, discord in game chat (/discord ... hi), remove specific animations, remove food completely (toggle mouse), advanced user stats (agma.io/stats.php), increase number of stackable animations

질문, 리뷰하거나, 이 스크립트를 신고하세요.
// ==UserScript==
// @name         Advanced Agma
// @namespace    http://tampermonkey.net/
// @version      0.1.1
// @author       Big watermelon (credits: Nersai, Vintrex)
// @description  clickable chat links, discord in game chat (/discord ... hi), remove specific animations, remove food completely (toggle mouse), advanced user stats (agma.io/stats.php), increase number of stackable animations
// @match        https://agma.io/*
// @match        https://discord.com/*
// @require      https://cdn.jsdelivr.net/npm/pako@2.0.4/dist/pako.min.js
// @license      GPL-3.0-or-later
// @icon         
// @grant        unsafeWindow
// @grant        GM_setValue
// @grant        GM_getValue
// @run-at       document-start
// ==/UserScript==

/*  WARNING:
This script defines properties on certain objects
so if other scripts define the same properties some stuff may break

Overwritten Properties
 | localChatMessage.cache
 | localChatMessage.cache.v
 | localChatMessage.cache.ie
 | localChatMessage.cache.color2
 | localChatMessage.cache.ctx
 | localChatMessage.cache.ctx.fillText
*/
/*
FIXIT: token isnt always grabbed
FIXIT: GM_ functions don't seem to work always so maybe use localStorage
       but that's kinda risky so maybe hash it with some values inside the script
TODO: message edit => edit message
TODO: message too long to fit in chat => wrap arround (well maybe not worth it)
*/

(function() {
    "use strict";

    if (unsafeWindow.top !== unsafeWindow.self || document.querySelector("title")?.textContent?.includes("Just a moment")) {
        return;
    }

    const settings = GM_getValue("settings", {
        removeFood: true,
        removeAnimations: [3, 6, 7, 8, 11],
        maxStackableAnimations: 3,
        discordChat: false,
        discordSavedChannels: [],
        discordPresence: false
    });

    const discordToken = GM_getValue("discordToken", null);
    if (settings.discordChat && unsafeWindow.location.href.startsWith("https://discord.com")) {
        if (!discordToken) {
            GM_setValue("discordToken", JSON.parse(unsafeWindow.localStorage.token));
            unsafeWindow.alert("Discord Chat for agma.io has updated your token !");
        }
        return;
    }

    const numberFormat = Intl.NumberFormat("fr-FR");
    const userprofiles = unsafeWindow.localStorage.userprofiles ? JSON.parse(unsafeWindow.localStorage.userprofiles) : {};
    if (!discordToken) {
        unsafeWindow.alert("Discord Chat can not work since you didnt log in into your browser with discord.\nIf you want to use the discord chat feature wait for an alert saying that your token has been saved.\n\nhttps://discord.com");
        settings.discordChat = false;
    }
    const discordPresence = {
        status: "online",
        since: 0,
        activities: [{
            name: "Agma.io",
            type: 0,
            url: "https://agma.io",
            details: "Playing Agma.io",
            timestamps: {
                start: Date.now()
            }
        }],
        afk: false
    }

    const animations = {
        1:  { name: "Recombine", style: "border-color: #337ab7; background-image: url('./img/store/recombine-min.png'); padding-left: 30px; padding-left: 30px;" },
        2:  { name: "Cell Select", style: "border-color: #fe3f3f;" },
        3:  { name: "Spin", style: "border-color: #b3b3b3;" },
        4:  { name: "360 Shot", style: "border-color: #337ab7; background-image: url('./img/push_lo.png'); padding-left: 30px;" },
        5:  { name: "Level Up", style: "border-color: #fe3f3f;" },
        6:  { name: "Flip Spin", style: "border-color: #b3b3b3;" },
        7:  { name: "Flip", style: "border-color: #b3b3b3;" },
        8:  { name: "Shake", style: "border-color: #b3b3b3;" },
        9:  { name: "Explosion", style: "border-color: #f0ad4e; background-image: url('./emotes/1f4a5.png'); padding-left: 30px;" },
        10: { name: "1st Medal", style: "border-color: #f0ad4e;" },
        11: { name: "Jump", style: "border-color: #b3b3b3;" },
        12: { name: "Wacky", style: "border-color: #f0ad4e; background-image: url('./emotes/1f61c.png'); padding-left: 30px;" },
        13: { name: "White cell for 1 frame", style: "border-color: #fe3f3f;" },
        14: { name: "Freeze", style: "border-color: #337ab7; background-image: url('./img/inv_freeze2.png'); padding-left: 30px;" },
        15: { name: "Speed", style: "border-color: #337ab7; background-image: url('./img/store/speed-min.png'); padding-left: 30px;" },
        16: { name: "Idk", style: "border-color: #fe3f3f;" }, // weird nothing
        17: { name: "Upgrade", style: "border-color: #fe3f3f;" },
        18: { name: "Snowball", style: "border-color: #fe3f3f;" },
        20: { name: "Anti freeze", style: "border-color: #337ab7; background-image: url('./skins/objects/20.png'); padding-left: 30px;" },
        21: { name: "Anti recombine", style: "border-color: #337ab7; background-image: url('./skins/objects/21.png'); padding-left: 30px;" },
        23: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(82, 152, 203, 0.6); padding-left: 30px;" },
        24: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(84, 211, 77, 0.6); padding-left: 30px;" },
        25: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(243, 46, 46, 0.6); padding-left: 30px;" },
        26: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(127, 59, 227, 0.6); padding-left: 30px;" },
        30: { name: "Wave", style: "border-color: #f0ad4e; background-image: url('./emotes/1f44b.png'); padding-left: 30px;" },
        31: { name: "Head Explosion", style: "border-color: #f0ad4e; background-image: url('./emotes/1f61cd.png'); padding-left: 30px;" },
        32: { name: "Hearts Face", style: "border-color: #f0ad4e; background-image: url('./emotes/1f60d.png'); padding-left: 30px;" },
        41: { name: "Angry Pumpkin", style: "border-color: #f0ad4e; background-image: url('./emotes/angry_emote3.png'); padding-left: 30px;" },
        42: { name: "Scared Pumpkin", style: "border-color: #f0ad4e; background-image: url('./emotes/scared_emote.png'); padding-left: 30px;" },
        43: { name: "Yawn Pumpkin", style: "border-color: #f0ad4e; background-image: url('./emotes/yawn_emote.png'); padding-left: 30px;" },
        44: { name: "Throwup", style: "border-color: #f0ad4e; background-image: url('./emotes/throwup.png'); padding-left: 30px;" },
        45: { name: "Hot face", style: "border-color: #f0ad4e; background-image: url('./emotes/hotface.png'); padding-left: 30px;" },
        46: { name: "Tears Joy", style: "border-color: #f0ad4e; background-image: url('./emotes/tearsjoy.png'); padding-left: 30px;" },
        47: { name: "No No", style: "border-color: #f0ad4e; background-image: url('./emotes/nonu.png'); padding-left: 30px;" },
        48: { name: "Clap", style: "border-color: #f0ad4e;" },
        49: { name: "Crying", style: "border-color: #f0ad4e;" },
        50: { name: "Devil Smile", style: "border-color: #f0ad4e;" },
        51: { name: "Eatman", style: "border-color: #f0ad4e;" },
        52: { name: "Trophy", style: "border-color: #f0ad4e;" },
        53: { name: "Hearts", style: "border-color: #f0ad4e;" }
    };
    const animationsIds = [
        1, 4, 14, 15, 20, 21, 23, 24, 25, 26, // Powers
        3, 6, 7, 8, 11, // Annoying
        9, 10, 12, 30, 31, 32, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, // Emotes
        2, 5, 13, 16, 17, 18 // System
    ];

    const discordIcon = new Image();
    discordIcon.src = "";
    const urlRegex = /https?:\/\/[^\s]+/g;
    const invisibleChar = "᠋";
    const wordsToReplace = ["https", "www", ".gg", ".com", ".io", ".net", ".biz", "miracle", "palestine"].map(word => [
        new RegExp(`\\b(${word})\\b`, "gi"),
        word[0] + invisibleChar + word.slice(1)
    ]);

    var discordUsername;
    var discordUserContext = null;

    var discordWebSocket;
    var discordHeartbeatInterval;
    var discordReconnectRetryCount = 0;
    const discordGatewayUrl = "wss://gateway.discord.gg/?v=9&encoding=json";
    const sentMessagesIds = [];
    const originalSend = WebSocket.prototype.send;
    const sendWebsocketDiscordMessage = (op, d) => originalSend.call(discordWebSocket, JSON.stringify({ op, d }));

    var chtTabs;
    var currentTab = '';
    var localChatMessages;
    var cellProtoOverwritten = false;

    function initDiscordWebSocket() {
        discordWebSocket = new WebSocket(discordGatewayUrl);
        discordWebSocket.onopen = () => console.debug("[🔵] Connected");
        discordWebSocket.onmessage = ({ data }) => {
            if (data instanceof Blob) {
                const reader = new FileReader();
                reader.onload = () => {
                    handleMessage(JSON.parse(pako.inflate(new Uint8Array(reader.result), { to: 'string' })));
                };
                reader.readAsArrayBuffer(data);
            } else {
                handleMessage(JSON.parse(data));
            }
        };
        discordWebSocket.onclose = () => {
            console.debug("[🔵] Disconnected");
            clearInterval(discordHeartbeatInterval);
            if (++discordReconnectRetryCount < 5) {
                initDiscordWebSocket();
            }
        };
    }
    function createDiscordChatMessage(message) {
        if (!message.channel_id || ![0, 19, 1, 2, 3, 6].includes(message.type) || sentMessagesIds.includes(message.id)) {
            return;
        }
        const channel = settings.discordSavedChannels.find(channel => channel.id === message.channel_id);
        if (!channel) {
            return;
        }
        sentMessagesIds.push(message.id);
        console.debug("[🔵] New message:", message);
        const tab = chtTabs.querySelector(`div[data-username="discord-${channel.id}"]`);
        if (!tab) {
            newDiscordTab(channel.name, channel.id);
        } else if (!tab.classList.contains('selected')) {
            tab.classList.add('blink');
            setTimeout(() => tab.classList.remove('blink'), 2000);
        }
        const author = message.author.global_name || message.author.username;
        switch (message.type) {
            case 0:  // DEFAULT
            case 19: // REPLY
                var content = message.content || '';
                message.mentions.forEach(mention => content = content.replaceAll(
                    `<@${mention.id}>`,
                    mention.global_name || mention.username
                ));
                if (message.attachments.length) {
                    content += ` +${message.attachments.length} files`;
                }
                localChatMessages.push({
                    O: true,
                    get v() {
                        discordUserContext = this.discordAuthor;
                        return -1;
                    },
                    o: 0,
                    P: 0,
                    U: 0,
                    name: author,
                    G: "discord-" + channel.id,
                    color: discordUsername === message.author.username ? "#313338" : "#5662E9",
                    message: content,
                    category: 2,
                    goldMember: 0,
                    L: 0,
                    Y: 0,
                    q: 0,
                    j: 0,
                    J: 0,
                    time: Date.now(),
                    _cache: null,
                    get cache() {
                        return this._cache;
                    },
                    set cache(value) {
                        value.icons = [ 14 ];
                        return this._cache = value;
                    },
                    discordAuthor: message.author
                });
                break;
            case 1: // RECIPIENT_ADD
            case 2: // RECIPIENT_REMOVE
                const member = message.mentions[0];
                localChatMessages.push({
                    O: false,
                    v: 0,
                    o: 0,
                    P: 0,
                    U: 0,
                    name: '',
                    G: "discord-" + channel.id,
                    color: message.type === 1 ? "#00FF00" : "#FF0000",
                    message: `${message.type === 1 ? "➡️" : "⬅️"} ${member.global_name || member.username} ${message.type === 1 ? "joined" : "left"} the group`,
                    category: 2,
                    goldMember: 0,
                    L: 0,
                    Y: 0,
                    q: 0,
                    j: 0,
                    J: 0,
                    time: Date.now(),
                    _cache: null,
                    get cache() {
                        return this._cache;
                    },
                    set cache(value) {
                        value.icons = [ 14 ];
                        Object.defineProperties(value, {
                            ie: { get: () => this.message },
                            color2: { get: () => this.color }
                        });
                        return this._cache = value;
                    }
                });
                break;
            case 3: // CALL
                localChatMessages.push({
                    O: false,
                    v: 0,
                    o: 0,
                    P: 0,
                    U: 0,
                    name: '',
                    G: "discord-" + channel.id,
                    color: "#00FF00",
                    message: `${author} started a call`,
                    category: 2,
                    goldMember: 0,
                    L: 0,
                    Y: 0,
                    q: 0,
                    j: 0,
                    J: 0,
                    time: Date.now(),
                    _cache: null,
                    get cache() {
                        return this._cache;
                    },
                    set cache(value) {
                        value.icons = [ 14 ];
                        Object.defineProperties(value, {
                            ie: { get: () => this.message },
                            color2: { get: () => this.color }
                        });
                        return this._cache = value;
                    }
                });
                break;
            case 6: // CHANNEL_PINNED_MESSAGE
                // message_reference: { message_id: "1259902171630145651", channel_id: "853556382417420298" }
                localChatMessages.push({
                    O: false,
                    v: 0,
                    o: 0,
                    P: 0,
                    U: 0,
                    name: '',
                    G: "discord-" + channel.id,
                    color: "#00FF00",
                    message: `${author} pinned a message`,
                    category: 2,
                    goldMember: 0,
                    L: 0,
                    Y: 0,
                    q: 0,
                    j: 0,
                    J: 0,
                    time: Date.now(),
                    _cache: null,
                    get cache() {
                        return this._cache;
                    },
                    set cache(value) {
                        value.icons = [ 14 ];
                        Object.defineProperties(value, {
                            ie: { get: () => this.message },
                            color2: { get: () => this.color }
                        });
                        return this._cache = value;
                    }
                });
                break;
        }
    }
    function handleMessage(message) {
        const { op, t, d } = message;
        switch (op) {
            case 10: // Hello
                console.debug("[🔵] Connection accepted");
                discordHeartbeatInterval = setInterval(() => sendWebsocketDiscordMessage(1, null), d.heartbeat_interval);
                sendWebsocketDiscordMessage(2, {
                    token: discordToken,
                    capabilities: 30717,
                    properties: {}
                });
                settings.discordPresence && setTimeout(() => sendWebsocketDiscordMessage(3, discordPresence), 2000);
                break;
            case 0: // Dispatch
                switch (t) {
                    case "MESSAGE_CREATE":
                        createDiscordChatMessage(d);
                        break;
                    case "READY":
                        discordUsername = d.user?.username || discordUsername;
                        console.debug("[🔵] Logged as", discordUsername);
                        break;
                    // default:
                    //     console.debug("[🔵] Unhandled message type:", t);
                }
                break;
            case 1: // HEARTBEAT
                sendWebsocketDiscordMessage(1, null);
                break;
            case 11: // HEARTBEAT_ACK
                break;
            default:
                console.debug("[🔵] Unhandled opcode type:", op);
        }
    }
    function findUrlToOpen(event) {
        const x = event.clientX - event.target.offsetLeft,
              y = event.clientY - event.target.offsetTop;
        for (const message of localChatMessages) {
            if (message.cache && message.cache.urls) {
                for (const { url, rect } of message.cache.urls) {
                    if (x >= rect.x0 && y >= rect.y0 && x <= rect.x1 && y <= rect.y1) {
                        unsafeWindow.open(url, '_blank');
                    }
                }
            }
        }
    }
    function timeFormat(s) {
        const h = Math.floor(s / 3600);
        const m = Math.floor(s % 3600 / 60);
        return (h ? h + 'h ' : '') + (m ? m + 'm ' : '') + Math.floor(s % 3600 % 60) + 's';
    }
    function getTable(payload) {
        if (payload === null) {
            return "<table style=\"width: 100%; text-align: left;\"><tr><td>Player not registered</td><tr><td>You need to play at least 1 game of batle royale to be registered on <a href=\"https://agma.io/stats.php\" target=\"_blank\">Agma Stats</a></td></tr></table>";
        }
        return (
            "<table style=\"width: 100%; text-align: left;\"><tr><td>Players Consumed: </td><td>"
            + numberFormat.format(payload.players_consumed)
            + "</td></tr><tr><td>Death Count: </td><td>"
            + numberFormat.format(payload.death_count)
            + "</td></tr><tr><td>K/D: </td><td>"
            + (payload.players_consumed / payload.death_count).toFixed(2)
            + "</td></tr><tr><td>Splits count: </td><td>"
            + numberFormat.format(payload.splits_count)
            + "</td></tr><tr><td>Total time alive: </td><td>"
            + timeFormat(payload.total_time_alive)
            + "</td></tr><tr><td>Splits per seconds: </td><td>"
            + (payload.splits_count / payload.total_time_alive).toFixed(2)
            + "</td></tr></table>"
        );
    }

    const originalDrawImage = CanvasRenderingContext2D.prototype.drawImage;
    unsafeWindow.CanvasRenderingContext2D.prototype.drawImage = function() {
        if (arguments[0] instanceof Image && arguments[0].src.startsWith("https://agma.io/img/chaticons") && arguments[1] === 280) {
            arguments[0] = discordIcon;
            arguments[1] = 0;
            arguments[6] += 3;
        }
        return originalDrawImage.apply(this, arguments);
    }
    const originalReplace = unsafeWindow.String.prototype.replace;
    unsafeWindow.String.prototype.replace = function(pattern, replacement) {
        return originalReplace.call(this, pattern, replacement instanceof Function && pattern instanceof RegExp && pattern.source.startsWith(":rolling") ? (match, offset) => match === ":/" && this?.[offset + 1] === '/' ? match : replacement(match, offset) : replacement);
    }
    const originalPush = unsafeWindow.Array.prototype.push;
    unsafeWindow.Array.prototype.push = function(elem) {
        if (elem?.time !== undefined && elem?.cache === null) {
            if (localChatMessages === undefined) {
                unsafeWindow.localChatMessages = localChatMessages = this;
            }
            if ((elem.message = elem.message.replaceAll(invisibleChar, "")).includes("https://")) {
                Object.defineProperty(elem, "cache", {
                    get: function() {
                        return this._cache;
                    },
                    set: function(cache) {
                        if (elem._cache !== undefined) {
                            cache.icons = [ 14 ];
                            // see if this breaks calls and stuff but shouldnt
                        }
                        Object.defineProperty(cache, "ctx", {
                            get: function() {
                                return this._ctx;
                            },
                            set: function(ctx) {
                                ctx.fillText = function(text, x, y) {
                                    if (this.fillStyle !== "#f5f6ce" && this.fillStyle !== "#444444") {
                                        return CanvasRenderingContext2D.prototype.fillText.call(this, text, x, y);
                                    }
                                    // ngl I asked chatgpt cuz Im lazy (and I edited some stuff)
                                    let match;
                                    let lastIndex = 0;
                                    let currentX = x;
                                    cache.urls = [];
                                    while ((match = urlRegex.exec(text)) !== null) {
                                        const url = match[0];
                                        const urlStart = match.index;
                                        const urlEnd = urlStart + url.length;

                                        const preText = text.slice(lastIndex, urlStart);
                                        if (preText) {
                                            CanvasRenderingContext2D.prototype.fillText.call(this, preText, currentX, y);
                                            currentX += ctx.measureText(preText).width;
                                        }
                                        ctx.save();
                                        ctx.fillStyle = '#1E90FF';
                                        CanvasRenderingContext2D.prototype.fillText.call(this, url, currentX, y);

                                        const urlWidth = ctx.measureText(url).width;
                                        const textSize = parseInt(ctx.font.match(/\d+/), 10);
                                        const underlineY = y + 2;

                                        ctx.beginPath();
                                        ctx.moveTo(currentX, underlineY);
                                        ctx.lineTo(currentX + urlWidth, underlineY);
                                        ctx.strokeStyle = '#1E90FF';
                                        ctx.lineWidth = 1;
                                        ctx.stroke();
                                        ctx.restore();
                                        cache.urls.push({
                                            url,
                                            rect: {
                                                x0: currentX,
                                                yd0: y - textSize,
                                                x1: currentX + urlWidth,
                                                yd1: y
                                            }
                                        });
                                        currentX += urlWidth;
                                        lastIndex = urlEnd;
                                    }
                                    const remainingText = text.slice(lastIndex);
                                    if (remainingText) {
                                        CanvasRenderingContext2D.prototype.fillText.call(this, remainingText, currentX, y);
                                    }
                                }
                                return this._ctx = ctx;
                            }
                        });
                        return this._cache = cache;
                    }
                });
            }
        } else if (elem?.ch !== undefined && elem?.x0 !== undefined) {
            elem.ch.cache?.urls?.forEach(urlObject => {
                urlObject.rect.y0 = urlObject.rect.yd0 + elem.y0;
                urlObject.rect.y1 = urlObject.rect.yd1 + elem.y0;
            });
        } else if (elem?.namePart !== undefined && elem?.clanPart !== undefined) {
            if (!cellProtoOverwritten) {
                if (settings.removeAnimations.length) {
                    elem.constructor.prototype.ge = function(animation) {
                        if (1 == this.a || settings.removeAnimations.includes(animation.H)) {
                            return;
                        }
                        animation = { H: animation.H, K: animation.K, received: animation.received };
                        if (this.Ne) {
                            for (var i = 0; i < this.Ne.length; i++) {
                                if (this.Ne[i].received > animation.received) {
                                    this.Ne.splice(i, 0, animation);
                                    if (this.Ne.length > settings.maxStackableAnimations) {
                                        this.Ne.splice(this.Ne.length - 2, 1);
                                    }
                                    return;
                                }
                            }
                            if (this.Ne.length < settings.maxStackableAnimations) {
                                originalPush.call(this.Ne, animation);
                            } else {
                                this.Ne[this.Ne.length - 1] = animation;
                            }
                        } else {
                            this.Ne = [animation];
                        }
                    }
                }
                cellProtoOverwritten = true;
            }
            if (settings.removeFood && elem.Pe) { // elem.a === 1
                return 0;
            }
        }
        return originalPush.apply(this, arguments);
    }

    unsafeWindow.setAnimation = (id, value) => settings.removeAnimations[settings.removeAnimations.includes(id) ? 'remove' : 'push'](id);
    unsafeWindow.setMaxStackableAnimations = value => settings.maxStackableAnimations = parseInt(value);
    unsafeWindow.setRemoveFood = value => settings.removeFood = value;
    unsafeWindow.setDiscordChat = value => {
        if (settings.discordChat = value) {
            !discordWebSocket && initDiscordWebSocket();
        } else if (discordWebSocket !== null) {
            discordWebSocket.close();
            discordWebSocket = null;
        }
    };
    unsafeWindow.setDiscordPresenceChat = value => settings.discordPresence = value;
    unsafeWindow.newDiscordTab = (name, channel) => chtTabs && (chtTabs.innerHTML += `<div data-category="2" data-username="discord-${channel}" data-insert data-tooltip="Discord chat: ${name}" class="chat-tab semi-selected">${name}</div>`);
    /*
    // could make your own messages show up faster but too annoying so no
    createDiscordChatMessage({
        id: null,
        channel_id: channel,
        type: 0,
        author: discordUser,
        content
    });
    */
    unsafeWindow.sendDiscordMessage = (channel, content) => settings.discordChat && discordUsername && fetch(`https://discord.com/api/v9/channels/${channel}/messages`, {
        method: 'POST',
        headers: {
            'Authorization': discordToken,
            'Content-Type': 'application/json'
        },
        body: JSON.stringify({
            content,
            tts: false
        })
    })
        .then(async response => createDiscordChatMessage(await response.json()))
        .catch(error => console.error('[🔵] Error while trying to send a message:', error));

    var loaded = false;
    unsafeWindow.addEventListener("load", () => {
        if (loaded || typeof swal === "undefined") return;
        loaded = true;
        const chtCanvas = document.getElementById("chtCanvas");
        chtCanvas.addEventListener("dblclick", findUrlToOpen);
        chtCanvas.addEventListener("click", event => event.ctrlKey && findUrlToOpen(event));
        document.getElementById("chtbox").addEventListener("keydown", function(event) {
            if (event.keyCode === 13) {
                if (!currentTab && this.value.startsWith("/discord ")) {
                    const args = this.value.slice(9).split(' ');
                    const channelName = args.shift();
                    this.value = args.join(' ').trim();
                    const channel = settings.discordSavedChannels.find(channel => channel.name === channelName);
                    if (channel && !chtTabs.querySelector(`div[data-username="discord-${channel.id}"]`)) {
                        newDiscordTab(channelName, channel.id);
                    }
                    if (this.value) {
                        sendDiscordMessage(channel.id, this.value);
                        this.value = '';
                    }
                } else if (currentTab.startsWith("discord-")) {
                    if (this.value = this.value.trim()) {
                        sendDiscordMessage(currentTab.slice(8), this.value);
                        this.value = '';
                    }
                } else {
                    for (const [reg, rep] of wordsToReplace) {
                        this.value = this.value.replaceAll(reg, rep);
                    }
                }
            }
        });
        // Script devs so annoying can't even use that for settings pages
        // typeof GM_info === "undefined" ? Date.now() : GM_info?.script?.position + 3;
        const settingPageId = Date.now();
        chtTabs = document.getElementById("chtTabs");
        chtTabs.addEventListener("click", () => setTimeout(() => currentTab = chtTabs.querySelector("div.chat-tab.selected").dataset.username, 1));
        $('#settingTab2').after(`<button id="settingTab${settingPageId}" class="setting-tablink" onclick="openSettingPage(${settingPageId});">Advanced</button>`);
        $('#settingPage3').after(`
            <div id="settingPage${settingPageId}" class="setting-tabcontent">
                <div class="col-md-10 col-md-offset-1 stng" style="padding-left:20px;padding-right:10px;max-height:550px;overflow-y:auto;overflow-x:hidden;margin:0;width:calc(100% - 5px);">
                    <span style="margin:0;" class="hotkey-paragraph"> Animations</span>
                    <div class="row stng-row" style="font-size:14px;">
                        <div style="width:100%;padding:4px;">
                            <div style="display:flex;flex-wrap:wrap;padding:0px;width:100%;">
                                ${animationsIds.map(id => `<div style="${animations[id].style}" onclick="setAnimation(${id}, !this.classList.toggle('disabled'));" class="emote${settings.removeAnimations.includes(id) ? " disabled" : ''}">${animations[id].name}</div>`).join('')}
                            </div>
                        </div>
                    </div>
                    <span style="margin:0;" class="hotkey-paragraph"> Other</span>
                    <div class="row stng-row" style="font-size:14px;">
                        <label>
                            <input type="number" min="0" style="max-width:30px" value="${settings.maxStackableAnimations}" onchange="setMaxStackableAnimations(this.value);">
                            <span> Stackable Animations *</span>
                        </label>
                        <br>
                        <label>
                            <input type="checkbox" onchange="setRemoveFood(this.checked);" ${settings.removeFood ? " checked" : ''}>
                            <span> Remove Food *</span>
                        </label>
                    </div>
                    <span style="margin:0;" class="hotkey-paragraph"> Discord</span>
                    <div class="row stng-row" style="font-size:14px;">
                        <label>
                            <input type="checkbox" onchange="setDiscordChat(this.checked);" ${settings.discordChat ? " checked" : ''}>
                            <span> Discord Chat *</span>
                        </label>
                        <br>
                        <label>
                            <input type="checkbox" onchange="setDiscordPresenceChat(this.checked);" ${settings.discordPresence ? " checked" : ''}>
                            <span> Discord Agma.io Presence *</span>
                        </label>
                        <br>
                        <label>
                            <span> Discord Saved Channels *</span>
                            <textarea id="discordSavedChannels" style="resize: vertical; min-height: 100px; width: 100%;" placeholder="agma.io,942193976063197214\nname,channelId">${settings.discordSavedChannels.map(channel => channel.name + ',' + channel.id).join('\n')}</textarea>
                        </label>
                    </div>
                </div>
            </div>
        `);
        const discordSavedChannelsTextarea = document.getElementById("discordSavedChannels");
        discordSavedChannelsTextarea.addEventListener("blur", function(event) {
            settings.discordSavedChannels = [];
            for (const line of this.value.split('\n')) {
                let [name, id] = line.split(',');
                name = name?.trim();
                id = id?.trim();
                if (name && id) {
                    settings.discordSavedChannels.push({ name, id });
                } else {
                    swal('Invalid channels', 'Please ensure each line has exactly one comma separating the key and value.', 'error');
                    break;
                }
            }
        });
        discordSavedChannelsTextarea.addEventListener("keydown", function(event) {
            event.stopPropagation();
        });
        const style = document.createElement("style");
        style.innerHTML = `
            div.chat-tab[data-username^="discord-"] {
                background: #4954c5;
            }
            #settingPage${settingPageId} > div::-webkit-scrollbar {
                width: 8px;
                height: 8px;
            }
            #settingPage${settingPageId} > div::-webkit-scrollbar-track {
                background: #282934;
                border-radius: 10px;
            }
            #settingPage${settingPageId} > div::-webkit-scrollbar-thumb {
                background-color: #df8500;
                border-radius: 10px;
                border: 2px solid #282934;
            }
            #settingPage${settingPageId} div.emote {
                min-width: 30px;
                height: 30px;
                padding: 3px;
                border-radius: 5px;
                border-width: 1px;
                border-style: solid;
                margin: 4px;
                background-size: contain;
                background-repeat: no-repeat;
                cursor: pointer;
                text-overflow: ellipsis;
            }
            #settingPage${settingPageId} div.emote.disabled {
                text-decoration: line-through;
                text-decoration-thickness: 3px;
                text-decoration-color: red;
            }
        `;
        document.body.appendChild(style);

        const originalFind = $.prototype.find;
        unsafeWindow.$.prototype.find = function() {
            const res = originalFind.apply(this, arguments);
            if (
                discordUserContext
                && this.selector === "#contextMenu"
                && arguments?.[0] === "li.enabled.hover"
                && res?.length
            ) {
                if (res[0].id === "contextUserProfile") {
                    swal({
                        title: `<img src="https://cdn.discordapp.com/avatars/${discordUserContext.id}/${discordUserContext.avatar}" width="128" height="128" style="border-radius:50%;"><br><br><span>${discordUserContext.global_name || discordUserContext.username}</span><br><span style="font-size:12px;">${discordUserContext.username}</span>`,
                        html: true
                    });
                    return [];
                } else if (res[0].id === "contextDiscordMention") {
                    document.getElementById("chtbox").value += `<@${discordUserContext.id}> `;
                    return [];
                }
            }
            return res;
        }
        const originalAddClass = $.prototype.addClass;
        unsafeWindow.$.prototype.addClass = function() {
            if (discordUserContext && this.selector === "#contextMute") {
                $("#contextDiscordMention").show();
                originalAddClass.call($("#contextUserProfile"), "enabled");
                $("#contextPlayerSkin").css({ 'background-image': `url(https://cdn.discordapp.com/avatars/${discordUserContext.id}/${discordUserContext.avatar})` });
                $("#contextPartyLeave").hide();
                $("#contextPartyMessage").hide();
                $("#contextModerate").hide();
                $("#contextPartyInvite").hide();
                $("#contextFriendAdd").hide();
                $("#contextPrivateMessage").hide();
                $("#contextSpectate").hide();
                $("#contextPickpocket").hide();
                $("#contextMute").hide();
                return this;
            }
            return originalAddClass.apply(this, arguments);
        }
        const originalHide = $.prototype.hide;
        unsafeWindow.$.prototype.hide = function() {
            if (discordUserContext && this.selector === "#contextMenu") {
                $("#contextDiscordMention").hide();
                $("#contextPartyLeave").show();
                $("#contextPartyMessage").show();
                $("#contextModerate").show();
                $("#contextPartyInvite").show();
                $("#contextFriendAdd").show();
                $("#contextPrivateMessage").show();
                $("#contextSpectate").show();
                $("#contextPickpocket").show();
                $("#contextMute").show();
                discordUserContext = null;
            }
            return originalHide.apply(this, arguments);
        }
        $("#contextUserProfile").after('<li id="contextDiscordMention" class="contextmenu-item enabled" style="display: none;"><div class="fa fa-at fa-2x context-icon"></div><p>Mention</p></li>');

        let waitingResponse = false;
        const originalSwal = unsafeWindow.swal;
        unsafeWindow.swal = function() {
            if (
                !discordUserContext
                && typeof arguments[0] === "object"
                && "title" in arguments[0]
                && arguments[0].title.startsWith("<img src=\"")
            ) {
                if (waitingResponse) {
                    return;
                }
                const username = arguments[0].title.match(/>([^>]+)<\/span>/)?.[1];
                if (!username) {
                    return originalSwal.apply(this, arguments);
                }
                if (Date.now() - userprofiles[username]?.at < 86400000) {
                    arguments[0].text += getTable(userprofiles[username].data);
                } else {
                    waitingResponse = true;
                    $.getJSON("https://agma.io/royale_stats.php?user=" + username, payload => {
                        userprofiles[username] = { at: Date.now(), data: payload };
                        arguments[0].text += getTable(payload);
                        waitingResponse = false;
                        originalSwal.apply(this, arguments);
                    }).fail(() => {
                        userprofiles[username] = { at: Date.now(), data: null };
                        arguments[0].text += getTable(null);
                        waitingResponse = false;
                        originalSwal.apply(this, arguments);
                    }).always(() => {
                        waitingResponse = false;
                    });
                    return;
                }
            }
            return originalSwal.apply(this, arguments);
        }
        unsafeWindow.swal.close = originalSwal.close;
    });
    unsafeWindow.addEventListener("beforeunload", () => {
        discordWebSocket && discordWebSocket.close();
        GM_setValue("settings", settings);
        unsafeWindow.localStorage.userprofiles = JSON.stringify(userprofiles);
    });
    settings.discordChat && initDiscordWebSocket();
    console.log("%cAdvanced Agma - 0.0.2 Loaded", "font-weight: bold; font-size: 20pt; color: black;");
})();