Come to Anime moments club 🎊 https://myanimelist.net/clubs.php?cid=93838 🎉! Add convenient Tenor.com gif image inserter into MyAnimeList.net comment editor.
// ==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="{"id":null,"src":"${imgSrc}","isPoster":false,"width":${insertWidthInputEl.value},"height":null,"isNoZoom":true,"class":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 });
})();