Twitter XFilter panel 2.4.62 (c) tapeavion

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

目前為 2025-08-06 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

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

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         Twitter XFilter panel 2.4.62 (c) tapeavion
// @namespace    http://tampermonkey.net/
// @version      2.4.62
// @description  Hide posts by keywords with the dashboard and hide posts from verified accounts
// @author       gullampis810
// @match        https://x.com/*
// @match        https://x.com/i/grok*
// @match        https://blank.org/*
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        GM_setValue
// @grant        GM_getValue
// @license      MIT
// @icon         https://www.pinclipart.com/picdir/big/450-4507608_twitter-circle-clipart.png
// @downloadURL
// @updateURL
// ==/UserScript==


 (function () {
    "use strict";

    // ========== Настройки и инициализация ========== //
    const STORAGE_KEY = "hiddenKeywords";
    const FAVORITE_USERS_KEY = "favoriteUsers";
    const BLOCKED_USERS_KEY = "blockedUsers";
    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")) || [];
    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));
    }

    // ========== Функция для обновления фильтров по языкам ========== //
    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 = "";
                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());
                }
            });

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

            if (shouldHideVerified || shouldHideNonVerified || shouldHideByLanguage || shouldHideByKeyword) {
                article.style.display = "none";
            } else {
                article.style.display = "";
            }
        });
    }

    // ========== 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.style.position = "fixed";
panel.style.left = "1015px"; // Заменяем right на left
panel.style.top = "89px"; // Заменяем bottom на top
panel.style.width = "335px";
panel.style.height = "510px";
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.cursor = "move"; // Курсор для всей панели

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

panel.innerHTML = `
 <h3 style="margin: 0; font-size: 16px;" id="panelHeader">Hiding Control</h3>
 <div style="display: flex; align-items: center; gap: 10px;">
 <span id="counter-keywords" title="Keywords">🔠0</span>
 <span id="counter-favorites" title="Favorite Users">⭐👤 0</span>
 <span id="counter-blocked" title="Blocked Users">👤🚫 0</span>
 </div>
 <h4 class="version-text">v2.4.62</h4>
 <div style="display: flex; align-items: center; gap: 5px; margin: 10px 0;">
 <input id="keywordInput" type="text" placeholder="Enter the word or @username" style="width: calc(100% - 95px); height: 40px; padding: 5px; border-radius: 5px; border: none; background: #15202b; color: #fff;">
 <button id="addKeyword" style="min-width: 79px; max-width: 80px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px;">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; height: 40px; padding: 5px; 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;">✖</span>
 </div>
 <button id="openLanguagePopup" style="width: 80px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px; font-size: 13px; font-weight: bold;">Language Filtering</button>
 </div>
 <button id="exportKeywords" style="flex: 1; min-width: 60px; max-width: 70px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px;">Export</button>
 <button id="importKeywords" style="flex: 1; min-width: 60px; max-width: 70px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px;">Import</button>
 <button id="toggleBlockKeywords" style="flex: 1; min-width: 80px; max-width: 90px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px; font-size: 13px; font-weight: bold;">Unblock All</button>
 <button id="clearKeywords" style="flex: 1; min-width: 60px; max-width: 80px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px;">Clear all</button>
 <button id="toggleFavoriteUsers" style="width: 80px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px; font-size: 13px; font-weight: bold;">Favorite Users</button>
 <button id="toggleBlockedUsers" style="width: 80px; padding: 8px; background: #203142; color: white; border: none; border-radius: 5px; height: 40px; font-size: 13px; font-weight: bold;">Blocked Users</button>
 <button id="toggleVerifiedPosts" style="width: 242px; padding: 8px; background: #203142; color: white; 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; padding: 8px; background: #203142; color: white; 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="margin-top: 10px; font-size: 14px; color: #fff;">List of Keywords</div>
 <ul id="keywordList" style="list-style: none; padding: 0; margin-top: 5px; font-size: 14px; color: #fff; max-height: 135px; overflow-y: auto; border: 1px solid #ccc; border-radius: 5px; background-color: #15202b; box-shadow: 0 2px 5px rgba(0,0,0,0.3); user-select: text;"></ul>
`;

document.body.appendChild(panel);

// ========== Функция обновления счетчиков ========== //
function updateCounters() {
 document.getElementById("counter-keywords").textContent = `🔠 ${hiddenKeywords.length}`;
 document.getElementById("counter-favorites").textContent = `⭐👤 ${favoriteUsers.length}`;
 document.getElementById("counter-blocked").textContent = `👤🚫 ${blockedUsers.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 = "460px";
    languagePopup.style.right = "65px";
    languagePopup.style.transform = "translate(-52%, 7%)";
    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.position = "relative";
    closeButton.style.width = "40px";
    closeButton.style.height = "40px";
    closeButton.style.borderRadius = "50%";
    closeButton.style.backgroundColor = "#203142";
    closeButton.style.color = "#fff";
    closeButton.style.border = "none";
    closeButton.style.display = "flex";
    closeButton.style.alignItems = "center";
    closeButton.style.justifyContent = "center";
    closeButton.style.cursor = "pointer";
    closeButton.style.marginTop = "10px";
    closeButton.style.left = "82%";
    closeButton.style.top = "56px";
    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 = "⚠️it may stops working";
    warningText.style.color = " #ffcc00";
    warningText.style.fontSize = "14px";
    warningText.style.marginBottom = "10px";
    warningText.style.textAlign = "end";
    warningText.style.right = "38px";
    warningText.style.position = "relative";
    warningText.style.top = "20px";
    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(90deg,rgb(0, 99, 44),rgb(94, 161, 122)) !important;
        box-shadow: 0 0 8px rgba(216, 240, 253, 0.8) !important;
        color: white !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 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;
    }



`;

if (typeof GM_addStyle !== 'undefined') {
    GM_addStyle(`
        /* Стили скроллбара для #keywordList */
        #keywordList {
            scrollbar-width: auto !important; /* Изменено с thin на auto для Firefox */
            scrollbar-color: #c1a5ef #205151 !important;
        }
        #keywordList::-webkit-scrollbar {
            width: 25px !important;
        }
        #keywordList::-webkit-scrollbar-thumb {
            background-color: #c1a5ef !important;
            border-radius: 8px !important;
            border: 3px solid #4f3e6a !important;
            min-height: 47px !important;
        }
        #keywordList::-webkit-scrollbar-thumb:hover {
            background-color: #a68cd4 !important;
        }
        #keywordList::-webkit-scrollbar-track {
            background: #205151 !important;
            border-radius: 8px !important;
        }
        #keywordList::-webkit-scrollbar-corner {
            background: transparent !important;
        }

        /* Стили скроллбара для панели */
        #panel {
            scrollbar-width: auto !important;
            scrollbar-color: #c1a5ef #205151 !important;
        }
        #panel::-webkit-scrollbar {
            width: 25px !important;
        }
        #panel::-webkit-scrollbar-thumb {
            background-color: #c1a5ef !important;
            border-radius: 8px !important;
            border: 3px solid #4f3e6a !important;
            min-height: 47px !important;
        }
        #panel::-webkit-scrollbar-thumb:hover {
            background-color: #a68cd4 !important;
        }
        #panel::-webkit-scrollbar-track {
            background: #205151 !important;
            border-radius: 8px !important;
        }
        #panel::-webkit-scrollbar-corner {
            background: transparent !important;
        }

        /* Глобальные стили для остальных скроллбаров на странице */
        :not(#keywordList):not(#panel) {
            scrollbar-width: auto !important;
            scrollbar-color: #c1a5ef transparent !important;
        }
        :not(#keywordList):not(#panel)::-webkit-scrollbar {
            width: 25px !important;
        }
        :not(#keywordList):not(#panel)::-webkit-scrollbar-thumb {
            background-color: #c1a5ef !important;
            border-radius: 8px !important;
            border: 3px solid #4f3e6a !important;
            min-height: 47px !important;
        }
        :not(#keywordList):not(#panel)::-webkit-scrollbar-thumb:hover {
            background-color: #a68cd4 !important;
        }
        :not(#keywordList):not(#panel)::-webkit-scrollbar-track {
            background: transparent !important;
            border-radius: 8px !important;
        }
        :not(#keywordList):not(#panel)::-webkit-scrollbar-corner {
            background: transparent !important;
        }
    `);
} else {
    const style = document.createElement("style");
    style.textContent = `/* Те же стили, что выше */`;
    document.head.appendChild(style);
}

    document.head.appendChild(style);

    // Принудительное применение стилей скроллбара
const keywordList = document.getElementById("keywordList");
keywordList.style.scrollbarWidth = 'thin';
keywordList.style.scrollbarColor = '#c1a5ef #205151';



    // ========== Кнопка переключения панели 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 = "485px";
            }, 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 = "510px";
            }, 10);
        }
        isPanelVisible = !isPanelVisible;
        localStorage.setItem("panelVisible", isPanelVisible.toString());
    }

    toggleButton.addEventListener("click", togglePanel);

    if (isPanelVisible) {
        panel.style.height = "510px";
        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");

    // ========== Обработчики событий ========== //
// Обновим обработчик для addKeywordBtn
addKeywordBtn.addEventListener("click", () => {
    const inputValue = keywordInput.value.trim();
    if (inputValue) {
        if (displayMode === "favorites" && inputValue.startsWith("@")) {
            const username = inputValue.slice(1).toLowerCase();
            if (!favoriteUsers.includes(username)) {
                favoriteUsers.push(username);
                saveFavoriteUsers();
                updateKeywordList();
                updateCounters();
                debouncedHidePosts();
                showNotification(`Added @${username} to favorites`, 'success');
            } else {
                showNotification(`@${username} is already in favorites`, 'error');
            }
        } else if (displayMode === "blocked" && inputValue.startsWith("@")) {
            const username = inputValue.slice(1).toLowerCase();
            if (!blockedUsers.includes(username)) {
                blockedUsers.push(username);
                saveBlockedUsers();
                updateKeywordList();
                updateCounters();
                debouncedHidePosts();
                showNotification(`Blocked @${username}`, 'success');
            } else {
                showNotification(`@${username} is already blocked`, 'error');
            }
        } else if (displayMode === "keywords" && !hiddenKeywords.includes(inputValue)) {
            hiddenKeywords.push(inputValue);
            saveKeywords();
            updateKeywordList();
            updateCounters();
            debouncedHidePosts();
            showNotification(`Added keyword: ${inputValue}`, 'success');
        } else {
            showNotification(`Keyword ${inputValue} already exists`, 'error');
        }
        keywordInput.value = "";
    }
});


    toggleNonVerifiedBtn.textContent = `Hide non-verified accounts: ${hideNonVerifiedAccounts ? "Turn OFF" : "Turn ON"}`;
// Обновим обработчик для toggleNonVerifiedBtn
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();
            } else {
                hiddenKeywords = [];
                unblockedKeywords = [];
                saveKeywords();
                saveUnblockedKeywords();
            }
            updateKeywordList();
            updateCounters();
            hidePosts();
        }
    });

    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 = hiddenKeywords;
            baseName = "hidden_keywords";
            nameWord = "keywords";
        }
        const dataStr = `data:text/json;charset=utf-8,${encodeURIComponent(JSON.stringify(data))}`;
        const count = 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);
    });

    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 (Array.isArray(importedData)) {
                        if (displayMode === "favorites") {
                            favoriteUsers = [...new Set([...favoriteUsers, ...importedData.map(u => u.startsWith("@") ? u.slice(1).toLowerCase() : u.toLowerCase())])];
                            saveFavoriteUsers();
                        } else if (displayMode === "blocked") {
                            blockedUsers = [...new Set([...blockedUsers, ...importedData.map(u => u.startsWith("@") ? u.slice(1).toLowerCase() : u.toLowerCase())])];
                            saveBlockedUsers();
                        } else {
                            hiddenKeywords = [...new Set([...hiddenKeywords, ...importedData])];
                            saveKeywords();
                        }
                        updateKeywordList();
                        updateCounters();
                        hidePosts();
                    } else {
                        alert("Incorrect file format.");
                    }
                } catch (e) {
                    alert("Error reading the file.");
                }
            };
            reader.readAsText(file);
        });
        input.click();
    });

    toggleBlockBtn.addEventListener("click", () => {
        if (displayMode === "keywords") {
            if (hiddenKeywords.length > 0) {
                unblockedKeywords = [...hiddenKeywords];
                hiddenKeywords = [];
                toggleBlockBtn.textContent = "Block All";
            } else {
                hiddenKeywords = [...unblockedKeywords];
                unblockedKeywords = [];
                toggleBlockBtn.textContent = "Unblock All";
            }
            saveKeywords();
            saveUnblockedKeywords();
            updateKeywordList();
            updateCounters();
            hidePosts();
        }
    });

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

// Обновим обработчик для toggleVerifiedBtn
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";

    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 дер";
    keywordInput.placeholder = displayMode === "favorites" || displayMode === "blocked" ? "Enter @username" : "Enter the word";

    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 filteredUsers = favoriteUsers.filter(user => user.toLowerCase().includes(searchQuery));
        if (filteredUsers.length === 0) {
            list.textContent = searchQuery ? "No matches found" : "Нет";
        } else {
            filteredUsers.forEach((user) => {
                const listItem = document.createElement("li");
                listItem.textContent = `@${user}`;

                const deleteButton = document.createElement("button");
                deleteButton.textContent = "❌";
                deleteButton.style.backgroundColor = "#f44336";
                deleteButton.style.color = "#fff";
                deleteButton.style.border = "none";
                deleteButton.style.borderRadius = "3px";
                deleteButton.style.cursor = "pointer";
                deleteButton.addEventListener("click", () => {
                    favoriteUsers.splice(favoriteUsers.indexOf(user), 1);
                    saveFavoriteUsers();
                    updateKeywordList();
                    updateCounters();
                    hidePosts();
                    showNotification(`Removed @${user} from favorites`, 'success');
                });

                listItem.appendChild(deleteButton);
                list.appendChild(listItem);
            });
        }
    } else if (displayMode === "blocked") {
        label.textContent = "Blocked Users";
        const filteredUsers = blockedUsers.filter(user => user.toLowerCase().includes(searchQuery));
        if (filteredUsers.length === 0) {
            list.textContent = searchQuery ? "No matches found" : "Нет";
        } else {
            filteredUsers.forEach((user) => {
                const listItem = document.createElement("li");
                listItem.textContent = `@${user}`;

                const deleteButton = document.createElement("button");
                deleteButton.textContent = "❌";
                deleteButton.style.backgroundColor = "#f44336";
                deleteButton.style.color = "#fff";
                deleteButton.style.border = "none";
                deleteButton.style.borderRadius = "3px";
                deleteButton.style.cursor = "pointer";
                deleteButton.addEventListener("click", () => {
                    blockedUsers.splice(blockedUsers.indexOf(user), 1);
                    saveBlockedUsers();
                    updateKeywordList();
                    updateCounters();
                    hidePosts();
                    showNotification(`Unblocked @${user}`, 'success');
                });

                listItem.appendChild(deleteButton);
                list.appendChild(listItem);
            });
        }
    } else {
        label.textContent = "List of Keywords";
        const filteredKeywords = hiddenKeywords.filter(keyword => keyword.toLowerCase().includes(searchQuery));
        if (filteredKeywords.length === 0) {
            list.textContent = searchQuery ? "No matches found" : "Нет";
        } else {
            filteredKeywords.forEach((keyword) => {
                const listItem = document.createElement("li");
                listItem.textContent = keyword;

                const deleteButton = document.createElement("button");
                deleteButton.textContent = "❌";
                deleteButton.style.backgroundColor = "#f44336";
                deleteButton.style.color = "#fff";
                deleteButton.style.border = "none";
                deleteButton.style.borderRadius = "3px";
                deleteButton.style.cursor = "pointer";
                deleteButton.addEventListener("click", () => {
                    hiddenKeywords.splice(hiddenKeywords.indexOf(keyword), 1);
                    saveKeywords();
                    updateKeywordList();
                    updateCounters();
                    hidePosts();
                    showNotification(`Removed keyword: ${keyword}`, 'success');
                });

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



    // ========== Создание кнопки избранного в профиле ========== //
    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 = "510px";
            }
            toggleButton.style.display = "flex";
        }
    }

    // ========== Наблюдение за изменениями DOM ========== //
    const observer = new MutationObserver(() => {
        debouncedHidePosts();
    });
    observer.observe(document.body, { childList: true, subtree: true });

    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");
});


})();