Greasy Fork

Greasy Fork is available in English.

净化评论区

PEACE AND LOVE,跟恶臭言论说拜拜。

// ==UserScript==
// @name         净化评论区
// @namespace    http://tampermonkey.net/
// @version      0.3
// @description  PEACE AND LOVE,跟恶臭言论说拜拜。
// @author       Young2fun
// @match        *://*/*
// @grant        none
// @icon         
// @license      GPL-3.0 License
// ==/UserScript==

(function () {
    'use strict';

    // 规则:domain 可以是正则,selectors 为数组(一个站点可能多个选择器)
    const RULES = [
        { domain: /baijiahao\.baidu\.com/, selectors: ['#commentModule', '.comment-module'] },
        { domain: /baidu\.com\/s/, selectors: ['#danmuWrapper'] },
        { domain: /qq\.com/, selectors: ['#qqcom-comment', '.comment'] },
        { domain: /haokan\.baidu\.com/, selectors: ['#commentnum'] },
        { domain: /mbd\.baidu\.com\/newspage\//, selectors: ['#commentModule'] },
        { domain: /mbd\.baidu\.com/, selectors: ['#page-comment'] },
        // 你可以在这里继续添加规则:{ domain: /example\.com/, selectors: ['#c1', '.c2'] }
    ];

    const rule = RULES.find(r => r.domain.test(location.href));
    if (!rule) {
        // 没匹配到站点规则则退出(若想在所有站点尝试,可注释掉此 return)
        // return;
        console.log('[隐藏评论区] 本页面未匹配到规则,仍将尝试泛匹配常见选择器。');
    } else {
        console.log('[隐藏评论区] 匹配站点规则:', rule.domain, 'selectors:', rule.selectors);
    }

    // 如果没有匹配规则,使用一组常见的选择器作泛匹配
    const fallbackSelectors = [
        '#comments', '#comment', '.comments', '.comment-list', '.comment-area',
        '.reply-area', '.comment-module', '#commentModule', '#page-comment', '#commentnum',
        '#qqcom-comment', '#danmuWrapper'
    ];
    const selectors = (rule && rule.selectors && rule.selectors.length) ? rule.selectors.concat(fallbackSelectors) : fallbackSelectors;

    // 注入全局 CSS(尽可能立即生效)
    function injectGlobalStyle(selArr) {
        try {
            const sheet = document.createElement('style');
            sheet.setAttribute('data-name', 'hide-comments-style');
            const rulesText = selArr.map(s => `${s} { display: none !important; visibility: hidden !important; height: 0 !important; margin: 0 !important; padding: 0 !important; overflow: hidden !important; }`).join('\n');
            sheet.textContent = rulesText;
            (document.head || document.documentElement).appendChild(sheet);
            console.log('[隐藏评论区] 已注入全局 CSS(主文档)');
        } catch (e) {
            console.warn('[隐藏评论区] 注入全局 CSS 失败:', e);
        }
    }

    // 在给定根节点(document 或 shadow/root)中查找并隐藏选择器
    function hideInRoot(root, selArr) {
        try {
            selArr.forEach(sel => {
                const list = (root || document).querySelectorAll(sel);
                list.forEach(el => {
                    if (el) {
                        if (el.style) {
                            el.style.setProperty('display', 'none', 'important');
                            el.style.setProperty('visibility', 'hidden', 'important');
                        } else {
                            // 对于一些非元素节点(保险)
                            el.setAttribute && el.setAttribute('hidden', 'true');
                        }
                        console.log('[隐藏评论区] 隐藏元素:', sel, el);
                    }
                });
            });
        } catch (e) {
            console.warn('[隐藏评论区] hideInRoot 错误: ', e);
        }
    }

    // 递归遍历 open shadow roots,尝试在 shadowRoot 中 hide
    function traverseShadowAndHide(rootNode, selArr) {
        if (!rootNode) return;
        const treeWalker = document.createTreeWalker(rootNode, NodeFilter.SHOW_ELEMENT, null, false);
        let node;
        const visited = new Set();
        while ((node = treeWalker.nextNode())) {
            // 防止死循环
            if (visited.has(node)) break;
            visited.add(node);

            // 如果节点有 shadowRoot(open),在其中执行 hide
            try {
                const sr = node.shadowRoot;
                if (sr) {
                    hideInRoot(sr, selArr);
                    // 继续递归遍历 shadow root 的子节点
                    traverseShadowAndHide(sr, selArr);
                }
            } catch (e) {
                // 有些 shadowRoot 可能不可访问或 closed
            }
        }
    }

    // 处理同源 iframe:注入 CSS 并隐藏其中匹配元素;对跨域 iframe:若 iframe.src 匹配规则域则直接隐藏 iframe 元素
    function handleIframes(selArr, domainPatterns) {
        const iframes = document.querySelectorAll('iframe');
        iframes.forEach(iframe => {
            try {
                const doc = iframe.contentDocument;
                if (doc && doc.head) {
                    // 在 iframe 中注入样式
                    const style = doc.createElement('style');
                    style.textContent = selArr.map(s => `${s} { display: none !important; visibility: hidden !important; height: 0 !important; overflow: hidden !important; }`).join('\n');
                    doc.head.appendChild(style);
                    // 直接尝试隐藏
                    selArr.forEach(s => {
                        doc.querySelectorAll(s).forEach(el => {
                            el.style && el.style.setProperty('display', 'none', 'important');
                            console.log('[隐藏评论区] 在同源 iframe 中隐藏:', s, el, iframe.src);
                        });
                    });
                } else {
                    // contentDocument 不可访问 -> 跨域 iframe
                    const src = iframe.getAttribute('src') || '';
                    if (domainPatterns && domainPatterns.some(p => p.test(src) || p.test(location.href))) {
                        // 如果 iframe 的 src 含有目标域名,直接隐藏 iframe 元素本身
                        iframe.style.setProperty('display', 'none', 'important');
                        console.log('[隐藏评论区] 隐藏跨域 iframe 元素(通过 src 匹配):', src);
                    } else {
                        // 若没有明确匹配,也可以选择隐藏 iframe(注释掉以避免误伤)
                        // iframe.style.display = 'none';
                    }
                }
            } catch (e) {
                // 访问跨域 iframe 会抛错
                const src = iframe.getAttribute('src') || '';
                if (domainPatterns && domainPatterns.some(p => p.test(src) || p.test(location.href))) {
                    iframe.style.setProperty('display', 'none', 'important');
                    console.log('[隐藏评论区] 跨域 iframe,已根据 src 隐藏:', src);
                } else {
                    // 无法访问也未匹配,跳过
                }
            }
        });
    }

    // 主运行逻辑:注入样式 -> 首次隐藏 -> 观察 DOM -> 处理 shadow & iframe -> 周期重试(小次数)
    function start() {
        injectGlobalStyle(selectors);
        hideInRoot(document, selectors);
        traverseShadowAndHide(document, selectors);
        handleIframes(selectors, RULES.map(r => r.domain));

        // MutationObserver 监听后续插入
        const observer = new MutationObserver(mutations => {
            // 当 DOM 变化时再次尝试隐藏
            hideInRoot(document, selectors);
            traverseShadowAndHide(document, selectors);
            handleIframes(selectors, RULES.map(r => r.domain));
        });
        if (document.body) {
            observer.observe(document.body, { childList: true, subtree: true });
            console.log('[隐藏评论区] 已启动 MutationObserver');
        } else {
            console.warn('[隐藏评论区] document.body 尚未就绪,稍后建立 observer');
            window.addEventListener('load', () => {
                observer.observe(document.body, { childList: true, subtree: true });
            });
        }

        // 周期性尝试(兼容某些站点在页面运行后很久才加载评论)
        let attempts = 0;
        const maxAttempts = 12; // 总共尝试约 12 次(每 2s -> 24s)
        const intervalId = setInterval(() => {
            attempts++;
            hideInRoot(document, selectors);
            traverseShadowAndHide(document, selectors);
            handleIframes(selectors, RULES.map(r => r.domain));
            if (attempts >= maxAttempts) {
                clearInterval(intervalId);
                console.log('[隐藏评论区] 周期重试结束,共尝试次数:', attempts);
            }
        }, 2000);
    }

    // 等待页面 load 更稳妥
    if (document.readyState === 'complete' || document.readyState === 'interactive') {
        start();
    } else {
        window.addEventListener('DOMContentLoaded', start);
        // 以及保险的 load
        window.addEventListener('load', start);
    }

})();