Greasy Fork

Advanced Agma

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

目前为 2024-07-06 提交的版本。查看 最新版本

// ==UserScript==
// @name         Advanced Agma
// @namespace    http://tampermonkey.net/
// @version      0.1.0
// @author       Big watermelon (credits: Nersai, Vintrex)
// @description  clickable chat links, discord in game chat, 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/[email protected]/dist/pako.min.js
// @license      GPL-3.0-or-later
// @icon         
// @grant        unsafeWindow
// @grant        GM_setValue
// @grant        GM_getValue
// @run-at       document-start
// ==/UserScript==

/*
TODO: message doesn't send => color in red
TODO: message edit => edit message
TODO: message too long to fit in chat => wrap arround
TODO: implement support for:
- RECIPIENT_ADD (1)             { color: "#00FF00", name: "→", message: `... joined the group` }
- RECIPIENT_REMOVE (2)          { color: "#FF0000", name: "←", message: `... left the group` }
- CALL (3)                      { color: "#00FF00", name: "📞", message: `... started a phone call` }
- CHANNEL_NAME_CHANGE (4)       { color: "#00FF00", name: "🖊️", message: `... changed the group name to ...` }
- CHANNEL_PINNED_MESSAGE (6)    { color: "#00FF00", name: "🖊️", message: `... pinned a message` }
- GUILD_MEMBER_JOIN (7)         { color: "#00FF00", name: "→", message: `... joined the server` }
*/

(function() {
    "use strict";

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

    const discordToken = GM_getValue("discordToken", null);
    if (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) : {};
    const settings = GM_getValue("settings", {
        removeFood: true,
        removeAnimations: [3, 6, 7, 8, 11],
        maxStackableAnimations: 3,
        discordChat: true,
        discordSavedChannels: {},
        discordPresence: true
    });
    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 discordIcon = new Image();
    discordIcon.src = "";

    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 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 discordAuthor = 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 (sentMessagesIds.includes(message.id)) {
            return;
        }
        for (const name in settings.discordSavedChannels) {
            if (settings.discordSavedChannels[name] === message.channel_id) {
                sentMessagesIds.push(message.id);
                console.debug("[🔵] New message:", message);
                switch (message.type) {
                    case 0:  // MESSAGE
                    case 19: // REPLY
                        const tab = chtTabs.querySelector(`div[data-username="discord-${message.channel_id}"]`);
                        if (!tab) {
                            newDiscordTab(name, message.channel_id);
                        } else if (!tab.classList.contains('selected')) {
                            tab.classList.add('blink');
                            setTimeout(() => tab.classList.remove('blink'), 2000);
                        }
                        localChatMessages.push({
                            O: true,
                            v: 0,
                            o: 0,
                            P: 0,
                            U: 0,
                            name: message.author.global_name,
                            G: "discord-" + message.channel_id,
                            get color() {
                                discordAuthor = this._discordAuthor;
                                return discordUsername === this._discordAuthor.username ? "#313338" : "#5662E9";
                            },
                            message: (message.content || '') + message?.attachments?.map?.(({ filename }) => `<${filename}>`)?.join?.(' '),
                            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;
                }
                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;
                }
                break;
            case 1: // HEARTBEAT
                sendWebsocketDiscordMessage(1, null);
                break;
            case 11: // HEARTBEAT_ACK
                break;
            default:
                console.debug("[🔵] Unhandled message type:", message);
        }
    }
    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) {
        s /= 1000;
        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 * 1000)
            + "</td></tr><tr><td>Splits per seconds: </td><td>"
            + (payload.splits_count / payload.total_time_alive).toFixed(2)
            + "</td></tr></table>"
        );
    }

    let jQueryWrapper = unsafeWindow.$;
    Object.defineProperty(unsafeWindow, "$", {
        set: function(jQuery) {
            const jQueryFind = jQuery.prototype.find;
            jQuery.prototype.find = function() {
                const res = jQueryFind.apply(this, arguments);
                if (
                    this.selector === "#contextMenu"
                    && arguments?.[0] === "li.enabled.hover"
                    && res?.length
                    && res[0].id === "contextUserProfile"
                    && discordAuthor
                ) {
                    // FIXIT: This is bugged but whatever I'm lazy
                    swal({
                        title: `<img src="https://cdn.discordapp.com/avatars/${discordAuthor.id}/${discordAuthor.avatar}" width="128" height="128" style="border-radius:50%;"><br><br><span>${discordAuthor.global_name}</span><br><span style="font-size:12px;">${discordAuthor.username}</span>`,
                        html: true
                    });
                    discordAuthor = null;
                    return [];
                }
                return res;
            }
            const jQueryShow = jQuery.prototype.show;
            jQuery.prototype.show = function() {
                if (discordAuthor !== null && this.selector === "#contextMenu") {
                    jQuery("#contextUserProfile").addClass("enabled");
                    jQuery("#contextFriendAdd").removeClass("enabled");
                    jQuery("#contextPrivateMessage").removeClass("enabled");
                    jQuery("#contextMute").removeClass("enabled");
                }
                return jQueryShow.apply(this, arguments)
            }
            const jQueryHide = jQuery.prototype.hide;
            jQuery.prototype.hide = function() {
                if (discordAuthor !== null && this.selector === "#contextMenu") {
                    discordAuthor = null;
                }
                return jQueryHide.apply(this, arguments)
            }
            return jQueryWrapper = jQuery
        },
        get: function() {
            return jQueryWrapper;
        }
    });
    let swalWrapper = unsafeWindow.swal;
    Object.defineProperty(unsafeWindow, "swal", {
        set: function(originalSwal) {
            let waitingResponse = false;
            swalWrapper = function(params, callback) {
                if (!discordAuthor && params?.title?.startsWith("<img src=\"")) {
                    if (waitingResponse) {
                        return;
                    }
                    const username = params.title.match(/>([^>]+)<\/span>/)?.[1];
                    if (!username) {
                        return originalSwal(params, callback);
                    }
                    if (Date.now() - userprofiles[username]?.at < 86400000) {
                        params.text += getTable(userprofiles[username].data);
                        originalSwal(params, callback);
                    } else {
                        waitingResponse = true;
                        $.getJSON("https://agma.io/royale_stats.php?user=" + username, payload => {
                            userprofiles[username] = { at: Date.now(), data: payload };
                            params.text += getTable(payload);
                            waitingResponse = false;
                            originalSwal(params, callback);
                        }).fail(() => {
                            userprofiles[username] = { at: Date.now(), data: null };
                            params.text += getTable(null);
                            waitingResponse = false;
                            originalSwal(params, callback);
                        }).always(() => {
                            waitingResponse = false;
                        });
                    }
                } else {
                    return originalSwal(params, callback);
                }
            }
            swalWrapper.close = function() {
                discordAuthor = null;
                return originalSwal.close.apply(this, arguments);
            }
            return swalWrapper;
        },
        get: function() {
            return swalWrapper;
        }
    });
    const originalDrawImage = unsafeWindow.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) {
                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._discordAuthor) {
                            cache.icons = [ 14 ];
                        }
                        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>`);
    unsafeWindow.sendDiscordMessage = (channel, content) => settings.discordChat && 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 documentLoaded = false;
    document.addEventListener("DOMContentLoaded", () => {
        if (documentLoaded) {
            return;
        }
        documentLoaded = 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();
                    if (settings.discordSavedChannels[channelName] && !chtTabs.querySelector(`div[data-username="discord-${settings.discordSavedChannels[channelName]}"]`)) {
                        newDiscordTab(channelName, settings.discordSavedChannels[channelName]);
                    }
                    if (this.value) {
                        sendDiscordMessage(settings.discordSavedChannels[channelName], 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);
                    }
                }
            }
        });
        chtTabs = document.getElementById("chtTabs");
        chtTabs.addEventListener("click", () => setTimeout(() => currentTab = chtTabs.querySelector("div.chat-tab.selected").dataset.username, 1));
        document.querySelector(".setting-tabs").innerHTML += `<button id="settingTab5" class="setting-tablink" onclick="openSettingPage(5);" style="width: 9%; font-size: calc(0.3vw + 7.5px);"><div class="fa fa-plus fa-lg" style="font-size: 15px; color: lightgray;"></div></button>`;
        document.getElementById("setting").innerHTML += `
            <div id="settingPage5" 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">${Object.entries(settings.discordSavedChannels).map(entry => entry.join(',')).join('\n')}</textarea>
                        </label>
                    </div>
                </div>
            </div>
        `;
        const discordSavedChannelsTextarea = document.getElementById("discordSavedChannels");
        discordSavedChannelsTextarea.addEventListener("blur", function(event) {
            const result = {};
            for (const line of this.value.split('\n')) {
                let [name, channel] = line.split(',');
                name = name.trim();
                channel = channel.trim();
                if (name && channel) {
                    result[name] = channel;
                } else {
                    alert('Invalid content. Please ensure each line has exactly one comma separating the key and value.');
                    this.value = Object.entries(settings.discordSavedChannels).map(entry => entry.join(',')).join('\n')
                    return 
                }
            }
            settings.discordSavedChannels = result;
        });
        discordSavedChannelsTextarea.addEventListener("keydown", function(event) {
            event.stopPropagation();
        });
        const style = document.createElement("style");
        style.innerHTML = `
            div.chat-tab[data-username^="discord-"] {
                background: #4954c5;
            }
            .setting-tablink {
                width: auto !important;
            }
            #settingTab4, #settingTab5 {
                width: 9% !important;
            }
            .setting-tabcontent > div::-webkit-scrollbar {
                width: 8px;
                height: 8px;
            }
            .setting-tabcontent > div::-webkit-scrollbar-track {
                background: #282934;
                border-radius: 10px;
            }
            .setting-tabcontent > div::-webkit-scrollbar-thumb {
                background-color: #df8500;
                border-radius: 10px;
                border: 2px solid #282934;
            }
            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;
            }
            div.emote.disabled {
                text-decoration: line-through;
                text-decoration-thickness: 3px;
                text-decoration-color: red;
            }
        `;
        document.body.appendChild(style);
    });
    unsafeWindow.addEventListener("beforeunload", () => {
        GM_setValue("settings", settings);
    });
    settings.discordChat && initDiscordWebSocket();
    console.log("%cAdvanced Agma - 0.0.2 Loaded", "padding: 40px; font-weight: bold; font-size: 32pt; color: black;");
})();