您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
多合一功能脚本: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); } }); })();