Twitter XFilter panel 2.4.67 (c) tapeavion

Hide posts by keywords with the dashboard and posts from verified accounts

// ==UserScript==
// @name         Twitter XFilter panel 2.4.67 (c) tapeavion 
// @version      2.4.67
// @description  Hide posts by keywords with the dashboard and posts from verified accounts
// @author       gullampis810
// @match        https://x.com/*
// @match        https://x.com/i/grok*
// @match        https://blank.org/*
// @grant        GM_addStyle
// @run-at       document-end
// @license      MIT 
// @icon         https://www.pinclipart.com/picdir/big/450-4507608_twitter-circle-clipart.png
// @namespace http://tampermonkey.net/
// ==/UserScript==
 
(function () {
    "use strict";

    // ========== Настройки и инициализация ========== //
    const STORAGE_KEY = "hiddenKeywords";
    const FAVORITE_USERS_KEY = "favoriteUsers";
    const BLOCKED_USERS_KEY = "blockedUsers";
    // NEW: Новый ключ для черного списка GIF
    const GIF_BLACKLIST_KEY = "gifBlacklist";
    let hiddenKeywords = JSON.parse(localStorage.getItem(STORAGE_KEY)) || [];
    let favoriteUsers = JSON.parse(localStorage.getItem(FAVORITE_USERS_KEY)) || [];
    let blockedUsers = JSON.parse(localStorage.getItem(BLOCKED_USERS_KEY)) || [];
    let unblockedKeywords = JSON.parse(localStorage.getItem("unblockedKeywords")) || [];
    // NEW: Инициализация черного списка для GIF
    // Глобальные переменные (добавьте в начало скрипта, если их нет)
// Глобальные переменные (добавьте это вне функции, в начале скрипта)
let filteredData = []; // Полный отфильтрованный массив данных
let itemHeight = 40; // Высота одного элемента (в пикселях, соответствует CSS)
let viewportHeight = 400; // Высота viewport (#keywordList)
let maxItems = 10; // Максимальное количество элементов для рендера
    let gifBlacklist = JSON.parse(localStorage.getItem(GIF_BLACKLIST_KEY)) || [];
    let hideVerifiedAccounts = false;
    let hideNonVerifiedAccounts = JSON.parse(localStorage.getItem("hideNonVerifiedAccounts")) || false;
    let displayMode = "keywords"; // keywords, favorites, blocked
    const languageFilters = {
        english: /[a-zA-Z]/,
        russian: /[А-Яа-яЁё]/,
        japanese: /[\p{Script=Hiragana}\p{Script=Katakana}\p{Script=Han}]/u,
        ukrainian: /[А-Яа-яІіЄєЇїҐґ]/,
        belarusian: /[А-Яа-яЎўЁёІі]/,
        tatar: /[А-Яа-яӘәӨөҮүҖҗ]/,
        mongolian: /[\p{Script=Mongolian}]/u,
        chinese: /[\p{Script=Han}]/u,
        german: /[a-zA-ZßÄäÖöÜü]/,
        polish: /[a-zA-ZąćęłńóśźżĄĆĘŁŃÓŚŹŻ]/,
        french: /[a-zA-Zàâçéèêëîïôûùüÿ]/,
        swedish: /[a-zA-ZåäöÅÄÖ]/,
        estonian: /[a-zA-ZäõöüÄÕÖÜ]/,
        danish: /[a-zA-Z帿ŨÆ]/,
        turkish: /[a-zA-ZıİçÇğĞöÖşŞüÜ]/,
        portuguese: /[a-zA-Zàáâãçéêíóôõúü]/,
    };
    let activeLanguageFilters = {};

    // =============== showNotification ================ //
    function showNotification(message, type = 'info') {
        const notification = document.createElement('div');
        notification.style.position = 'fixed';
        notification.style.top = '20px';
        notification.style.right = '20px';
        notification.style.padding = '10px 20px';
        notification.style.borderRadius = '8px';
        notification.style.zIndex = '10001';
        notification.style.color = 'white';
        notification.style.fontFamily = 'Arial, sans-serif';

        switch(type) {
            case 'success':
                notification.style.background = 'linear-gradient(90deg,rgb(31, 90, 33), #81C784)';
                break;
            case 'error':
                notification.style.background = 'linear-gradient(90deg,rgb(94, 41, 38),rgb(212, 134, 121))';
                break;
            default:
                notification.style.background = 'linear-gradient(90deg,rgb(33, 70, 100), #64B5F6)';
        }
        notification.textContent = message;
        document.body.appendChild(notification);

        setTimeout(() => {
            notification.style.transition = 'opacity 0.5s';
            notification.style.opacity = '0';
            setTimeout(() => {
                notification.remove();
            }, 500);
        }, 2000);
    }

    // Получение имени пользователя из профиля
    function getUsernameFromProfile() {
        const profileUrl = window.location.href;
        const match = profileUrl.match(/x\.com\/([^\?\/]+)/);
        if (match && match[1]) {
            return match[1].toLowerCase();
        }
        const usernameElement = document.querySelector('a[href*="/"][role="link"] div[dir="ltr"] span');
        return usernameElement ? usernameElement.textContent.replace('@', '').toLowerCase() : '';
    }

    // ========== Сохранение в localStorage ========== //
    function saveKeywords() {
        localStorage.setItem(STORAGE_KEY, JSON.stringify(hiddenKeywords));
    }

    function saveFavoriteUsers() {
        localStorage.setItem(FAVORITE_USERS_KEY, JSON.stringify(favoriteUsers));
    }

    function saveBlockedUsers() {
        localStorage.setItem(BLOCKED_USERS_KEY, JSON.stringify(blockedUsers));
    }

    function saveUnblockedKeywords() {
        localStorage.setItem("unblockedKeywords", JSON.stringify(unblockedKeywords));
    }

    // NEW: Сохранение черного списка GIF
    function saveGifBlacklist() {
        localStorage.setItem(GIF_BLACKLIST_KEY, JSON.stringify(gifBlacklist));
    }

    // ========== Функция для обновления фильтров по языкам ========== //
    function updateLanguageFilter(language) {
        if (activeLanguageFilters[language]) {
            delete activeLanguageFilters[language];
        } else {
            activeLanguageFilters[language] = languageFilters[language];
        }
        hidePosts();
    }

    // ========== Проверка языка текста ========== //
    function isTextInLanguage(text) {
        for (const [language, regex] of Object.entries(activeLanguageFilters)) {
            if (regex.test(text)) {
                return true;
            }
        }
        return false;
    }

    // ========== Функция скрытия постов ========== //
    function hidePosts() {
    document.querySelectorAll("article").forEach((article) => {
        const textContent = article.innerText.toLowerCase();
        const isVerifiedAccount = article.querySelector('[data-testid="icon-verified"]');
        const usernameElement = article.querySelector('a[href*="/"]');
        const usernameFromSpan = article.querySelector('a[href*="/"] div[dir="ltr"] span')?.textContent.replace('@', '').toLowerCase() || '';
        const username = usernameElement ? usernameElement.getAttribute("href").slice(1).toLowerCase() : usernameFromSpan;

        // Проверяем, является ли пользователь избранным
        const isFavoriteUser = favoriteUsers.includes(username);
        // Проверяем, является ли пользователь заблокированным
        const isBlockedUser = blockedUsers.includes(username);

        // Если пользователь в избранных, показываем пост
        if (isFavoriteUser) {
            article.style.display = "";
            // NEW: Добавляем элементы управления для GIF
            addGifControls(article);
            return;
        }

        // Если пользователь заблокирован, скрываем пост
        if (isBlockedUser) {
            article.style.display = "none";
            return;
        }

        // Проверка на ключевые слова
        const matchesKeyword = hiddenKeywords.some((keyword) => {
            try {
                return new RegExp(keyword, "i").test(textContent);
            } catch (e) {
                return textContent.includes(keyword.toLowerCase());
            }
        });

        // Проверка на запрещенные GIF
        let matchesGif = false;
        const gifElements = article.querySelectorAll('video[aria-label], img[alt="GIF"]');
        gifElements.forEach((gif) => {
            const gifLabel = gif.getAttribute('aria-label')?.toLowerCase() || '';
            const gifSrc = gif.getAttribute('src')?.toLowerCase() || '';
            const gifPoster = gif.getAttribute('poster')?.toLowerCase() || '';
            const gifContent = gifLabel + gifSrc + gifPoster;
            matchesGif = gifBlacklist.some((gifKeyword) => gifContent.includes(gifKeyword.toLowerCase()));
            if (matchesGif) return;
        });

        // Проверка на посты без текста (только GIF)
        const hasOnlyGif = gifElements.length > 0 &&
                           !article.querySelector('[data-testid="tweetText"]') &&
                           textContent.trim().replace(/[@\w]+|\d+\s?[чмдн]/gi, '').trim().length < 10;

        // Логика скрытия постов
        const shouldHideVerified = hideVerifiedAccounts && isVerifiedAccount;
        const shouldHideNonVerified = hideNonVerifiedAccounts && !isVerifiedAccount;
        const shouldHideByLanguage = isTextInLanguage(textContent);
        const shouldHideByKeyword = matchesKeyword;
        const shouldHideByGif = matchesGif || hasOnlyGif;

        if (shouldHideVerified || shouldHideNonVerified || shouldHideByLanguage || shouldHideByKeyword || shouldHideByGif) {
            article.style.display = "none";
        } else {
            article.style.display = "";
            // NEW: Добавляем элементы управления для GIF, если пост не скрыт
            addGifControls(article);
        }
    });
}

// NEW: Функция для добавления элементов управления GIF
function addGifControls(article) {
    const gifElements = article.querySelectorAll('video[aria-label]');
    gifElements.forEach((gif) => {
        // Проверяем, существует ли уже контейнер управления
        if (gif.parentElement.querySelector('.gif-control-container')) {
            return; // Пропускаем, если контейнер уже есть
        }

        const gifLabel = gif.getAttribute('aria-label') || 'GIF';
        if (!gifLabel) return; // Пропускаем, если нет aria-label

        // Создаем контейнер для элементов управления
        const container = document.createElement('div');
        container.className = 'gif-control-container';

        // Добавляем текст с названием GIF
        const label = document.createElement('span');
        label.className = 'gif-label';
        label.textContent = gifLabel;
        container.appendChild(label);

        // Добавляем кнопку "Копировать"
        const copyButton = document.createElement('button');
        copyButton.textContent = 'copy gif name'; // копировать
        copyButton.addEventListener('click', () => {
            navigator.clipboard.writeText(gifLabel).then(() => {
                showNotification(`Скопировано: ${gifLabel}`, 'success');
            }).catch(() => {
                showNotification('Ошибка копирования', 'error');
            });
        });
        container.appendChild(copyButton);

        // Вставляем контейнер в родительский элемент видео
        gif.parentElement.style.position = 'relative';
        gif.parentElement.appendChild(container);
    });
}

    // ========== Debounce для оптимизации ========== //
    function debounce(func, wait) {
        let timeout;
        return function (...args) {
            clearTimeout(timeout);
            timeout = setTimeout(() => func.apply(this, args), wait);
        };
    }

    const debouncedHidePosts = debounce(hidePosts, 200);

    // ========== Создание панели управления ========== //
    const panel = document.createElement("div");
    panel.id = "xpanelxfilter-xai-he5h5eh-controlPanel";  // Здесь задаем ID для панели (можно использовать для CSS или querySelector)
     
    panel.style.position = "fixed";
    panel.style.left = "1015px"; // Заменяем right на left
    panel.style.top = "89px"; // Заменяем bottom на top
    panel.style.width = "335px";
    panel.style.height = "710px";
    panel.style.padding = "8px";
    panel.style.fontFamily = "Arial, sans-serif";
    panel.style.backgroundColor = "#34506c";
    panel.style.color = "#fff";
    panel.style.borderRadius = "8px";
    panel.style.boxShadow = "0 0 10px rgba(0,0,0,0.5)";
    panel.style.zIndex = "9999";
    panel.style.overflow = "auto";
    panel.style.transition = "height 0.3s ease"; 

    // Принудительное применение стилей скроллбара для панели
    panel.style.scrollbarWidth = 'auto';
    panel.style.scrollbarColor = '#c1a5ef #205151';

    // NEW: Добавляем счетчик для GIF в панель
    panel.innerHTML = `
 <h3 style="
     margin: 0; 
    font-size: 16px;
 " id="panelHeader"> XFilter </h3>
  <div style="
    display: flex;
    align-items: center;
    gap: 20px;
    top: 20px;
    position: relative;
    width: 250px;
    left: 10px;
">
  <span id="counter-keywords" class="counter-span">
   <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 16 16">
   <title>File-earmark-text-fill SVG Icon</title>
   <path fill="currentColor" d="M9.293 0H4a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2V4.707A1 1 0 0 0 13.707 4L10 .293A1 1 0 0 0 9.293 0M9.5 3.5v-2l3 3h-2a1 1 0 0 1-1-1M4.5 9a.5.5 0 0 1 0-1h7a.5.5 0 0 1 0 1zM4 10.5a.5.5 0 0 1 .5-.5h7a.5.5 0 0 1 0 1h-7a.5.5 0 0 1-.5-.5m.5 2.5a.5.5 0 0 1 0-1h4a.5.5 0 0 1 0 1z"/>
   </svg>
    <span class="counter-number">0</span>
    <span class="counter-tooltip">Keywords</span>
  </span>

  <span id="counter-favorites" class="counter-span">
  <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 32 32">
  <title>User-favorite SVG Icon</title>
    <path fill="currentColor" d="M27.303 12a2.662 2.662 0 0 0-1.908.806l-.393.405l-.397-.405a2.662 2.662 0 0 0-3.816 0a2.8 2.8 0 0 0 0 3.896L25.002 21l4.209-4.298a2.8 2.8 0 0 0 0-3.896A2.662 2.662 0 0 0 27.303 12M2 30h2v-5a5.006 5.006 0 0 1 5-5h6a5.006 5.006 0 0 1 5 5v5h2v-5a7.008 7.008 0 0 0-7-7H9a7.008 7.008 0 0 0-7 7zM12 4a5 5 0 1 1-5 5a5 5 0 0 1 5-5m0-2a7 7 0 1 0 7 7a7 7 0 0 0-7-7"/>
  </svg>
    <span class="counter-number">0</span>
    <span class="counter-tooltip">Favorite Users</span>
  </span>

  <span id="counter-blocked" class="counter-span">
    <svg class="counter-icon" xmlns="http://www.w3.org/2000/svg" width="24" height="24"
         viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"
         stroke-linecap="round" stroke-linejoin="round">
      <path d="M16 21v-2a4 4 0 0 0-4-4H5a4 4 0 0 0-4 4v2"/>
      <circle cx="8.5" cy="7" r="4"/>
      <line x1="18" y1="8" x2="22" y2="12"/>
      <line x1="22" y1="8" x2="18" y2="12"/>
    </svg>
    <span class="counter-number">0</span>
    <span class="counter-tooltip">Blocked Users</span>
  </span>

  <span id="counter-gif" class="counter-span">
    <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 16 16">
    <title>Filetype-gif SVG Icon</title>
    <path fill="currentColor" fill-rule="evenodd" d="M14 4.5V14a2 2 0 0 1-2 2H9v-1h3a1 1 0 0 0 1-1V4.5h-2A1.5 1.5 0 0 1 9.5 3V1H4a1 1 0 0 0-1 1v9H2V2a2 2 0 0 1 2-2h5.5zM3.278 13.124a1.4 1.4 0 0 0-.14-.492a1.3 1.3 0 0 0-.314-.407a1.5 1.5 0 0 0-.48-.275a1.9 1.9 0 0 0-.636-.1q-.542 0-.926.229a1.5 1.5 0 0 0-.583.632a2.1 2.1 0 0 0-.199.95v.506q0 .408.105.745q.105.336.32.58q.213.243.533.377q.323.132.753.132q.402 0 .697-.111a1.29 1.29 0 0 0 .788-.77q.097-.261.097-.551v-.797H1.717v.589h.823v.255q0 .199-.09.363a.67.67 0 0 1-.273.264a1 1 0 0 1-.457.096a.87.87 0 0 1-.519-.146a.9.9 0 0 1-.305-.413a1.8 1.8 0 0 1-.096-.615v-.499q0-.547.234-.85q.237-.3.665-.301a1 1 0 0 1 .3.044q.136.044.236.126a.7.7 0 0 1 .17.19a.8.8 0 0 1 .097.25zm1.353 2.801v-3.999H3.84v4h.79Zm1.493-1.59v1.59h-.791v-3.999H7.88v.653H6.124v1.117h1.605v.638z"/>
    </svg>
    <span class="counter-number">0</span>
    <span class="counter-tooltip">GIF Blacklist</span>
  </span>
</div>

 <h4 class="version-text">v2.4.67</h4>
 <div style="display: flex;
       align-items: center;
       gap: 5px;
       margin: 10px 0;
      ">
 <input id="keywordInput" type="text" placeholder="Enter the word, @username or gif:keyword" style="
    width: calc(100% - 95px);
    height: 35px;
    padding: 0px;
    border-radius: 5px;
    border: none;
    background: #15202b;
    color: #fff;
      ">
 <button id="addKeyword" style="
    min-width: 75px;
    max-width: 80px;
    background: #203142;
    color: #a9b6c4;
    border: none;
    border-radius: 5px;
    height: 35px;
    font-size: 20px;
      ">Add it</button>
 </div>
 <div style="display: flex;
       flex-wrap: wrap;
       gap: 5px;
       position: relative;
      ">
 <div style="display: flex;
       align-items: center;
       gap: 5px;
      ">
 <div id="searchWrapper" style="position: relative;
      ">
 <input id="searchInput" type="text" placeholder="Search keywords or users" style="width: 240px;
    width: 235px;
    height: 35px;
    border-radius: 5px;
    border: none;
    background-color: #15202b;
    color: #fff;
      ">
 <span id="clearSearch" style="display: none;
       position: absolute;
       right: 10px;
       top: 50%;
       transform: translateY(-50%);
       color: #fff;
       cursor: pointer;
      "><svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg></span>
  </div>
 <button id="openLanguagePopup" style="
       width: 75px; 
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 35px;
       font-size: 14px;
       font-weight: bold;
      ">Language Filtering</button>
 </div>
 <button id="exportKeywords" style="flex: 1;
       min-width: 75px;
       max-width: 70px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
      ">Export</button>
 <button id="importKeywords" style="flex: 1;
       min-width: 75px;
       max-width: 70px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
      ">Import</button>
 <button id="toggleBlockKeywords" style="flex: 1;
       min-width: 80px;
       max-width: 90px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
       font-size: 13px;
       font-weight: bold;
      ">Unblock All</button>
 <button id="clearKeywords" style="flex: 1;
       min-width: 75px;
       max-width: 80px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
      ">Clear all</button>
 <button id="toggleFavoriteUsers" style="width: 80px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
       font-size: 13px;
       font-weight: bold;
      ">Favorite Users</button>
 <button id="toggleBlockedUsers" style="width: 80px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
       font-size: 13px;
       font-weight: bold;
      ">Blocked Users</button>
 <button id="toggleVerifiedPosts" style="width: 242px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
       font-size: 13px;
       font-weight: bold;
      ">Hide verified accounts: Click to Enable</button>
 <button id="toggleNonVerifiedPosts" style="width: 242px;
      
       background: #203142;
       color: #a9b6c4;
       border: none;
       border-radius: 5px;
       height: 40px;
       font-size: 13px;
       font-weight: bold;
      ">Hide non-verified accounts: Turn ON</button>
 </div>
 <div id="listLabel" style="
     width: 222px;
     color: #15202b !important;
     font-size: 18px !important;
     top: 30px;
     position: inherit;
     font-weight: 700;
 ">List of Keywords</div>
 <ul id="keywordList" style="
     list-style: none;
     padding: 0px;
     font-size: 14px;
     color: rgb(255, 255, 255);
     max-height: 275px;
     overflow-y: auto;
     background-color: rgb(21, 32, 43);
     box-shadow: inset rgb(0 0 0 / 31%) 0px 11px 24px 0px;
     user-select: text; 
     top: 19px;
 "></ul>
`;

    document.body.appendChild(panel);

    // ========== Функция обновления счетчиков ========== //
   function updateCounters() {
    document.querySelector("#counter-keywords .counter-number").textContent = hiddenKeywords.length;
    document.querySelector("#counter-favorites .counter-number").textContent = favoriteUsers.length;
    document.querySelector("#counter-blocked .counter-number").textContent = blockedUsers.length;
    document.querySelector("#counter-gif .counter-number").textContent = gifBlacklist.length;
}

    const searchInput = document.getElementById("searchInput");
    const clearSearch = document.getElementById("clearSearch");

    searchInput.addEventListener("input", () => {
        clearSearch.style.display = searchInput.value.trim() ? "block" : "none";
        updateKeywordList();
        updateCounters();
    });

    clearSearch.addEventListener("click", () => {
        searchInput.value = "";
        clearSearch.style.display = "none";
        updateKeywordList();
        updateCounters();
    });

    const lengthFilterInput = document.createElement("input");
    lengthFilterInput.type = "number";
    lengthFilterInput.placeholder = "Min length";
    lengthFilterInput.style.width = "80px";
    lengthFilterInput.style.marginTop = "10px";
    panel.appendChild(lengthFilterInput);

    lengthFilterInput.addEventListener("change", () => {
        debouncedHidePosts();
    });

    // ========== Перетаскивание панели ========== //
    let isDragging = false;
    let offsetX = 0;
    let offsetY = 0;

    panel.addEventListener("mousedown", (event) => {
        // Проверяем, выделен ли текст
        if (window.getSelection().toString()) {
            return; // Если текст выделен, не начинаем перетаскивание
        }

        // Проверяем, что клик не по элементу управления
        const target = event.target;
        if (
            target.tagName === "INPUT" ||
            target.tagName === "BUTTON" ||
            target.tagName === "UL" ||
            target.closest("#keywordList") ||
            target.closest("#clearSearch")
        ) {
            return; // Не начинаем перетаскивание
        }

        isDragging = true;
        offsetX = event.clientX - panel.offsetLeft;
        offsetY = event.clientY - panel.offsetTop;
        panel.style.cursor = "grabbing";
    });

    document.addEventListener("mousemove", (event) => {
        if (isDragging) {
            panel.style.left = event.clientX - offsetX + "px";
            panel.style.top = event.clientY - offsetY + "px";
        }
    });

    document.addEventListener("mouseup", () => {
        isDragging = false;
        panel.style.cursor = "move";
    });

    // ========== Создание попапа для языков ========== //
    const languagePopup = document.createElement("div");
    languagePopup.style.display = "none";
    languagePopup.style.position = "fixed";
    languagePopup.style.top = "100px";
    languagePopup.style.right = "65px";
    languagePopup.style.transform = "translate(-145%, 45%)";
    languagePopup.style.backgroundColor = " #34506c";
    languagePopup.style.padding = "20px";
    languagePopup.style.borderRadius = "8px";
    languagePopup.style.zIndex = "10000";
    languagePopup.style.width = "288px";
    languagePopup.style.boxShadow = "0 0 10px rgba(0,0,0,0.5)";
    languagePopup.style.fontFamily = "Arial, sans-serif";

    for (const language in languageFilters) {
        const checkbox = document.createElement("input");
        checkbox.type = "checkbox";
        checkbox.id = `lang-${language}`;
        checkbox.name = language;

        const label = document.createElement("label");
        label.htmlFor = `lang-${language}`;
        label.textContent = language.charAt(0).toUpperCase() + language.slice(1);

        const wrapper = document.createElement("div");
        wrapper.appendChild(checkbox);
        wrapper.appendChild(label);
        languagePopup.appendChild(wrapper);
    }

    const closeButton = document.createElement("button");
    closeButton.textContent = "X";
    closeButton.style.fontSize = "22px";
    closeButton.style.position = "relative";
    closeButton.style.width = "40px";
    closeButton.style.height = "40px";
    closeButton.style.borderRadius = "50%";
    closeButton.style.backgroundColor = "#203142";
    closeButton.style.color = " rgb(237 131 123) ";
    closeButton.style.border = "none";
    closeButton.style.display = "flex";
    closeButton.style.alignItems = "center";
    closeButton.style.justifyContent = "center";
    closeButton.style.cursor = "pointer"; 
    closeButton.style.left = "1%";
    closeButton.style.top = "20px";
    closeButton.addEventListener("click", () => {
        languagePopup.style.display = "none";
    });
    languagePopup.appendChild(closeButton);

    document.body.appendChild(languagePopup);

    document.getElementById("openLanguagePopup").addEventListener("click", () => {
        languagePopup.style.display = "block";
    });

    const warningText = document.createElement("div");
    warningText.textContent = "⚠️ Careful, it might stop working if it lags";
    warningText.style.color = " #ffcc00";
    warningText.style.fontSize = "14px";
    warningText.style.marginBottom = "10px";
    warningText.style.textAlign = "end";
    warningText.style.right = "0px";
    warningText.style.position = "relative";
    warningText.style.top = "25px";
     warningText.style.width = "260px";
    languagePopup.appendChild(warningText);

    for (const language in languageFilters) {
        document.getElementById(`lang-${language}`).addEventListener("change", () => {
            updateLanguageFilter(language);
        });
    }

   // ========== Стили для подсветки, скроллбара и выравнивания кнопок ========== //
const style = document.createElement("style");
style.textContent = `
button {
        transition: box-shadow 0.3s, transform 0.3s, background-color 0.3s;
    }
    button:hover {
        box-shadow: 0 0 10px rgba(255, 255, 255, 0.7);
    }
    button:active {
        transform: scale(0.95);
        box-shadow: 0 0 5px rgba(255, 255, 255, 0.7);
    }
    button.active,
    #toggleVerifiedPosts.active,
    #toggleNonVerifiedPosts.active,
    #toggleFavoriteUsers.active,
    #toggleBlockedUsers.active {
          background: linear-gradient(5deg, rgb(74 169 117), rgb(7 100 54)) !important;
          box-shadow: 0 0 8px rgb(216 240 253 / 51%) !important;
          color: rgb(21 32 43) !important;
    }
    .version-text {
        left: 275px;
        position: relative;
        bottom: 18px;
        color: #15202b;
        margin: 0;
        font-size: 14px;
        width: 47px;
    }
    #keywordInput {
        cursor: pointer;
        background: #15202b;
    }
    #favoriteUserButton:hover img {
        filter: brightness(1.2);
    }
    #favoriteUserButton:active img {
        transform: scale(0.9);
    }
  #keywordList button {
    background-color: transparent; /* Убираем красный фон */
    color: #f44336; /* Цвет иконки (красный, как раньше) */
    border: none;
    border-radius: 3px;
    cursor: pointer;
    width: 24px; /* Фиксированная ширина для SVG */
    height: 24px; /* Фиксированная высота для SVG */
    display: flex;
    align-items: center;
    justify-content: center;
    padding: 0; /* Убираем отступы для центрирования */
}
  ul#keywordList {
     width: 325px !important;
}
#keywordList li {
    width: 290px !important;
} 
 
 
#keywordList li { 
    border-top: 2px solid #1c6596;
    color: rgb(159 173 182);
}
 

 #keywordList {  
    border: 1px solid #1c6596;
    border-radius: 10px;
    background-color: rgb(21, 32, 43); 
}


h3#panelHeader {
    width: 140px;
    color: #15202b;
}


 

 #keywordList {
  width: 300px !important !important;
  overflow-x: hidden !important;
}
#keywordList li {  
    border-top: 2px solid #1c6596;
    color: rgb(136 151 161);
}
    #keywordList li {
        display: flex;
        justify-content: space-between;
        align-items: center;
        padding: 5px 10px;
    }
    #keywordList li button {
        width: 30px;
        height: 30px;
        display: flex;
        justify-content: center;
        align-items: center;
    }
 #keywordList {
    height: 400px; /* Фиксированная высота viewport, подберите под дизайн */
    overflow-y: auto; /* Скроллбар только по вертикали */
    position: relative; /* Для абсолютного позиционирования элементов */
}
#keywordList li {
    height: 40px; /* Фиксированная высота каждого элемента, для точного расчёта */
    line-height: 40px; /* Выравнивание текста */
    display: flex;
    justify-content: space-between;
    align-items: center;
}
   /*------ NEW: Стили для контейнера GIF и кнопок---------- */
.gif-control-container {
    position: absolute;
    bottom: 65px;
    left: 35px;
    display: flex;
    align-items: center;
    gap: 5px;
    background: rgb(0 0 0 / 49%);
    padding: 0px 4px;
    border-radius: 8px;
    z-index: 1000;
    font-family: Arial, sans-serif;
    color: #87fff5;
    font-size: 11px;
    opacity: 0; /* Скрыт по умолчанию */
    visibility: hidden; /* Скрыт по умолчанию */
    transition: opacity 0.3s ease, visibility 0.3s ease; /* Плавное появление */
}

/* Показываем контейнер при наведении на элемент с data-testid="tweetPhoto" */
div[data-testid="tweetPhoto"]:hover .gif-control-container {
    opacity: 1;
    visibility: visible;
}

.gif-control-container .gif-label {
    max-width: 150px;
    white-space: nowrap;
    overflow: hidden;
    text-overflow: ellipsis;
}
.gif-control-container button {
       padding: 3px 5px;
    background: #34506c;
    color: #fff;
    border: none;
    border-radius: 5px;
    cursor: pointer;
    font-size: 11px;
}
.gif-control-container button:hover {
    background: #3b5a7b;
}
.gif-control-container button:active {
    transform: scale(0.95);
}
    /* Стили для кастомных тултипов */
.counter-span {
    position: relative;
    cursor: help; /* Курсор-подсказка при наведении */
}
#counter-keywords:hover,
#counter-gif:hover,
#counter-blocked:hover,
#counter-favorites:hover {
    color: #00ffad  !important;
    scale: 1.5  !important;
}  



#counter-favorites {
    color: #ffc400;
}

 
#counter-blocked {
    color: #ea7878;
}

#counter-gif {
    color: rgb(45 154 118);
}

#counter-keywords {
    color: #00a5ff;
}

.counter-tooltip {
    visibility: hidden;
    opacity: 0;
    position: absolute;
    bottom: 93%; /* Позиция над элементом */
    left: 50%;
    transform: translateX(-50%);
    background-color: #15202b; /* Цвет фона   */
    color: #fff;
    text-align: center;
    border-radius: 4px;
    padding: 5px 8px;
    font-size: 12px;
    white-space: nowrap;
    z-index: 100000;
    transition: opacity 0.3s, visibility 0.3s;
    box-shadow: 0 2px 5px rgba(0,0,0,0.3);
    border: 1px solid #203142;
}

.counter-span:hover .counter-tooltip {
    visibility: visible;
    opacity: 1;
} 


.counter-tooltip::after {
    content: "";
    position: absolute;
    top: 100%; /* Стрелка вниз */
    left: 50%;
    margin-left: -5px;
    border-width: 5px;
    border-style: solid;
    border-color: #15202b transparent transparent transparent;
}
    .counter-icon {
    width: 16px;
    height: 16px;
    vertical-align: middle;
    margin-right: 2px;
}
#clearSearch svg {
    width: 18px;
    height: 18px;
}
 input[id^="lang-"] {
    appearance: none !important;
    background: #2c4561 !important;
    border: 2px solid rgb(21, 32, 43);
    border-radius: 5px !important;
    width: 20px;
    height: 20px;
}
input[id^="lang-"]:checked {
    background: #243a51 !important; /* Цвет при выборе */
    border-color: #d7b415 !important; /* Зелёная рамка при выборе */
}
input[id^="lang-"]:checked::after {
    content: "✓"; /* Галочка при выборе */
    position: absolute; 
    left: 9%; 
    color: rgb(215 180 21);
    font-size: 14px;
    font-weight: bold;
}
label[for^="lang-"] {
    cursor: pointer;
    font-size: 14px;
    margin-left: 5px;
}
div:has(input[id^="lang-"]) {
    display: flex;
    align-items: center;
    margin-bottom: 10px;
}

    /* Стили для скроллбара (замена GM_addStyle) */
    #keywordList:hover::-webkit-scrollbar {
        width: 25px !important;
    }
    #keywordList::-webkit-scrollbar {
        width: 18px !important;
    }
    
    #keywordList {
        scrollbar-width: auto;
        scrollbar-color: auto ;
    }
`;
document.head.appendChild(style);

// Принудительное применение стилей скроллбара (опционально, для немедленного эффекта)
const keywordList = document.getElementById("keywordList");
if (keywordList) {
    keywordList.style.scrollbarColor = 'auto';
    keywordList.style.scrollbarWidth = 'auto';
}

    // ========== Кнопка переключения панели FilterX ========== //
    let isSwitchOn = localStorage.getItem("isSwitchOn") === "true";
    const toggleButton = document.createElement("div");
    toggleButton.style.display = "flex";
    toggleButton.style.alignItems = "center";
    toggleButton.style.gap = "10px";
    toggleButton.style.background = " #15202b";
    toggleButton.style.border = "4px solid #6c7e8e";
    toggleButton.style.borderRadius = "45px";
    toggleButton.style.padding = "8px 12px";
    toggleButton.style.marginTop = "10px";
    toggleButton.style.cursor = "pointer";
    toggleButton.style.width = "auto";
    toggleButton.className = "css-175oi2r r-16y2uox";

    function isGrokPage() {
        return window.location.href.startsWith("https://x.com/i/grok");
    }

    const toggleLabel = document.createElement("label");
    toggleLabel.style.display = "inline-block";
    toggleLabel.style.width = "50px";
    toggleLabel.style.height = "25px";
    toggleLabel.style.borderRadius = "25px";
    toggleLabel.style.backgroundColor = isSwitchOn ? " #425364" : " #0d1319";
    toggleLabel.style.position = "relative";
    toggleLabel.style.cursor = "pointer";
    toggleLabel.style.transition = "background-color 0.3s";
    toggleLabel.style.top = "0px";
    toggleLabel.style.left = "75px";

    const toggleSwitch = document.createElement("div");
    toggleSwitch.style.position = "absolute";
    toggleSwitch.style.width = "21px";
    toggleSwitch.style.height = "21px";
    toggleSwitch.style.borderRadius = "50%";
    toggleSwitch.style.backgroundColor = " #6c7e8e";
    toggleSwitch.style.top = "2px";
    toggleSwitch.style.left = isSwitchOn ? "calc(100% - 23px)" : "2px";
    toggleSwitch.style.transition = "left 0.3s ease";
    toggleSwitch.style.boxShadow = "rgb(21, 32, 43) -1px 1px 4px 1px";

    function toggleSwitchState(event) {
        event.stopPropagation();
        isSwitchOn = !isSwitchOn;
        localStorage.setItem("isSwitchOn", isSwitchOn.toString());
        toggleSwitch.style.left = isSwitchOn ? "calc(100% - 23px)" : "2px";
        toggleLabel.style.backgroundColor = isSwitchOn ? " #425364" : " #0d1319";

        if (isSwitchOn) {
            panel.style.display = "block";
            setTimeout(() => {
                panel.style.height = "710px";
            }, 10);
        } else {
            panel.style.height = "0px";
            setTimeout(() => {
                panel.style.display = "none";
            }, 300);
        }
        isPanelVisible = isSwitchOn;
        localStorage.setItem("panelVisible", isPanelVisible.toString());
    }

    const debouncedToggleSwitchState = debounce(toggleSwitchState, 300);

    toggleButton.addEventListener("click", debouncedToggleSwitchState);
    toggleLabel.appendChild(toggleSwitch);
    toggleButton.appendChild(toggleLabel);

    const toggleText = document.createElement("span");
    toggleText.textContent = "FilterX";
    toggleText.style.color = " #6c7e8e";
    toggleText.style.fontFamily = "Arial, sans-serif";
    toggleText.style.fontSize = "16px";
    toggleText.style.fontWeight = "bold";
    toggleText.style.position = "absolute";
    toggleText.style.top = "12px";
    toggleText.style.left = "25px";
    toggleButton.appendChild(toggleText);

    function waitForPostButton(callback) {
        const interval = setInterval(() => {
            const postButton = document.querySelector("[data-testid='SideNav_NewTweet_Button']");
            const postButtonContainer = postButton?.parentElement;
            if (postButtonContainer) {
                clearInterval(interval);
                callback(postButtonContainer);
            } else {
                console.warn("Контейнер кнопки 'Опубликовать пост' не найден");
            }
        }, 500);
    }

    waitForPostButton((postButtonContainer) => {
        toggleButton.style.display = isGrokPage() ? "none" : "flex";
        postButtonContainer.appendChild(toggleButton);
    });

    // ========== Управление высотой панели ========== //
    let isPanelVisible = localStorage.getItem("panelVisible") === "true";

    function togglePanel() {
        if (isPanelVisible) {
            panel.style.height = "0px";
            setTimeout(() => {
                panel.style.display = "none";
            }, 300);
        } else {
            panel.style.display = "block";
            setTimeout(() => {
                panel.style.height = "710px";
            }, 10);
        }
        isPanelVisible = !isPanelVisible;
        localStorage.setItem("panelVisible", isPanelVisible.toString());
    }

    toggleButton.addEventListener("click", togglePanel);

    if (isPanelVisible) {
        panel.style.height = "710px";
        panel.style.display = "block";
    } else {
        panel.style.height = "0px";
        panel.style.display = "none";
    }

    // ========== Элементы управления ========== //
    const addKeywordBtn = document.getElementById("addKeyword");
    const clearKeywordsBtn = document.getElementById("clearKeywords");
    const exportKeywordsBtn = document.getElementById("exportKeywords");
    const importKeywordsBtn = document.getElementById("importKeywords");
    const toggleVerifiedBtn = document.getElementById("toggleVerifiedPosts");
    const toggleNonVerifiedBtn = document.getElementById("toggleNonVerifiedPosts");
    const toggleBlockBtn = document.getElementById("toggleBlockKeywords");
    const openLanguagePopupBtn = document.getElementById("openLanguagePopup");
    const toggleFavoriteUsersBtn = document.getElementById("toggleFavoriteUsers");
    const toggleBlockedUsersBtn = document.getElementById("toggleBlockedUsers");

    // ========== Обработчики событий ========== //
    // NEW: Модифицированный обработчик добавления с поддержкой gif:
    addKeywordBtn.addEventListener("click", () => {
        const inputValue = keywordInput.value.trim().toLowerCase();
        if (inputValue) {
            if (inputValue.startsWith("gif:")) {
                const gifKeyword = inputValue.slice(4).trim();
                if (gifKeyword && !gifBlacklist.includes(gifKeyword)) {
                    gifBlacklist.push(gifKeyword);
                    saveGifBlacklist();
                    showNotification(`Добавлено в черный список GIF: ${gifKeyword}`, 'success');
                } else {
                    showNotification(`Ключевое слово GIF уже существует: ${gifKeyword}`, 'error');
                }
            } else if (displayMode === "favorites" && inputValue.startsWith("@")) {
                const username = inputValue.slice(1);
                if (!favoriteUsers.includes(username)) {
                    favoriteUsers.push(username);
                    saveFavoriteUsers();
                    showNotification(`Добавлен в избранное: @${username}`, 'success');
                } else {
                    showNotification(`@${username} уже в избранном`, 'error');
                }
            } else if (displayMode === "blocked" && inputValue.startsWith("@")) {
                const username = inputValue.slice(1);
                if (!blockedUsers.includes(username)) {
                    blockedUsers.push(username);
                    saveBlockedUsers();
                    showNotification(`Заблокирован: @${username}`, 'success');
                } else {
                    showNotification(`@${username} уже заблокирован`, 'error');
                }
            } else if (displayMode === "keywords" && !hiddenKeywords.includes(inputValue)) {
                hiddenKeywords.push(inputValue);
                saveKeywords();
                showNotification(`Добавлено ключевое слово: ${inputValue}`, 'success');
            } else {
                showNotification(`Ключевое слово уже существует: ${inputValue}`, 'error');
            }
            keywordInput.value = "";
            updateKeywordList();
            updateCounters();
            debouncedHidePosts();
        }
    });

    toggleNonVerifiedBtn.textContent = `Hide non-verified accounts: ${hideNonVerifiedAccounts ? "Turn OFF" : "Turn ON"}`;
    toggleNonVerifiedBtn.addEventListener("click", () => {
        hideNonVerifiedAccounts = !hideNonVerifiedAccounts;
        localStorage.setItem("hideNonVerifiedAccounts", JSON.stringify(hideNonVerifiedAccounts));
        toggleNonVerifiedBtn.textContent = `Hide non-verified accounts: ${hideNonVerifiedAccounts ? "Turn OFF" : "Turn ON"}`;
        toggleNonVerifiedBtn.classList.toggle('active', hideNonVerifiedAccounts);
        hidePosts();
    });

    clearKeywordsBtn.addEventListener("click", () => {
        if (confirm("Are you sure you want to clear the list?")) {
            if (displayMode === "favorites") {
                favoriteUsers = [];
                saveFavoriteUsers();
            } else if (displayMode === "blocked") {
                blockedUsers = [];
                saveBlockedUsers();
            // NEW: Очистка GIF-списка, если режим keywords
            } else if (displayMode === "keywords") {
                hiddenKeywords = [];
                gifBlacklist = [];
                unblockedKeywords = [];
                saveKeywords();
                saveGifBlacklist();
                saveUnblockedKeywords();
            }
            updateKeywordList();
            updateCounters();
            hidePosts();
        }
    });

    // NEW: Модифицированный экспорт с поддержкой GIF
    exportKeywordsBtn.addEventListener("click", () => {
        let data, baseName, nameWord;
        if (displayMode === "favorites") {
            data = favoriteUsers;
            baseName = "favorite_users";
            nameWord = "users";
        } else if (displayMode === "blocked") {
            data = blockedUsers;
            baseName = "blocked_users";
            nameWord = "users";
        } else {
            data = { keywords: hiddenKeywords, gif: gifBlacklist }; // NEW: Экспорт как объект с GIF
            baseName = "hidden_keywords_and_gif";
            nameWord = "items";
        }
        const dataStr = `data:text/json;charset=utf-8,${encodeURIComponent(JSON.stringify(data))}`;
        const count = displayMode === "keywords" ? hiddenKeywords.length + gifBlacklist.length : data.length;
        const fileName = `${baseName}_${count}_${nameWord}.json`;
        const downloadAnchor = document.createElement("a");
        downloadAnchor.setAttribute("href", dataStr);
        downloadAnchor.setAttribute("download", fileName);
        document.body.appendChild(downloadAnchor);
        downloadAnchor.click();
        document.body.removeChild(downloadAnchor);
    });

    // NEW: Модифицированный импорт с поддержкой GIF
    importKeywordsBtn.addEventListener("click", () => {
        const input = document.createElement("input");
        input.type = "file";
        input.accept = "application/json";
        input.addEventListener("change", (event) => {
            const file = event.target.files[0];
            const reader = new FileReader();
            reader.onload = () => {
                try {
                    const importedData = JSON.parse(reader.result);
                    if (displayMode === "favorites" || displayMode === "blocked") {
                        const list = Array.isArray(importedData) ? importedData : [];
                        if (displayMode === "favorites") {
                            favoriteUsers = [...new Set([...favoriteUsers, ...list.map(u => u.startsWith("@") ? u.slice(1) : u)])];
                            saveFavoriteUsers();
                        } else {
                            blockedUsers = [...new Set([...blockedUsers, ...list.map(u => u.startsWith("@") ? u.slice(1) : u)])];
                            saveBlockedUsers();
                        }
                    } else if (displayMode === "keywords") {
                        if (Array.isArray(importedData)) {
                            hiddenKeywords = [...new Set([...hiddenKeywords, ...importedData])];
                        } else if (typeof importedData === 'object') {
                            hiddenKeywords = [...new Set([...hiddenKeywords, ...(importedData.keywords || [])])];
                            gifBlacklist = [...new Set([...gifBlacklist, ...(importedData.gif || [])])];
                            saveGifBlacklist();
                        }
                        saveKeywords();
                    }
                    updateKeywordList();
                    updateCounters();
                    hidePosts();
                } catch (e) {
                    alert("Error reading the file.");
                }
            };
            reader.readAsText(file);
        });
        input.click();
    });

    toggleBlockBtn.addEventListener("click", () => {
        if (displayMode === "keywords") {
            if (hiddenKeywords.length > 0 || gifBlacklist.length > 0) {
                unblockedKeywords = [...hiddenKeywords, ...gifBlacklist.map(k => `gif:${k}`)]; // NEW: Сохраняем GIF с префиксом
                hiddenKeywords = [];
                gifBlacklist = [];
                toggleBlockBtn.textContent = "Block All";
            } else {
                hiddenKeywords = unblockedKeywords.filter(k => !k.startsWith("gif:"));
                gifBlacklist = unblockedKeywords.filter(k => k.startsWith("gif:")).map(k => k.slice(4));
                unblockedKeywords = [];
                toggleBlockBtn.textContent = "Unblock All";
            }
            saveKeywords();
            saveGifBlacklist();
            saveUnblockedKeywords();
            updateKeywordList();
            updateCounters();
            hidePosts();
        }
    });

    toggleBlockBtn.textContent = hiddenKeywords.length > 0 || gifBlacklist.length > 0 ? "Unblock All" : "Block All";

    toggleVerifiedBtn.addEventListener("click", () => {
        hideVerifiedAccounts = !hideVerifiedAccounts;
        toggleVerifiedBtn.textContent = `Hide verified accounts: ${hideVerifiedAccounts ? "Turn OFF" : "Turn ON"}`;
        toggleVerifiedBtn.classList.toggle('active', hideVerifiedAccounts);
        hidePosts();
    });

    toggleFavoriteUsersBtn.addEventListener("click", () => {
        displayMode = displayMode === "favorites" ? "keywords" : "favorites";
        toggleFavoriteUsersBtn.textContent = displayMode === "favorites" ? "Keywords" : "Favorite Users";
        toggleBlockedUsersBtn.textContent = "Blocked Users";
        keywordInput.placeholder = displayMode === "favorites" || displayMode === "blocked" ? "Enter @username" : "Enter the word or gif:keyword"; // NEW: Уточнение плейсхолдера

        toggleFavoriteUsersBtn.classList.toggle('active', displayMode === "favorites");
        toggleBlockedUsersBtn.classList.remove('active');
        updateKeywordList();
        updateCounters();
    });

    toggleBlockedUsersBtn.addEventListener("click", () => {
        displayMode = displayMode === "blocked" ? "keywords" : "blocked";
        toggleBlockedUsersBtn.textContent = displayMode === "blocked" ? "Keywords" : "Blocked Users";
        toggleFavoriteUsersBtn.textContent = "Favorite Users";
        keywordInput.placeholder = displayMode === "favorites" || displayMode === "blocked" ? "Enter @username" : "Enter the word or gif:keyword"; // NEW: Уточнение плейсхолдера

        toggleBlockedUsersBtn.classList.toggle('active', displayMode === "blocked");
        toggleFavoriteUsersBtn.classList.remove('active');
        updateKeywordList();
        updateCounters();
    });

    openLanguagePopupBtn.addEventListener("click", () => {
        const panelRect = panel.getBoundingClientRect();
        languagePopup.style.top = `${panelRect.top - 320}px`;
        languagePopup.style.left = `${panelRect.right - 10}px`;
        languagePopup.style.display = "block";
    });

   // ========== Обновление списка ключевых слов, избранных или заблокированных пользователей ========== //
function updateKeywordList() {
    const list = document.getElementById("keywordList");
    const label = document.getElementById("listLabel");
    const searchQuery = searchInput.value.trim().toLowerCase();
    list.innerHTML = ""; // Оставляем, чтобы очистить при обновлении

    if (displayMode === "favorites") {
        label.textContent = "Favorite Users";
        const tempData = favoriteUsers.filter(user => user.toLowerCase().includes(searchQuery)).map(user => ({ type: 'user', value: user }));
       filteredData = tempData; // Используем полный массив без обрезки
if (filteredData.length === 0) {
    list.innerHTML = searchQuery ? "<li>No matches found</li>" : "<li>Нет</li>";
} else {
    renderVisibleItems(list);
}
    } else if (displayMode === "blocked") {
        label.textContent = "Blocked Users";
        const tempData = blockedUsers.filter(user => user.toLowerCase().includes(searchQuery)).map(user => ({ type: 'user', value: user }));
      filteredData = tempData; // Используем полный массив без обрезки
if (filteredData.length === 0) {
    list.innerHTML = searchQuery ? "<li>No matches found</li>" : "<li>Нет</li>";
} else {
    renderVisibleItems(list);
}
    } else {
        label.textContent = "List of Keywords and GIF";
        const allItems = [
            ...hiddenKeywords.map(k => ({ type: 'keyword', value: k })),
            ...gifBlacklist.map(k => ({ type: 'gif', value: `gif:${k}` }))
        ];
        const tempData = allItems.filter(item => item.value.toLowerCase().includes(searchQuery));
       filteredData = tempData; // Используем полный массив без обрезки
if (filteredData.length === 0) {
    list.innerHTML = searchQuery ? "<li>No matches found</li>" : "<li>Нет</li>";
} else {
    renderVisibleItems(list);
}
    }

    if (filteredData.length > 0) {
        list.removeEventListener('scroll', handleScroll);
        list.addEventListener('scroll', handleScroll);
    }
}

function renderVisibleItems(container) {
    const scrollTop = container.scrollTop;
    const startIndex = Math.floor(scrollTop / itemHeight);
    const visibleCount = Math.ceil(viewportHeight / itemHeight) + 2; // Видимые + буфер (2 сверху/снизу)
    const endIndex = Math.min(startIndex + visibleCount, filteredData.length);

    container.innerHTML = "";

    for (let i = startIndex; i < endIndex; i++) {
        const item = filteredData[i];
        const listItem = document.createElement("li");
        listItem.style.position = 'absolute';
        listItem.style.top = `${i * itemHeight}px`;
        listItem.style.width = '100%';
        listItem.textContent = item.value;

        const deleteButton = document.createElement("button");
        deleteButton.innerHTML = `
  <svg xmlns="http://www.w3.org/2000/svg" 
          width="24" 
          height="24"
          shape-rendering="geometricPrecision" 
          text-rendering="geometricPrecision" 
          image-rendering="optimizeQuality" 
          fill-rule="evenodd" 
          clip-rule="evenodd" 
          viewBox="0 0 456 511.82">
          <path fill=" #FD3B3B" 
          d="M48.42 140.13h361.99c17.36 0 29.82 9.78 28.08 28.17l-30.73 317.1c-1.23 13.36-8.99 26.42-25.3 26.42H76.34c-13.63-.73-23.74-9.75-25.09-24.14L20.79 168.99c-1.74-18.38 9.75-28.86 27.63-28.86zM24.49 38.15h136.47V28.1c0-15.94 10.2-28.1 27.02-28.1h81.28c17.3 0 27.65 11.77 27.65 28.01v10.14h138.66c.57 0 1.11.07 1.68.13 10.23.93 18.15 9.02 18.69 19.22.03.79.06 1.39.06 2.17v42.76c0 5.99-4.73 10.89-10.62 11.19-.54 0-1.09.03-1.63.03H11.22c-5.92 0-10.77-4.6-11.19-10.38 0-.72-.03-1.47-.03-2.23v-39.5c0-10.93 4.21-20.71 16.82-23.02 2.53-.45 5.09-.37 7.67-.37zm83.78 208.38c-.51-10.17 8.21-18.83 19.53-19.31 11.31-.49 20.94 7.4 21.45 17.57l8.7 160.62c.51 10.18-8.22 18.84-19.53 19.32-11.32.48-20.94-7.4-21.46-17.57l-8.69-160.63zm201.7-1.74c.51-10.17 10.14-18.06 21.45-17.57 11.32.48 20.04 9.14 19.53 19.31l-8.66 160.63c-.52 10.17-10.14 18.05-21.46 17.57-11.31-.48-20.04-9.14-19.53-19.32l8.67-160.62zm-102.94.87c0-10.23 9.23-18.53 20.58-18.53 11.34 0 20.58 8.3 20.58 18.53v160.63c0 10.23-9.24 18.53-20.58 18.53-11.35 0-20.58-8.3-20.58-18.53V245.66z"/>
 <style xmlns="http://www.w3.org/1999/xhtml" 
          id="mh1i307m.ol">
::selection {
      background: #d0fefe4f !important;
      color: #bee4e0 !important;
}
::-moz-selection {
    background: #d0fefe4f !important;
    color: #bee4e0 !important;
}        
</style></svg>
        `;
        deleteButton.style.border = "none";
        deleteButton.style.borderRadius = "3px";
        deleteButton.style.cursor = "pointer";

        deleteButton.addEventListener("click", () => {
            if (item.type === 'gif') {
                const gifKeyword = item.value.slice(4);
                gifBlacklist.splice(gifBlacklist.indexOf(gifKeyword), 1);
                saveGifBlacklist();
                showNotification(`Удалено из GIF: ${gifKeyword}`, 'success');
            } else if (item.type === 'user' && displayMode === 'favorites') {
                favoriteUsers.splice(favoriteUsers.indexOf(item.value), 1);
                saveFavoriteUsers();
                showNotification(`Removed @${item.value} from favorites`, 'success');
            } else if (item.type === 'user' && displayMode === 'blocked') {
                blockedUsers.splice(blockedUsers.indexOf(item.value), 1);
                saveBlockedUsers();
                showNotification(`Unblocked @${item.value}`, 'success');
            } else {
                hiddenKeywords.splice(hiddenKeywords.indexOf(item.value), 1);
                saveKeywords();
                showNotification(`Удалено ключевое слово: ${item.value}`, 'success');
            }
            updateKeywordList();
            updateCounters();
            hidePosts();
        });

        listItem.appendChild(deleteButton);
        container.appendChild(listItem);
    }

    // Ограничиваем высоту до 10 элементов
    container.style.height = `${filteredData.length * itemHeight}px`; // Полная виртуальная высота
}

//  обработчик скролла
function handleScroll() {
    renderVisibleItems(this);
}

    // ========== Создание кнопки избранного в профиле ========== //
    function addFavoriteButtonToProfile() {
        const checkProfileLoaded = setInterval(() => {
            const buttonContainer = document.querySelector('.css-175oi2r.r-obd0qt.r-18u37iz.r-1w6e6rj.r-1h0z5md.r-dnmrzs') ||
                                   document.querySelector('[data-testid="userActions"]');
            const username = getUsernameFromProfile();

            if (buttonContainer && username && !buttonContainer.querySelector('#favoriteUserButton')) {
                clearInterval(checkProfileLoaded);

                const isFavorite = favoriteUsers.includes(username);

                const favoriteButton = document.createElement('button');
                favoriteButton.id = 'favoriteUserButton';
                favoriteButton.setAttribute('aria-label', isFavorite ? 'Удалить из избранного' : 'Добавить в избранное');
                favoriteButton.setAttribute('role', 'button');
                favoriteButton.type = 'button';
                favoriteButton.className = 'css-175oi2r r-sdzlij r-1phboty r-rs99b7 r-lrvibr r-6gpygo r-1wron08 r-2yi16 r-1qi8awa r-1loqt21 r-o7ynqc r-6416eg r-1ny4l3l css-175oi2r r-18u37iz r-1wtj0ep';
                favoriteButton.style.borderColor = 'rgb(83, 100, 113)';
                favoriteButton.style.backgroundColor = 'rgba(0, 0, 0, 0)';

                const buttonContent = document.createElement('div');
                buttonContent.dir = 'ltr';
                buttonContent.className = 'css-146c3p1 r-bcqeeo r-qvutc0 r-1qd0xha r-q4m81j r-a023e6 r-rjixqe r-b88u0q r-1awozwy r-6koalj r-18u37iz r-16y2uox r-1777fci';
                buttonContent.style.color = 'rgb(239, 243, 244)';

                const icon = document.createElement('img');
                icon.src = '';
                icon.style.width = '25px';
                icon.style.height = '25px';
                icon.style.filter = isFavorite ? 'hue-rotate(300deg) saturate(2)' : 'none';
                icon.setAttribute('aria-hidden', 'true');
                icon.className = 'r-4qtqp9 r-yyyyoo r-dnmrzs r-bnwqim r-lrvibr r-m6rgpd r-z80fyv r-19wmn03';

                buttonContent.appendChild(icon);
                favoriteButton.appendChild(buttonContent);

                favoriteButton.addEventListener('click', () => {
                    const currentlyFavorite = favoriteUsers.includes(username);
                    if (currentlyFavorite) {
                        favoriteUsers = favoriteUsers.filter(user => user !== username);
                        favoriteButton.setAttribute('aria-label', 'Добавить в избранное');
                        icon.style.filter = 'none';
                    } else {
                        favoriteUsers.push(username);
                        favoriteButton.setAttribute('aria-label', 'Удалить из избранного');
                        icon.style.filter = 'hue-rotate(300deg) saturate(2)';
                    }
                    saveFavoriteUsers();
                    updateKeywordList();
                    updateCounters();
                    debouncedHidePosts();
                });

                buttonContainer.insertBefore(favoriteButton, buttonContainer.firstChild);
                console.log('Favorite button added for:', username);
            }
        }, 500);

        setTimeout(() => clearInterval(checkProfileLoaded), 5000);
    }

    // ========== Автоматическое скрытие панели при открытии фото ========== //
    function isPhotoViewerOpen() {
        const currentUrl = window.location.href;
        const isPhotoOpen = /\/photo\/\d+$/.test(currentUrl);
        const photoModal = document.querySelector('div[aria-label="Image"]') || document.querySelector('div[data-testid="imageViewer"]');
        return isPhotoOpen || !!photoModal;
    }

    function updateToggleButtonVisibility() {
        toggleButton.style.display = isGrokPage() ? "none" : "flex";
    }

    function toggleElementsVisibility() {
        const isPhotoOpen = isPhotoViewerOpen();
        if (isPhotoOpen || isGrokPage()) {
            panel.style.display = "none";
            toggleButton.style.display = "none";
        } else {
            if (isPanelVisible) {
                panel.style.display = "block";
                panel.style.height = "710px";
            }
            toggleButton.style.display = "flex";
        }
    }

    // ========== Наблюдение за изменениями DOM ========== //
   const observer = new MutationObserver((mutations) => {
    for (const mutation of mutations) {
        // Игнорируем изменения, связанные с .gif-control-container
        if (mutation.target.classList?.contains('gif-control-container')) {
            continue;
        }
        debouncedHidePosts();
        break; // Обрабатываем только первое релевантное изменение
    }
});
observer.observe(document.body, {
    childList: true,
    subtree: true,
    attributes: false // Игнорируем изменения атрибутов, если не нужны
});

    const profileObserver = new MutationObserver(() => {
        if (window.location.href.includes('x.com/') && !window.location.href.includes('/status/')) {
            addFavoriteButtonToProfile();
        }
    });
    profileObserver.observe(document.body, { childList: true, subtree: true });

    toggleElementsVisibility();
    window.addEventListener("popstate", () => {
        updateToggleButtonVisibility();
        toggleElementsVisibility();
    });

    const urlObserver = new MutationObserver(() => {
        updateToggleButtonVisibility();
        toggleElementsVisibility();
    });
    urlObserver.observe(document.body, { childList: true, subtree: true });

    document.addEventListener("click", (event) => {
        if (event.target.closest('div[aria-label="Close"]') || event.target.closest('div[data-testid="imageViewer-close"]')) {
            setTimeout(toggleElementsVisibility, 100);
        }
    });

    // ========== Инициализация ========== //
    updateKeywordList();
    updateCounters();
    hidePosts();

    // Добавим инициализацию активного состояния при загрузке
    document.addEventListener('DOMContentLoaded', () => {
        toggleVerifiedBtn.classList.toggle('active', hideVerifiedAccounts);
        toggleNonVerifiedBtn.classList.toggle('active', hideNonVerifiedAccounts);
        toggleFavoriteUsersBtn.classList.toggle('active', displayMode === "favorites");
        toggleBlockedUsersBtn.classList.toggle('active', displayMode === "blocked");
    });

})();








QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址