【中羽在线】新闻过滤与自动展开

多合一功能脚本:1. 过滤低评论新闻 2. 过滤疑似广告 3. 智能后台加载(5次/轮) 4. 新标签页打开新闻详情。5. 光标悬停新闻卡片显示手型

// ==UserScript==
// @name         【中羽在线】新闻过滤与自动展开
// @namespace   https://github.com/realSilasYang
// @version         2025-9-13
// @description    多合一功能脚本:1. 过滤低评论新闻 2. 过滤疑似广告 3. 智能后台加载(5次/轮)  4. 新标签页打开新闻详情。5. 光标悬停新闻卡片显示手型
// @author          阳熙来
// @match          https://www.badmintoncn.com/*
// @icon            
// @license      GNU GPLv3
// @grant        GM_openInTab
// @grant        GM_addStyle
// @run-at       document-start
// ==/UserScript==

(function () {
    'use strict';

    // --------------------------------------------------------------------------------
    // 模块0: 添加自定义样式
    // 作用:让新闻列表和热点新闻在鼠标悬停时显示为可点击状态
    // --------------------------------------------------------------------------------
    GM_addStyle(`
        .news_list, .newst.hot {
            cursor: pointer;
        }
    `);

    // --------------------------------------------------------------------------------
    // 模块1: 新闻过滤 (评论数)
    // 作用:只保留评论数 >= 20 的新闻,低于此数的直接移除
    // --------------------------------------------------------------------------------
    function shouldKeep(box) {
        // 找到评论图标
        const pjImg = box.querySelector('img.news_pj');
        if (!pjImg) return false;

        // 评论数在图标的下一个文本节点
        const nextNode = pjImg.nextSibling;
        if (!nextNode || nextNode.nodeType !== 3 || !nextNode.nodeValue) return false;

        // 转换为数字
        const commentNum = parseInt(nextNode.nodeValue.trim(), 10);

        // 保留评论数 >= 20 的新闻
        return !isNaN(commentNum) && commentNum >= 20;
    }

    // 过滤单条新闻
    function filterSingle(box) {
        if (!shouldKeep(box)) {
            box.remove();
        }
    }

    // 过滤整个页面的新闻
    function filterAll(root = document) {
        root.querySelectorAll('.news_list').forEach(filterSingle);
    }

    // 监听 DOM 变化,动态过滤新加载的新闻
    const newsObserver = new MutationObserver(mutations => {
        for (const m of mutations) {
            for (const node of m.addedNodes) {
                if (node.nodeType !== 1) continue; // 只处理元素节点
                if (node.classList && node.classList.contains('news_list')) {
                    filterSingle(node);
                } else {
                    filterAll(node);
                }
            }
        }
    });

    // --------------------------------------------------------------------------------
    // 模块2: 过滤疑似广告内容
    // 作用:移除包含特定广告关键词的新闻
    // --------------------------------------------------------------------------------
    function filterExclamationNews() {
        const mainNewsContainer = document.querySelector('div.news.main');
        if (!mainNewsContainer) return;

        // 定义广告关键词(大小写不敏感)
        const adKeywords = [
            '!', '品牌', '胜利', 'VICTOR', 'YONEX', '尤尼克斯', '李宁', '薰风', '薰',
            'KUMPOOO', '川崎', 'Kawasaki', '得物', '波力', 'BONNY', '极光', '亚瑟士',
            'ASICS', '耐克', '欧击', '蟹羽', '球拍', '球鞋', '羽鞋', '球线', '新品',
            '产品', '上市', '发布', '评测', '测评', '赏析', '试打', '试穿', '上手',
            '限量', '抢先', '纪念', '隆重', '报名', '签约', '代言'
        ];

        // 转换为正则(忽略大小写)
        const adRegex = new RegExp(adKeywords.join('|'), 'i');

        // 遍历新闻列表
        const listItems = mainNewsContainer.querySelectorAll('li');
        listItems.forEach(li => {
            if (li.textContent && adRegex.test(li.textContent)) {
                li.remove(); // 删除匹配的新闻
            }
        });
    }

    // --------------------------------------------------------------------------------
    // 模块3: 混合模式自动点击展开
    // 作用:自动点击“加载更多”按钮,最多点击5次
    // --------------------------------------------------------------------------------
    let isInitialLoad = true;

    // 延迟函数
    function sleep(ms) {
        return new Promise(resolve => setTimeout(resolve, ms));
    }

    // 批量点击“加载更多”
    async function performClickBatch() {
        console.log('开始新一轮自动加载(最多5次)...');
        const CLICK_LIMIT = 5;
        const CLICK_DELAY = 500; // 每次点击间隔 0.5 秒

        for (let i = 0; i < CLICK_LIMIT; i++) {
            const currentButton = document.getElementById('loadingButton');
            if (currentButton && currentButton.offsetParent !== null) {
                currentButton.click();
                console.log(`后台自动点击第 ${i + 1} 次。`);
                await sleep(CLICK_DELAY);
            } else {
                console.log('“加载更多”按钮消失,提前结束本轮。');
                break;
            }
        }
        console.log('本轮点击完成。');
    }

    // 监听“加载更多”按钮是否进入视口
    const buttonObserver = new IntersectionObserver(async (entries) => {
        const buttonEntry = entries[0];
        if (!isInitialLoad && buttonEntry.isIntersecting) {
            console.log('检测到您已滚动到底部,再次启动自动加载...');
            const button = buttonEntry.target;
            buttonObserver.unobserve(button);
            await performClickBatch();

            const buttonAfterClick = document.getElementById('loadingButton');
            if (buttonAfterClick) {
                 console.log('加载完成,进入“待命”状态,等待您下一次滚动...');
                 buttonObserver.observe(buttonAfterClick);
            } else {
                 console.log('所有内容已加载完毕。');
            }
        }
    }, { threshold: 0.1 });

    // --------------------------------------------------------------------------------
    // 模块4: 点击新闻区域打开新标签页
    // 作用:点击新闻列表项时,在新标签页打开新闻详情
    // --------------------------------------------------------------------------------
    function handleNewsClick(event) {
        const newsList = event.target.closest('.news_list');
        if (!newsList) return;

        // 如果点击的是标签(如分类标签),则不触发
        if (event.target.closest('.gray_label')) return;

        // 获取新闻 ID
        const newsIdElement = newsList.querySelector('.newst');
        if (!newsIdElement) return;

        const newsId = newsIdElement.getAttribute('title');
        if (newsId) {
            event.preventDefault();
            event.stopPropagation();
            const url = `https://www.badmintoncn.com/newsm.php?a=view&id=${newsId}&mag_hide_progress=1`;
            console.log(`打开新标签页: ${url}`);
            GM_openInTab(url, { active: true });
        }
    }

    // --------------------------------------------------------------------------------
    // 模块5: 脚本初始化
    // --------------------------------------------------------------------------------
    document.addEventListener('DOMContentLoaded', async () => {
        // 1. 启动两种新闻过滤
        filterAll(); // 过滤低评论新闻
        filterExclamationNews(); // 过滤疑似广告新闻
        newsObserver.observe(document.documentElement, { childList: true, subtree: true });

        // 2. 启动点击打开新标签页功能
        document.body.addEventListener('click', handleNewsClick, true);

        // 3. 启动混合模式加载
        let loadButton = document.getElementById('loadingButton');
        while (!loadButton) {
            await sleep(500);
            loadButton = document.getElementById('loadingButton');
        }

        if (isInitialLoad) {
            await performClickBatch();
            isInitialLoad = false;
        }

        const buttonToObserve = document.getElementById('loadingButton');
        if (buttonToObserve) {
             console.log('首次加载完成,进入“待命”状态,等待您滚动到底部...');
             buttonObserver.observe(buttonToObserve);
        }
    });
})();