// ==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);
}
});
})();