MyAnimeList.net GIF inserter

Come to Anime moments club 🎊 https://myanimelist.net/clubs.php?cid=93838 🎉! Add convenient Tenor.com gif image inserter into MyAnimeList.net comment editor.

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         MyAnimeList.net GIF inserter
// @namespace    http://tampermonkey.net/
// @version      2025-11-26.1
// @description  Come to Anime moments club 🎊 https://myanimelist.net/clubs.php?cid=93838 🎉! Add convenient Tenor.com gif image inserter into MyAnimeList.net comment editor.
// @author       AlexDEV.pro
// @match        *://myanimelist.net/*
// @icon         
// @grant        none
// ==/UserScript==

(function() {
    'use strict';

    const apiKey = 'AIzaSyDwtuo8eUG5sg6KPbBW_1-gizZBjAiRIqE';
    const clientKey = 'MALGI';

    const popularImagesLocalStorageKey = 'malgiPopularImages';
    const insertOptionWidthLocalStorageKey = 'malgiInsertOptionWidth';
    const displayOptionSquareLocalStorageKey = 'malgiDisplayOptionSquare';

    const i18n = {
        en: {
            insertWidthLabelText: 'Insert width (px):',
            isCoverObjectFitCheckboxLabelText: 'Display square',
            loadMoreButtonText: 'More',
            searchQueryPrefixOptionAllText: 'All',
            searchQueryPrefixOptionAnimeText: 'Anime',
            searchFilterOptionGifsText: 'GIFs',
            searchFilterOptionAnimatedStickersText: 'Animated stickers',
            searchFilterOptionStaticStickersText: 'Static stickers',
            searchFilterOptionAllStickersText: 'All stickers'
        },
        ru: {
            insertWidthLabelText: 'Ширина вставки (пиксели):',
            isCoverObjectFitCheckboxLabelText: 'Квадратное отображение',
            loadMoreButtonText: 'Ещё',
            searchQueryPrefixOptionAllText: 'Всё',
            searchQueryPrefixOptionAnimeText: 'Аниме',
            searchFilterOptionGifsText: 'Гифки',
            searchFilterOptionAnimatedStickersText: 'Анимированные стикеры',
            searchFilterOptionStaticStickersText: 'Статические стикеры',
            searchFilterOptionAllStickersText: 'Все стикеры'
        }
    };

    const currentUserLocale = navigator.language.split('-')[0] || 'en';
    const t = i18n[currentUserLocale] || i18n.en;

    const popupSpacingModes = {
        'default': undefined,
        'sceditor': 'sc',
        'table': 't'
    };
    const popupSpacingPx = 41;
    const popupSpacingInScEditorPx = 30;
    const popupSpacingInTablePx = 19;

    const searchQueryPrefixOptions = [
        { value: '', text: t.searchQueryPrefixOptionAllText },
        { value: 'anime ', text: t.searchQueryPrefixOptionAnimeText }
    ];
    const searchQueryPrefixDefaultOption = searchQueryPrefixOptions[1];

    const searchFilterOptions = [
        { value: '', text: t.searchFilterOptionGifsText },
        { value: 'sticker,-static', text: t.searchFilterOptionAnimatedStickersText },
        { value: 'sticker,static', text: t.searchFilterOptionStaticStickersText },
        { value: 'sticker', text: t.searchFilterOptionAllStickersText }
    ];
    const searchFilterDefaultOption = searchFilterOptions[0];

    const searchQueryRowsCount = 2;

    // How close to the bottom before triggering load more function.
    const loadMoreTriggerDistancePx = 5;

    const insertWidthMinPx = 40;
    const insertWidthMaxPx = 660;
    const insertWidthStepPx = 5;
    const insertWidthPresetsPx = [40, 50, 60, 80, 90, 100, 110, 120, 150, 180, 200, 235];
    let insertWidthDefaultPx = insertWidthPresetsPx[3];

    let displayOptionSquareDefault = true;

    const imageGridMinWidthPx = 100;
    const imageGridGapPx = 5;

    const maxPopularImagesCount = 500;

    let searchRequestAbortController;
    let popupContainerEl, insertWidthInputEl, searchQueryPrefixSelectEl, searchFilterSelectEl, searchInputEl, searchButtonEl, imagesContainerEl, resultsContainerEl, loadMoreButtonEl, popularImagesContainerEl;
    let currentAnchorEl;

    const popularImageClickInterval = 500;

    const style = document.createElement('style');
    style.textContent = `
        :root {
            --malgi-text-color: #5E5E5E;
            --malgi-popup-height: 300px;
            --malgi-popup-spacing: 0px;
            --malgi-popup-extra-height: 0;
        }

        .malgi-popup-anchor { position: relative; }
        .malgi-dialog-open-button { display: flex; width: 1.8em; height: 1.8em; margin: -3px 3px 0 3px; padding: 0; font-size: 1.1em; align-items: center; justify-content: center; position: absolute; top: 0; right: 0; }
        .malgi-dialog-open-button.sceditor-button { position: unset; margin-bottom: 2px; margin-left: 8px; border-width: 1px; }
        #malgi-popup-container { display: none; z-index: 99; position: absolute; top: calc(-1 * var(--malgi-popup-height) - var(--malgi-popup-spacing)); transition: top 0.3s ease; height: var(--malgi-popup-height); padding: 0 5px 5px 5px; overflow-y: scroll; color: var(--malgi-text-color); margin-left: 1px; margin-right: 1px; background: white; box-shadow: 0 0 0.2em #BABABA; }
        #malgi-popup-container img { cursor: pointer; width: ${imageGridMinWidthPx}px; height: 100px; object-fit: cover; margin: auto; background-color: #EEE; background-image: linear-gradient(90deg, #EEE 25%, #F5F5F5 50%, #EEE 75%); background-size: 200% 100%; animation: loading-shimmer 1.5s infinite; }
        #malgi-popup-container img.loaded { background: none; }
        #malgi-popup-container img:hover { opacity: 0.5; }
        #malgi-popup-container button:active { box-shadow: inset 0 0 4px lightgray; }

        #malgi-popular-img-container, #malgi-results-container { display: grid; grid-template-columns: repeat(auto-fit, minmax(${imageGridMinWidthPx}px, 1fr)); justify-content: center; gap: ${imageGridGapPx}px; }

        #malgi-images-container.malgi-object-fit-contain img { object-fit: contain; }
        #malgi-images-container { padding: 5px; }

        #malgi-toolbar-container { position: sticky; top: 0; z-index: 1000; background: white; padding: 5px; margin: -5px; display: flex; align-items: center; column-gap: 1em; row-gap: 0.5em; flex-wrap: wrap; }
        #malgi-toolbar-container div { display: flex; align-items: center; }
        #malgi-toolbar-container div:first-child { flex: 1 }
        #malgi-toolbar-container input, #malgi-toolbar-container select { align-self: stretch; padding: 0.5em; border: 1px solid lightgray; font-size: 1em; color: var(--malgi-text-color); }
        #malgi-toolbar-container input { box-sizing: border-box; height: auto; line-height: normal; }
        #malgi-toolbar-container select { padding-top: 0.34em; }
        #malgi-toolbar-container button { border: 1px solid lightgray; color: var(--malgi-text-color); font-size: 13px; height: 30px; padding: 0 5px; align-items: center; display: flex; }

        #malgi-search-input { flex: 1; }
        #malgi-search-button { width: 30px; justify-content: center; }

        #malgi-load-more-button-container { display: flex; justify-content: center; padding: 1em; }
        #malgi-load-more-button { border: 1px solid lightgray; padding: 5px; text-align: center; display: none; }
        #malgi-load-more-button.is-loading { background-size: contain; background-repeat: no-repeat; background-position: center; border-color: transparent; color: transparent; background-image: url("");  }

        .malgi-popup-extra-height { margin-top: calc(var(--malgi-popup-extra-height) + var(--malgi-popup-spacing)) !important; }
        .malgi-left-border-stack > *:not(:first-child) { border-left: 0; }
        .malgi-flex-wrap { display: flex; flex-wrap: wrap; row-gap: 0.25em; }
        .malgi-label { margin-rigth: 0.5em; }
        .malgi-checkbox-label { display: flex; align-items: center; gap: 0.5em; }
        .malgi-no-selection {
            user-select: none; /* Prevents text selection */
            -webkit-user-select: none; /* Safari/Chrome */
            -moz-user-select: none; /* Firefox */
            -ms-user-select: none; /* IE/Edge */
        }

        @keyframes loading-shimmer {
            0% {
                background-position: 200% 0;
            }
            100% {
                background-position: -200% 0;
            }
        }

        @media(max-width: 375px) {
            #malgi-toolbar-container #malgi-search-input { width: 1em }
        }
        @media(max-width: 450px) {
            #malgi-toolbar-container #malgi-search-filter-select { width: 5em }
        }
    `;
    document.head.appendChild(style);

    // Initializes the popup. There must be only one popup but it can used in multiple editors, one at a time.
    const initPopupElements = () => {
        popupContainerEl = document.createElement('div');
        popupContainerEl.id = 'malgi-popup-container';

        const toolbarContainerEl = document.createElement('div');
        toolbarContainerEl.id = 'malgi-toolbar-container';

        const searchGroupEl = document.createElement('div');
        searchGroupEl.className = 'malgi-left-border-stack';

        searchQueryPrefixSelectEl = document.createElement('select');
        searchQueryPrefixSelectEl.id = 'malgi-search-query-prefix-select';
        searchQueryPrefixOptions.forEach(option => {
            const optionEl = document.createElement('option');

            optionEl.value = option.value;
            optionEl.textContent = option.text;

            if (option === searchQueryPrefixDefaultOption) optionEl.selected = true;

            searchQueryPrefixSelectEl.appendChild(optionEl);
        });
        searchQueryPrefixSelectEl.addEventListener('change', onSearchParamsChange);

        searchFilterSelectEl = document.createElement('select');
        searchFilterSelectEl.id = 'malgi-search-filter-select';
        searchFilterOptions.forEach(option => {
            const optionEl = document.createElement('option');

            optionEl.value = option.value;
            optionEl.textContent = option.text;

            if (option === searchFilterDefaultOption) optionEl.selected = true;

            searchFilterSelectEl.appendChild(optionEl);
        });
        searchFilterSelectEl.addEventListener('change', onSearchParamsChange);

        searchInputEl = document.createElement('input');
        searchInputEl.id = 'malgi-search-input';
        searchInputEl.type = 'text';
        searchInputEl.addEventListener('keypress', (event) => { if (event.key === 'Enter') { event.preventDefault(); event.stopPropagation(); handleSearch(); } });
        searchInputEl.addEventListener('focus', () => searchInputEl.select());

        const searchClearButtonEl = document.createElement('button');
        searchClearButtonEl.id = 'malgi-search-clear-button';
        searchClearButtonEl.type = 'button';
        searchClearButtonEl.textContent = '❌';
        searchClearButtonEl.addEventListener('click', goToPopularImagesScreen);

        searchButtonEl = document.createElement('button');
        searchButtonEl.id = 'malgi-search-button';
        searchButtonEl.type = 'button';
        searchButtonEl.textContent = '🔎';
        searchButtonEl.addEventListener('click', handleSearch);

        searchGroupEl.appendChild(searchQueryPrefixSelectEl);
        searchGroupEl.appendChild(searchFilterSelectEl);
        searchGroupEl.appendChild(searchInputEl);
        searchGroupEl.appendChild(searchClearButtonEl);
        searchGroupEl.appendChild(searchButtonEl);

        const insertWidthGroupEl = document.createElement('div');
        insertWidthGroupEl.className = 'malgi-flex-wrap';

        const insertWidthLabelEl = document.createElement('span');
        insertWidthLabelEl.className = 'malgi-label malgi-no-selection';
        insertWidthLabelEl.textContent = t.insertWidthLabelText;

        insertWidthInputEl = document.createElement('input');
        insertWidthInputEl.id = 'malgi-insert-width-input';
        insertWidthInputEl.type = 'number';
        insertWidthInputEl.min = insertWidthMinPx;
        insertWidthInputEl.max = insertWidthMaxPx;
        insertWidthInputEl.step = insertWidthStepPx;
        insertWidthInputEl.value = insertWidthDefaultPx;
        insertWidthInputEl.addEventListener('change', (event) => localStorage.setItem(insertOptionWidthLocalStorageKey, event.target.value));

        insertWidthGroupEl.appendChild(insertWidthLabelEl);
        insertWidthGroupEl.appendChild(insertWidthInputEl);
        for(const presetValue of insertWidthPresetsPx) {
            const insertWidthPresetButton = document.createElement('button');

            insertWidthPresetButton.type = 'button';
            insertWidthPresetButton.textContent = presetValue;
            insertWidthPresetButton.addEventListener('click', () => handlePresetClick(presetValue));

            insertWidthGroupEl.appendChild(insertWidthPresetButton);
        }

        const displayOptionsGroupEl = document.createElement('div');
        displayOptionsGroupEl.id = 'malgi-display-options-container';
        displayOptionsGroupEl.className = 'malgi-flex-wrap';

        const isCoverObjectFitCheckboxEl = document.createElement('input');
        isCoverObjectFitCheckboxEl.type = 'checkbox';
        isCoverObjectFitCheckboxEl.checked = displayOptionSquareDefault;
        isCoverObjectFitCheckboxEl.addEventListener('change', (event) => setObjectFitCover(event.target.checked));

        const isCoverObjectFitCheckboxLabelEl = document.createElement('label');
        isCoverObjectFitCheckboxLabelEl.className = 'malgi-checkbox-label';
        isCoverObjectFitCheckboxLabelEl.appendChild(isCoverObjectFitCheckboxEl);
        isCoverObjectFitCheckboxLabelEl.append(t.isCoverObjectFitCheckboxLabelText);

        displayOptionsGroupEl.appendChild(isCoverObjectFitCheckboxLabelEl);

        toolbarContainerEl.appendChild(searchGroupEl);
        toolbarContainerEl.appendChild(insertWidthGroupEl);
        toolbarContainerEl.appendChild(displayOptionsGroupEl);

        imagesContainerEl = document.createElement('div');
        imagesContainerEl.id = 'malgi-images-container';
        imagesContainerEl.className = 'malgi-no-selection';
        if (!displayOptionSquareDefault) {
            imagesContainerEl.classList.add('malgi-object-fit-contain');
        }

        resultsContainerEl = document.createElement('div');
        resultsContainerEl.id = 'malgi-results-container';

        popularImagesContainerEl = document.createElement('div');
        popularImagesContainerEl.id = 'malgi-popular-img-container';

        const loadMoreButtonContainerEl = document.createElement('div');
        loadMoreButtonContainerEl.id = 'malgi-load-more-button-container';

        loadMoreButtonEl = document.createElement('button');
        loadMoreButtonEl.id = 'malgi-load-more-button';
        loadMoreButtonEl.type = 'button';
        loadMoreButtonEl.textContent = t.loadMoreButtonText;
        loadMoreButtonEl.addEventListener('click', () => {
            searchTenor(searchInputEl.value, true);
        });

        loadMoreButtonContainerEl.appendChild(loadMoreButtonEl);

        imagesContainerEl.appendChild(resultsContainerEl);
        imagesContainerEl.appendChild(popularImagesContainerEl);
        imagesContainerEl.appendChild(loadMoreButtonContainerEl);

        popupContainerEl.appendChild(toolbarContainerEl);
        popupContainerEl.appendChild(imagesContainerEl);
    }

    // Initializes the popup open button and popup anchor for every comment editor on the page. Each button click appends the single popup to its corresponding popup anchor.
    const initButtonAndAnchorForEveryEditor = (injectionTargets) => {
        if (!injectionTargets.length) return; // No editor toolbars found on the current page, nothing to do here.

        // Ensure the targets are fully loaded.
        setTimeout(() => {
            for(const target of injectionTargets) {
                // Skip the target if it was processed before.
                //if (target.dataset.malgiProcessed) continue;
                if (target.parentElement.querySelector('.malgi-popup-anchor')) return;

                const popupAnchorEl = document.createElement('div');
                popupAnchorEl.className = 'malgi-popup-anchor';

                let buttonTargetEl = target.parentElement.querySelector('.sceditor-toolbar');
                const isScEditor = !!buttonTargetEl;
                if (!buttonTargetEl) {
                    buttonTargetEl = target.parentElement.parentElement;
                }
                if (buttonTargetEl.classList.contains('reply-container')) {
                    continue; // Skip reply container as this one catches as an intermediate state of dynamically loaded editor.
                }
                if (buttonTargetEl) {
                    // Set button target element position to relative for absolute button positioning.
                    buttonTargetEl.style.position = 'relative';

                    // Insert anchor element only if button target element is also found.
                    target.parentElement.insertBefore(popupAnchorEl, target.parentElement.firstChild);

                    // Mark the table mode on the popup anchor element if it is a table mode for further spacing adjuctments.
                    if (buttonTargetEl.tagName === 'TR') {
                        popupAnchorEl.dataset.spacingMode = popupSpacingModes.table;
                    }

                    const dialogOpenButtonEl = document.createElement('button');
                    dialogOpenButtonEl.className = 'malgi-dialog-open-button';
                    dialogOpenButtonEl.type = 'button';
                    dialogOpenButtonEl.textContent = '🌊';
                    dialogOpenButtonEl.addEventListener('click', () => onDialogOpenButtonClick(popupAnchorEl));

                    if (isScEditor) {
                        popupAnchorEl.dataset.spacingMode = popupSpacingModes.sceditor;

                        dialogOpenButtonEl.classList.add('sceditor-button');

                        const scEditorGroupEl = document.createElement('div');
                        scEditorGroupEl.className = 'sceditor-group';
                        scEditorGroupEl.appendChild(dialogOpenButtonEl);

                        buttonTargetEl.appendChild(scEditorGroupEl);
                    } else {
                        buttonTargetEl.appendChild(dialogOpenButtonEl);
                    }
                }

                //target.dataset.malgiProcessed = true;
            }
        }, 0);
    }

    const getEditorEl = () => {
        return currentAnchorEl.parentElement.querySelector('textarea:not(.g-recaptcha-response)');
    }

    const handleSearch = () => {
        if (!searchButtonEl) throw new Error('Search button is not found.');
        if (!searchInputEl) throw new Error('Search button is not found.');

        if (searchInputEl.value) {
            if (searchButtonEl.disabled) return; // Prevent requests while search button is disabled.

            searchTenor(searchInputEl.value)
        } else {
            // If no search query provided then display popular images.
            goToPopularImagesScreen();
        }
    }

    const handlePresetClick = (presetValue) => {
        if (!insertWidthInputEl) throw new Error('No insert width element found.');

        insertWidthInputEl.value = presetValue;
        localStorage.setItem(insertOptionWidthLocalStorageKey, presetValue)
    }

    const handleImageSelection = (editorEl, imgSrc, tenorPageUrl) => {
        if (!editorEl) return; // If the user is in preview mode, the editor element doesn't exist and there is nothing to do.

        // In case editor isn't focused.
        editorEl.focus();

        // Ensure focus has been set.
        setTimeout(() => {
            const selection = window.getSelection();
            if (selection.rangeCount > 0) {
                const anchorNode = selection.anchorNode;
                if (isSelectionInsideContentEditable(selection)) {
                    // Get current content editable element selection.
                    const range = selection.getRangeAt(0);

                    //// Remove current selection.
                    //range.deleteContents();

                    // Prepare insertion HTML. The line breaks are odd but they prevent unnecessary spaces while keeping it convenient to edit.
                    const htmlSnippet =
`<span data-vue-node-view-wrapper="" contenteditable="false" draggable="true" style="white-space: normal;"><span
    class="b-image check-width" data-attrs="{&quot;id&quot;:null,&quot;src&quot;:&quot;${imgSrc}&quot;,&quot;isPoster&quot;:false,&quot;width&quot;:${insertWidthInputEl.value},&quot;height&quot;:null,&quot;isNoZoom&quot;:true,&quot;class&quot;:null}" data-image="[img]">
        <div class="controls">
            <a class="prosemirror-open" href="${imgSrc}" target="_blank"></a><!----><div class="delete"></div>
        </div>
        <img src="${imgSrc}"
    ></span
></span>`;

                    // Create a temporary container to turn the HTML string into nodes.
                    const temp = document.createElement('div');
                    temp.innerHTML = htmlSnippet;
                    const fragment = document.createDocumentFragment();
                    let node;
                    while ((node = temp.firstChild)) fragment.appendChild(node); // Move node one by one from temp into fragment.

                    // If the selection is inside a text node, split it.
                    if (range.startContainer.nodeType === Node.TEXT_NODE) {
                        const textNode = range.endContainer;
                        const offset = range.endOffset;
                        const afterNode = textNode.splitText(offset);
                        range.setStartBefore(afterNode);
                        range.setEndBefore(afterNode);
                    }

                    const lastNode = fragment.lastChild;
                    range.insertNode(fragment); // Insert fragment.

                    // Move caret right after inserted node.
                    range.setStartAfter(lastNode);
                    range.collapse(true);
                    selection.removeAllRanges();
                    selection.addRange(range);
                } else {
                    // Find textarea element.
                    const textAreaEl = Array.from(anchorNode.childNodes).find(node => node instanceof HTMLTextAreaElement);

                    // Insert text after caret/selection.
                    const selectionEndIndex = textAreaEl.selectionEnd;
                    const text = `[img no-zoom width=${insertWidthInputEl.value}]${imgSrc}[/img]`;
                    textAreaEl.value = textAreaEl.value.slice(0, selectionEndIndex) + text + textAreaEl.value.slice(selectionEndIndex);

                    // Move caret after inserted text.
                    textAreaEl.selectionStart = textAreaEl.selectionEnd = selectionEndIndex + text.length;

                    // Trigger input event.
                    textAreaEl.dispatchEvent(new Event('input', { bubbles: true }));
                }
            }

            // Output image Tenor page URL into console in case you want to open it and add it to your library. TODO: Make "Add to library" button and use modifier key to add or to reveal the add button overlay?
            if (tenorPageUrl) {
                console.info(tenorPageUrl);
            }
        }, 0);
    }

    const isSelectionInsideContentEditable = (selection) => {
        if (!selection.rangeCount) return false;

        let node = selection.anchorNode;
        while (node) {
            if (node.nodeType === Node.ELEMENT_NODE && node.isContentEditable) {
                return true;
            }

            node = node.parentNode;
        }

        return false;
    }

    const goToPopularImagesScreen = () => {
        // Hide load more button and disable scroll/pull to load more function while popular images section is displaying.
        setPullToLoadMoreListenerState(false);

        // Cancel the search request if any.
        abortCurrentSearchRequest("Search query has been cancelled as it's no longer relevant.");

        // Clear search query input if not empty.
        if (searchInputEl.value) searchInputEl.value = '';

        // Reset the search query parameters memory because the new query after displaying popular images is allowed to be the same.
        prevQueryDynamicPart = prevQueryDynamicPartWithPos = null;

        // Clear search results.
        clearSearchResults();

        // Display popular images section.
        setPopularImagesDisplay(true);
    }

    const clearSearchResults = () => {
        for (const img of resultsContainerEl.querySelectorAll('img')) {
            img.removeEventListener('click', onSearchResultImageClick); // Just in case...
            img.removeEventListener('load', onImageLoad);
        }

        resultsContainerEl.innerHTML = '';
    }

    const onDialogOpenButtonClick = (popupAnchorEl) => {
        if (popupContainerEl.parentElement === popupAnchorEl) {
            // The popup element is already attached to the specified popup anchor element, just togghe the popup visibility then.
            popupContainerEl.style.display = getComputedStyle(popupContainerEl).display === 'none' ? 'block' : 'none';
        } else {
            // Append existing popup to the specified anchor element and show it.
            popupAnchorEl.appendChild(popupContainerEl);
            popupContainerEl.style.display = 'block';

            // Store the last anchor element where the popup was attached, so we can later find the corresponding editor element relative to it.
            currentAnchorEl = popupAnchorEl;
        }

        // If there are not enough space for the popup - make the temporary space.
        const elementToEnlarge = document.querySelector('#content') || document.body;
        if (popupContainerEl.style.display !== 'none') {
            const headerCalculatedHeightPx = document.querySelector('#menu')?.getBoundingClientRect().bottom || 0;
            const spaceAboveTheAnchorPx = popupAnchorEl.getBoundingClientRect().top + window.scrollY - headerCalculatedHeightPx;
            const popupTopOffsetPx = parseInt(getComputedStyle(document.documentElement).getPropertyValue('--malgi-popup-height'));

            let actualPopupSpacingPx;
            switch (popupAnchorEl.dataset.spacingMode) {
                case popupSpacingModes.sceditor:
                    actualPopupSpacingPx = popupSpacingInScEditorPx;

                    break;

                case popupSpacingModes.table:
                    actualPopupSpacingPx = popupSpacingInTablePx;

                    break;

                case popupSpacingModes.default:
                    actualPopupSpacingPx = popupSpacingPx;

                    break;

                default:
                    console.warn(`Unprocessed popup spacing mode: ${popupAnchorEl.dataset.spacingMode}.`);

                    break;
            }
            document.documentElement.style.setProperty('--malgi-popup-spacing', `${actualPopupSpacingPx}px`); // Triggers the popup animation and moves it to the height corresponding to the editor type.

            const deltaPx = popupTopOffsetPx + actualPopupSpacingPx - spaceAboveTheAnchorPx; // How much space is lacking to display the whole popup.
            if (deltaPx > 0) {
                const enlargementBeautifyDeltaPx = 14; // Just looks better with some additional spacing.
                document.documentElement.style.setProperty('--malgi-popup-extra-height', `${Math.ceil(deltaPx + enlargementBeautifyDeltaPx)}px`);
                elementToEnlarge.classList.add('malgi-popup-extra-height');
            } else {
                elementToEnlarge.classList.remove('malgi-popup-extra-height');
            }

            // If the popup element is visible then focus the search input element.
            searchInputEl.focus();
        } else {
            document.documentElement.style.setProperty('--malgi-popup-spacing', '0px'); // Resets the popup position to prepare for its show animation.
            elementToEnlarge.classList.remove('malgi-popup-extra-height');
        }
    }

    const onSearchParamsChange = () => {
        if (searchInputEl.value) handleSearch();
        else searchInputEl.focus();
    }

    const onSearchResultImageClick = (event) => {
        const imgEl = event.currentTarget;

        handleImageSelection(getEditorEl(), imgEl.src, imgEl.dataset.tenorPageUrl);

        savePopularImageAndSync(imgEl);
    }

    const onImageLoad = (event) => {
        event.currentTarget.classList.add('loaded');
    }

    const onPopularImageClick = event => {
        const img = event.currentTarget;
        const now = Date.now();
        const lastClickTime = Number(img.dataset.lastClickTime) || 0;
        let count = Number(img.dataset.clickCount) || 0;

        if (now - lastClickTime < popularImageClickInterval) {
            ++count;
        } else {
            count = 1;
        }

        img.dataset.clickCount = count;
        img.dataset.lastClickTime = now;

        if (count < 2) {
            // First click on image should be handled as image selection.
            handleImageSelection(getEditorEl(), img.src, img.dataset.tenorPageUrl);
        } else if (count >= 3) {
            // Triple click removes the image from popular images.
            img.removeEventListener('click', onPopularImageClick); // Just in case.
            img.removeEventListener('load', onImageLoad);
            img.remove();
            img.dataset.clickCount = 0;

            removePopularImageAndSync(img);
        }
    };

    let startY = null;
    // Desktop drag or touch.
    const onPointerDown = event => { startY = event.clientY; }
    const onPointerUp = event => {
        if (startY !== null && startY - event.clientY > loadMoreTriggerDistancePx) loadMoreButtonEl.click();
        startY = null;
    };
    // Mobile.
    const onTouchStart = event => { startY = event.touches[0].clientY; }
    const onTouchEnd = event => {
        if (startY !== null && startY - event.changedTouches[0].clientY > loadMoreTriggerDistancePx) loadMoreButtonEl.click();
        startY = null;
    };
    // Mouse scroll for desktop.
    const onScroll = event => {
        if (popupContainerEl.scrollTop + popupContainerEl.clientHeight >= popupContainerEl.scrollHeight - loadMoreTriggerDistancePx) {
            loadMoreButtonEl.click();
        }
    };
    // Mouse wheel (for case if there is nothing to scroll yet) for desktop.
    const onWheel = event => {
        if (popupContainerEl.scrollHeight <= popupContainerEl.clientHeight && event.deltaY > 0) loadMoreButtonEl.click();
    };
    const setPullToLoadMoreListenerState = (state) => {
        if (state) {
            loadMoreButtonEl.style.display = 'block';

            popupContainerEl.addEventListener('pointerdown', onPointerDown);
            popupContainerEl.addEventListener('pointerup', onPointerUp);
            popupContainerEl.addEventListener('touchstart', onTouchStart);
            popupContainerEl.addEventListener('touchend', onTouchEnd);
            popupContainerEl.addEventListener('scroll', onScroll);
            popupContainerEl.addEventListener('wheel', onWheel);
        } else {
            popupContainerEl.removeEventListener('pointerdown', onPointerDown);
            popupContainerEl.removeEventListener('pointerup', onPointerUp);
            popupContainerEl.removeEventListener('touchstart', onTouchStart);
            popupContainerEl.removeEventListener('touchend', onTouchEnd);
            popupContainerEl.removeEventListener('scroll', onScroll);
            popupContainerEl.removeEventListener('wheel', onWheel);

            loadMoreButtonEl.style.display = 'none';
        }
    }

    window.addEventListener('storage', e => {
        if (e.key === popularImagesLocalStorageKey) {
            // TODO: Handle storage event? It can cause possible races if more than 3 tabs with popular images changes are opened.
            // const newImages = JSON.parse(e.newValue || '[]');
            // renderImages(newImages);
        }
    });

    const getStoredPopularImages = () => {
        return JSON.parse(localStorage.getItem(popularImagesLocalStorageKey)) || [];
    }
    const restorePopularImages = () => {
        if (!popularImagesContainerEl) throw new Error('Popular images container is not found.');

        const initialPopularImages = getStoredPopularImages();
        for (let i = initialPopularImages.length - 1; i >= 0; --i) {
            const popularImage = initialPopularImages[i];
            addPopularImageEl(popularImage.url, popularImage.tenorPageUrl);
        }

        // If current popular image count limit reached, remove excessive images from local storage and update the storage.
        if (initialPopularImages.length > maxPopularImagesCount) {
            storePopularImages(initialPopularImages);
        }
    }
    const storePopularImages = (popularImages) => {
        // Remove excessive popular images that may have been added in another tab and synced.
        popularImages.length = Math.min(popularImages.length, maxPopularImagesCount);

        localStorage.setItem(popularImagesLocalStorageKey, JSON.stringify(popularImages));
    }
    // TODO: Optimize popular images code. Iterate through Maps, update popular images container once by using Fragment.
    const savePopularImageAndSync = (imgEl) => {
        const initialPopularImageElsArray = getPopularImageEls();
        const storedPopularImages = getStoredPopularImages();
        let isFoundInStored = false,
            isFoundInInitial = false;

        // For performance purposes.
        const initialPopularImageElsMap = Object.fromEntries(initialPopularImageElsArray.map(popularImageEl => [popularImageEl.src, popularImageEl.dataset.tenorPageUrl]));
        const storedPopularImagesMap = Object.fromEntries(storedPopularImages.map(popularImage => [popularImage.url, popularImage]));

        for(const popularImage of storedPopularImages) {
            if (popularImage.url == imgEl.src) {
                isFoundInStored = true;
            } else {
                // In case any new popular images were added in another tab, add them here too so the user doesn't have to reload the page to get them.
                if (!initialPopularImageElsMap[popularImage.url]) {
                    addPopularImageEl(popularImage.url, popularImage.tenorPageUrl);
                }
            }
        }

        for (let i = initialPopularImageElsArray.length - 1; i >= 0; i--) {
            const popularImageEl = initialPopularImageElsArray[i];

            if (popularImageEl.src == imgEl.src) {
                isFoundInInitial = true;
            } else {
                // In case any new popular images were removed in another tab, remove them here too so the user doesn't have to reload the page to get rid of them.
                if (!storedPopularImagesMap[popularImageEl.src]) {
                    removePopularImageEl(popularImageEl);
                }
            }
        }

        if (!isFoundInInitial || !isFoundInStored) {
            const newPopularImage = { url: imgEl.src, tenorPageUrl: imgEl.dataset.tenorPageUrl };

            if (!isFoundInInitial) {
                addPopularImageEl(newPopularImage.url, newPopularImage.tenorPageUrl);
            }
            if (!isFoundInStored) {
                storedPopularImages.unshift(newPopularImage);
                storePopularImages(storedPopularImages);
            }
        }
    }
    const removePopularImageAndSync = (imgEl) => {
        const initialPopularImageElsArray = getPopularImageEls();
        const storedPopularImages = getStoredPopularImages();
        let isFoundInStored = false;

        // For performance purposes.
        const initialPopularImageElsMap = Object.fromEntries(initialPopularImageElsArray.map(popularImageEl => [popularImageEl.src, popularImageEl.dataset.tenorPageUrl]));
        const storedPopularImagesMap = Object.fromEntries(storedPopularImages.map(popularImage => [popularImage.url, popularImage]));

        const popularImagesAddedInParallel = [];
        for (let i = storedPopularImages.length - 1; i >= 0; --i) {
            const popularImage = storedPopularImages[i];

            // Remove specified image from stored popular images.
            if (popularImage.url == imgEl.src) {
                storedPopularImages.splice(i, 1);
                storePopularImages(storedPopularImages);
            } else {
                // In case any new popular images were added in another tab, add them here too so the user doesn't have to reload the page to get them.
                if (!initialPopularImageElsMap[popularImage.url]) {
                    popularImagesAddedInParallel.unshift(popularImage);
                }
            }
        }
        // Add popular images added in another tab.
        if (popularImagesAddedInParallel.length) {
            for(const popularImage of popularImagesAddedInParallel) {
                addPopularImageEl(popularImage.url, popularImage.tenorPageUrl);
            }
        }

        for (let i = initialPopularImageElsArray.length - 1; i >= 0; --i) {
            const popularImageEl = initialPopularImageElsArray[i];

            // Remove specified image from initial popular images.
            if (popularImageEl.src == imgEl.src) {
                removePopularImageEl(popularImageEl);
            } else {
                // In case any new popular images were removed in another tab, remove them here too so the user doesn't have to reload the page to get rid of them.
                if (!storedPopularImagesMap[popularImageEl.src]) {
                    removePopularImageEl(popularImageEl);
                }
            }
        }
    }
    const addPopularImageEl = (url, tenorPageUrl) => {
        const imgEl = document.createElement('img');
        imgEl.className = 'malgi-popular-img';
        imgEl.src = url;
        imgEl.dataset.tenorPageUrl = tenorPageUrl;
        imgEl.addEventListener('click', onPopularImageClick);
        imgEl.addEventListener('load', onImageLoad);

        // Remove excessive popular image in case limit is reached.
        if (popularImagesContainerEl.childElementCount >= maxPopularImagesCount) popularImagesContainerEl.lastElementChild?.remove();

        popularImagesContainerEl.insertBefore(imgEl, popularImagesContainerEl.firstChild);
    }
    const removePopularImageEl = (popularImageEl) => {
        popularImageEl.remove();
    }

    const restoreOptions = () => {
        insertWidthDefaultPx = JSON.parse(localStorage.getItem(insertOptionWidthLocalStorageKey)) || insertWidthDefaultPx;
        displayOptionSquareDefault = JSON.parse(localStorage.getItem(displayOptionSquareLocalStorageKey)) ?? displayOptionSquareDefault;
    }

    const setPopularImagesDisplay = (display) => {
        popularImagesContainerEl.style.display = display ? 'grid' : 'none';
    }

    const setObjectFitCover = (isCoverMode) => {
        if (isCoverMode) imagesContainerEl.classList.remove('malgi-object-fit-contain');
        else imagesContainerEl.classList.add('malgi-object-fit-contain');

        localStorage.setItem(displayOptionSquareLocalStorageKey, isCoverMode);
    }

    const getPopularImageEls = () => {
        return Array.from(popularImagesContainerEl.querySelectorAll('img'));
    }

    const getResultLimit = () => {
        if (!resultsContainerEl) throw new Error('Results container element is not found.');

        return Math.floor((resultsContainerEl.clientWidth + imageGridGapPx) / (imageGridMinWidthPx + imageGridGapPx)) * searchQueryRowsCount;
    }

    const setLoadingState = (isLoading) => {
        if (isLoading) {
            searchButtonEl.disabled = true;
            searchButtonEl.textContent = '⏳';
            loadMoreButtonEl.classList.add('is-loading');

        } else {
            searchButtonEl.textContent = '🔎';
            loadMoreButtonEl.classList.remove('is-loading');
            searchButtonEl.disabled = false;
        }
    }

    let prevQueryDynamicPart, prevQueryDynamicPartWithPos;
    let nextPos = null;
    const searchTenor = (searchQuery, loadMore = false) => {
        const limit = getResultLimit();

        const queryDynamicPart = `&q=${searchQueryPrefixSelectEl.value}${searchQuery}&searchfilter=${searchFilterSelectEl.value}&limit=${limit}`;
        // Tenor API v2 is weird. If any dynamic query part is changed then it's a new query and pos parameter value must be nullified.
        if (queryDynamicPart != prevQueryDynamicPart) {
            nextPos = null;
        } else {
            // If the user presses the search button again then treat is as load more request.
            loadMore = true;
        }
        prevQueryDynamicPart = queryDynamicPart;

        const queryDynamicPartWithPos = `${queryDynamicPart}&pos=${nextPos}`;
        // Prevent same query spam.
        if (queryDynamicPartWithPos == prevQueryDynamicPartWithPos) {
            return;
        }
        prevQueryDynamicPartWithPos = queryDynamicPartWithPos;

        // If there is already a current request - cancel it.
        abortCurrentSearchRequest('Replacing with a new request');

        if (!loadMore) {
            clearSearchResults();
        }

        searchRequestAbortController = new AbortController();

        setLoadingState(true);

        fetch(`https://tenor.googleapis.com/v2/search?key=${apiKey}&client_key=${clientKey}&contentfilter=low${queryDynamicPartWithPos}`, { signal: searchRequestAbortController.signal })
            .then(r => r.json())
            .then(data => {
            data.results.forEach(responseObject => {
                const img = document.createElement('img');
                img.className = 'search-result-img';
                img.src = responseObject.media_formats?.gif_transparent?.url || responseObject.media_formats.gif.url; // If there is transparent version - use it.
                img.dataset.tenorPageUrl = responseObject.url;
                img.alt = responseObject.title || responseObject.content_description;
                img.addEventListener('click', onSearchResultImageClick);
                img.addEventListener('load', onImageLoad);
                resultsContainerEl.appendChild(img);
            });

            nextPos = data.next;

            const hasResult = data.results.length > 0;
            setPullToLoadMoreListenerState(hasResult);
            setPopularImagesDisplay(!hasResult);
        })
            .catch(e => console.error(e))
            .finally(() => setLoadingState(false));
    }

    const abortCurrentSearchRequest = (reason) => {
        searchRequestAbortController?.abort(reason);
    }

    // Hide the popup after the corresponding form submission.
    document.addEventListener('submit', e => {
        const editorEl = getEditorEl();
        if (editorEl) {
            const closestForm = editorEl.closest('form');
            if (closestForm && e.target === closestForm) {
                popupContainerEl.style.display = 'none';
            }
        }
    }, true);

    // Restore options from local storage during the init phase, before any usage.
    restoreOptions();

    const init = (injectionTargets) => {
        initButtonAndAnchorForEveryEditor(injectionTargets);

        if (!window.malgiBaseInitializationComplete) {
            initPopupElements();
            restorePopularImages();
            window.malgiBaseInitializationComplete = true;
        }
    }

    const debounce = (fn, wait) => {
        let t;

        return (injectionTargets) => {
            clearTimeout(t);
            t = setTimeout(fn(injectionTargets), wait);
        };
    };
    const di = debounce(init, 200);

    const isVisible = (el) => {
        const st = getComputedStyle(el);

        return (
            el.offsetParent !== null &&
            st.display !== 'none' &&
            st.visibility !== 'hidden' &&
            st.opacity !== '0'
        );
    }

    const getInjectionTargets = () => {
        return [...document.querySelectorAll('textarea:not(.g-recaptcha-response)')].filter(el => isVisible(el));
    }

    const checkTargets = () => {
        const targets = getInjectionTargets();
        if (targets.length) {
            di(targets);
        }
    }

    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', checkTargets, { once: true });
    } else {
        checkTargets();
    }

    // Look for every new toolbar/menubar target for script button injection.
    const mo = new MutationObserver(checkTargets);
    // Start observing the body in case new editor elements appear on the page.
    mo.observe(document.documentElement, { childList: true, subtree: true }); // Only observing document.documentElement allows tracking all editor appearances during SPA navigation.

    const fallbackTargetsCheckInterval = setInterval(checkTargets, 3000);
    window.addEventListener('beforeunload', () => clearInterval(fallbackTargetsCheckInterval));

    const _wrap = m => {
        const orig = history[m];
        return (...args) => {
            const ret = orig.apply(history, args);
            window.dispatchEvent(new Event('locationchange'));

            return ret;
        };
    };
    history.pushState = _wrap('pushState');
    history.replaceState = _wrap('replaceState');
    window.addEventListener('popstate', () => window.dispatchEvent(new Event('locationchange')));
    window.addEventListener('locationchange', () => {
        if (popupContainerEl) popupContainerEl.style.display = 'none'; // Hide the popup on other page navigation.
        checkTargets();
    }, { passive: true });
    window.addEventListener('pageshow', checkTargets, { passive: true });
    document.addEventListener('visibilitychange', () => { if (!document.hidden) checkTargets(); }, { passive: true });
})();