DeepCo Morale Elevator

Toggle emoji picker appended to chat input

K instalaci tototo skriptu si budete muset nainstalovat rozšíření jako Tampermonkey, Greasemonkey nebo Violentmonkey.

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

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Userscripts.

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

K instalaci tohoto skriptu si budete muset nainstalovat manažer uživatelských skriptů.

(Už mám manažer uživatelských skriptů, nechte mě ho nainstalovat!)

Advertisement:

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

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

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

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

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

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

(Už mám manažer uživatelských stylů, nechte mě ho nainstalovat!)

Advertisement:

// ==UserScript==
// @name         DeepCo Morale Elevator
// @namespace    deepco
// @version      2026.04.16
// @description  Toggle emoji picker appended to chat input
// @author       M3P / ChatGPT
// @match        https://deepco.app/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=deepco.app
// @license      MIT
// @grant        GM_getValue
// @grant        GM_setValue
// ==/UserScript==

(function () {
    'use strict';

    //--------------------------------------------------
    // Default emoji string
    const DEFAULT_EMOJIS = "⚙︎😁👍🥳😉😂🤣😛🤔🤨🙄🧐😱🤯😭😢🐸";

    //--------------------------------------------------
    // Grapheme segmentation
    const segmenter =
          typeof Intl !== 'undefined' && Intl.Segmenter
    ? new Intl.Segmenter(undefined, { granularity: 'grapheme' })
    : null;

    function splitEmojis(str) {
        if (!str) return [];
        if (segmenter) {
            return Array.from(segmenter.segment(str), s => s.segment);
        }
        return Array.from(str);
    }

    //--------------------------------------------------
    // Emoji detection
    const EXTENDED_PICTOGRAPHIC_RE = /\p{Extended_Pictographic}/u;
    const REGIONAL_INDICATOR_RE = /^\p{Regional_Indicator}{2}$/u;
    const KEYCAP_RE = /^[0-9#*]\uFE0F?\u20E3$/u;

    function isEmojiLike(grapheme) {
        return (
            EXTENDED_PICTOGRAPHIC_RE.test(grapheme) ||
            REGIONAL_INDICATOR_RE.test(grapheme) ||
            KEYCAP_RE.test(grapheme)
        );
    }

    function uniqueEmojiListFromString(str) {
        const seen = new Set();
        const out = [];

        for (const g of splitEmojis(str)) {
            if (isEmojiLike(g) && !seen.has(g)) {
                seen.add(g);
                out.push(g);
            }
        }

        return out;
    }

    //--------------------------------------------------
    // Storage-backed emoji list
    function loadEmojiList() {
        const raw = GM_getValue('emoji_string', DEFAULT_EMOJIS);

        if (Array.isArray(raw)) {
            return uniqueEmojiListFromString(raw.join(''));
        }

        if (typeof raw === 'string') {
            const list = uniqueEmojiListFromString(raw);
            return list.length ? list : uniqueEmojiListFromString(DEFAULT_EMOJIS);
        }

        return uniqueEmojiListFromString(DEFAULT_EMOJIS);
    }

    let emojiList = loadEmojiList();

    function saveEmojiList() {
        GM_setValue('emoji_string', emojiList.join(''));
    }

    // Optional helper for updating emoji string from console
    window.setEmojiString = function (str) {
        emojiList = uniqueEmojiListFromString(String(str || ''));
        saveEmojiList();
        renderPickerGrid(document.querySelector('#tm-emoji-grid'));
    };

    //--------------------------------------------------
    function isVisible(el) {
        return !!(el && el.offsetParent !== null);
    }

    function renderPickerGrid(grid) {
        if (!grid) return;

        const frag = document.createDocumentFragment();

        for (const emoji of emojiList) {
            const btn = document.createElement('button');
            btn.textContent = emoji;
            btn.type = 'button';
            btn.style.fontSize = '18px';
            btn.style.cursor = 'pointer';
            btn.style.padding = '2px 4px';

            btn.addEventListener('click', (e) => {
                e.preventDefault();
                e.stopPropagation();

                insertEmoji(emoji);
                promoteEmoji(emoji);
            });
            frag.appendChild(btn);
        }

        grid.replaceChildren(frag);
    }

    function promoteEmoji(emoji) {
        const idx = emojiList.indexOf(emoji);
        if (idx === 0) return false;

        if (idx > -1) {
            emojiList.splice(idx, 1);
        }

        emojiList.unshift(emoji);
        saveEmojiList();
        renderPickerGrid(document.querySelector('#tm-emoji-grid'));
        return true;
    }

    function extractEmojis(text) {
        const found = [];
        const seen = new Set();

        for (const g of splitEmojis(text)) {
            if (isEmojiLike(g) && !seen.has(g)) {
                seen.add(g);
                found.push(g);
            }
        }

        return found;
    }

    function ingestEmojisFromMessage(text) {
        const found = extractEmojis(text);
        if (!found.length) return;

        let changed = false;

        // Reverse the per-message list so the order in the typed message
        // is preserved when items are unshifted onto the front.
        for (const emoji of found.slice().reverse()) {
            const idx = emojiList.indexOf(emoji);

            if (idx === 0) {
                continue;
            }

            if (idx > -1) {
                emojiList.splice(idx, 1);
            }

            emojiList.unshift(emoji);
            changed = true;

            if (idx === -1) {
                console.log(`New Emoji detected! ${emoji}`);
            }
        }

        if (changed) {
            saveEmojiList();
            renderPickerGrid(document.querySelector('#tm-emoji-grid'));
        }
    }

    //--------------------------------------------------
    function insertEmoji(emoji) {
        const el = document.querySelector('#message');
        if (!el) return;

        if (el.tagName === 'TEXTAREA' || el.tagName === 'INPUT') {
            const start = el.selectionStart ?? el.value.length;
            const end = el.selectionEnd ?? el.value.length;
            el.value = el.value.slice(0, start) + emoji + el.value.slice(end);
            el.selectionStart = el.selectionEnd = start + emoji.length;
            el.dispatchEvent(new Event('input', { bubbles: true }));
            return;
        }

        if (el.isContentEditable) {
            document.execCommand('insertText', false, emoji);
        }
    }

    //--------------------------------------------------
    function buildPicker() {
        const picker = document.createElement('div');
        picker.id = 'tm-emoji-picker';
        Object.assign(picker.style, {
            display: 'none',
            marginTop: '6px',
            padding: '6px',
            border: '1px solid #555',
            borderRadius: '6px',
            background: '#222',
        });

        const grid = document.createElement('div');
        grid.id = 'tm-emoji-grid';
        Object.assign(grid.style, {
            display: 'flex',
            flexWrap: 'wrap',
            gap: '4px',
        });

        picker.appendChild(grid);
        renderPickerGrid(grid);


        return picker;
    }

    function bindSubmissionTracking(input) {
        if (input.dataset.tmEmojiHooked) return;
        input.dataset.tmEmojiHooked = '1';

        const processCurrentMessage = () => {
            ingestEmojisFromMessage(input.value);
        };

        const form = input.closest('form');
        if (form) {
            form.addEventListener(
                'submit',
                () => {
                    processCurrentMessage();
                },
                true
            );
            return;
        }

        // Fallback for non-form submit flows
        input.addEventListener('keydown', e => {
            if (e.key === 'Enter' && !e.shiftKey && !e.isComposing) {
                setTimeout(processCurrentMessage, 0);
            }
        });
    }

    //--------------------------------------------------
    function inject() {
        const input = document.querySelector('#message');
        if (input) {
            bindSubmissionTracking(input);
        }

        const panel = document.querySelector('#chat');
        if (!isVisible(panel)) return;

        const title = [...panel.querySelectorAll('h2')].find(
            h => h.textContent.trim() === 'Comm Terminal'
        );
        if (!title) return;

        const headerContainer = title.parentElement?.parentElement;
        if (!headerContainer) return;

        if (!document.querySelector('#tm-emoji-toggle')) {
            const toggle = document.createElement('button');
            toggle.id = 'tm-emoji-toggle';
            toggle.textContent = '😁';
            toggle.type = 'button';
            toggle.style.marginLeft = '8px';
            toggle.style.fontSize = '20px';
            toggle.style.cursor = 'pointer';

            toggle.addEventListener("click", (e)=>{
                e.preventDefault();
                e.stopPropagation();

                const picker = document.querySelector('#tm-emoji-picker');
                if (!picker) return;

                picker.style.display =
                    picker.style.display === "none" ? "block" : "none";
            });

            headerContainer.style.display = 'flex';
            headerContainer.style.alignItems = 'center';

            title.parentElement.insertAdjacentElement('afterend', toggle);

            const chatpanel = document.querySelector('#chat-messages');
            if (chatpanel?.parentElement && !document.querySelector('#tm-emoji-picker')) {
                chatpanel.parentElement.prepend(buildPicker());
            }
        }
    }

    //--------------------------------------------------
    const observer = new MutationObserver(() => {
        inject();
    });

    observer.observe(document.body, {
        childList: true,
        subtree: true,
        attributes: true,
        attributeFilter: ['class', 'style'],
    });

    inject();
})();