// ==UserScript==
// @name PT Medal Monitor
// @namespace http://tampermonkey.net/
// @version 2025-01-08
// @description 监控所有站点的勋章状况
// @author Schalkiii
// @match http*://*/medal*.php*
// @match http*://*/badge*.php*
// @icon 
// @grant GM_xmlhttpRequest
// @license MIT
// ==/UserScript==
// Changelog 2025-01-03:
// 添加、修改了一些站点和match规则
// Changelog 2024-12-29:
// 右上角添加按钮,添加只显示有背景色行功能,即只保留最关键的可领取和未来才可领取的勋章。
// Changelog 2024-12-28:
// 右上角添加按钮,点击才运行。
// 抓取结果展示添加按钮跳转方便领取。
// 可领取勋章背景绿色高亮。
// 未来才开始领取的勋章背景橙色高亮。
// 更新、移除部分网址。
// Changelog 2024-12-27-1:
// 修复手续费抓取。
(function() {
'use strict';
// 定义站点配置
const SITES = [
{ name: '1PTBA', url: 'https://1ptba.com/medal.php' },
{ name: '52PT', url: 'https://52pt.site/medal.php' },
{ name: 'Audiences', url: 'https://audiences.me/medal.php' },
{ name: 'BTSchool', url: 'https://pt.btschool.club/medal.php' },
{ name: 'BYR', url: 'https://byr.pt/medal.php' },
{ name: 'CHDBits', url: 'https://ptchdbits.co/medal.php' },
{ name: 'CMCT', url: 'https://springsunday.net/badges.php' },
{ name: 'CarPt', url: 'https://carpt.net/medal.php' },
{ name: 'DiscFan', url: 'https://discfan.net/medal.php' },
{ name: 'Dragon', url: 'https://www.dragonhd.xyz/medal.php' },
{ name: 'FreeFarm', url: 'https://pt.0ff.cc/medal.php' },
{ name: 'GPW', url: 'https://greatposterwall.com/medal.php' },
{ name: 'HDArea', url: 'https://hdarea.club/medal.php' },
{ name: 'HDAtmos', url: 'https://hdatmos.club/medal.php' },
{ name: 'HDCity', url: 'https://hdcity.city/medal.php' },
{ name: 'HDDolby', url: 'https://www.hddolby.com/medals.php' },
{ name: 'HDHome', url: 'https://hdhome.org/medal.php' },
{ name: 'HDPost', url: 'https://pt.hdpost.top/medal.php' },
{ name: 'HDPt', url: 'https://hdpt.xyz/medal.php' },
{ name: 'HDSky', url: 'https://hdsky.me/medal.php' },
{ name: 'HDSpace', url: 'https://hd-space.org/medal.php' },
{ name: 'HDTime', url: 'https://hdtime.org/medal.php' },
{ name: 'HDU', url: 'https://pt.hdupt.com/medal.php' },
{ name: 'HDVideo', url: 'https://hdvideo.one/medal.php' },
{ name: 'HDfans', url: 'http://hdfans.org/medal.php' },
{ name: 'HITPT', url: 'https://www.hitpt.com/medal.php' },
{ name: 'HUDBT', url: 'https://hudbt.hust.edu.cn/medal.php' },
{ name: 'HaiDan', url: 'https://www.haidan.video/medal.php' },
{ name: 'HaresClub', url: 'https://club.hares.top/medal.php' },
{ name: 'ITZMX', url: 'https://pt.itzmx.com/medal.php' },
{ name: 'JoyHD', url: 'https://www.joyhd.net/medal.php' },
{ name: 'NanYang', url: 'https://nanyangpt.com/medal.php' },
{ name: 'Oshen', url: 'http://www.oshen.win/medal.php' },
{ name: 'OurBits', url: 'https://ourbits.club/medal.php' },
{ name: 'PTT', url: 'https://www.pttime.org/medal.php' },
{ name: 'PThome', url: 'https://www.pthome.net/medal.php' },
{ name: 'PTsbao', url: 'https://ptsbao.club/medal.php' },
{ name: 'SoulVoice', url: 'https://pt.soulvoice.club/medal.php' },
{ name: 'TCCF', url: 'https://et8.org/medal.php' },
{ name: 'TJUPT', url: 'https://www.tjupt.org/medal.php' },
{ name: 'TLFbits', url: 'https://pt.eastgame.org/medal.php' },
{ name: 'TTG', url: 'https://totheglory.im/medal.php' },
{ name: 'WT-Sakura', url: 'https://wintersakura.net/medal.php' },
{ name: 'HHClub', url: 'https://hhanclub.top/medal.php' },
{ name: 'Monika', url: 'https://monikadesign.uk/medal.php' },
{ name: '红叶', url: 'https://leaves.red/medal.php' },
{ name: 'ICC', url: 'https://www.icc2022.com/medal.php' },
{ name: 'CyanBug', url: 'https://cyanbug.net/medal.php' },
{ name: 'ZHUQUE', url: 'https://zhuque.in/medal.php' },
{ name: 'DaJiao', url: 'https://dajiao.cyou/medal.php' },
{ name: '海棠', url: 'https://www.htpt.cc/medal.php' },
{ name: '杏林', url: 'https://xingtan.one/medal.php' },
{ name: 'SRVFI', url: 'https://srvfi.top/medal.php' },
{ name: 'OKPT', url: 'https://www.okpt.net/medal.php' },
{ name: 'GGPT', url: 'https://www.gamegamept.com/medal.php' },
{ name: 'RS', url: 'https://resource.xidian.edu.cn/medal.php' },
{ name: 'Panda', url: 'https://pandapt.net/medal.php' },
{ name: 'KuFei', url: 'https://kufei.org/medal.php' },
{ name: 'RouSi', url: 'https://rousi.zip/medal.php' },
{ name: '悟空', url: 'https://wukongwendao.top/medal.php' },
{ name: 'PTCafe', url: 'https://ptcafe.club/medal.php' },
{ name: 'PTChina', url: 'https://ptchina.org/medal.php' },
{ name: 'GTK', url: 'https://pt.gtkpw.xyz/medal.php' },
{ name: '麒麟', url: 'https://www.hdkyl.in/medal.php' },
{ name: 'AGSV', url: 'https://www.agsvpt.com/medal.php' },
{ name: 'ECUST', url: 'https://pt.ecust.pp.ua/medal.php' },
{ name: 'iloli', url: 'https://share.ilolicon.com/medal.php' },
{ name: 'CrabPt', url: 'https://crabpt.vip/medal.php' },
{ name: 'QingWa', url: 'https://qingwapt.com/medal.php' },
{ name: 'FNP', url: 'https://fearnopeer.com/medal.php' },
{ name: 'YemaPT', url: 'https://www.yemapt.org/medal.php' },
{ name: 'PTFans', url: 'https://ptfans.cc/medal.php' },
{ name: '影', url: 'https://star-space.net/medal.php' },
{ name: 'PTzone', url: 'https://ptzone.xyz/medal.php' },
{ name: '雨', url: 'https://raingfh.top/medal.php' },
{ name: 'PTLGS', url: 'https://ptlgs.org/medal.php' },
//{ name: 'NJTUPT', url: 'https://njtupt.top/medal.php' },
{ name: 'LemonHD', url: 'https://lemonhd.club/medal.php' },
{ name: 'FRDS', url: 'https://pt.keepfrds.com/medal.php' },
{ name: 'U2', url: 'https://u2.dmhy.org/medal.php' },
{ name: 'jpop', url: 'https://jpopsuki.eu/medal.php' },
{ name: 'HONE', url: 'https://hawke.uno/medal.php' },
{ name: 'Ubits', url: 'https://ubits.club/medal.php' },
{ name: 'Piggo', url: 'https://piggo.me/medal.php' },
{ name: 'KamePT', url: 'https://kamept.com/medal.php' }
];
const usupportUrls = [
{ name: 'PTer', url: 'https://pterclub.com/medal.php' },
{ name: 'ZMPT', url: 'https://zmpt.cc/medal.php' },
{ name: '象岛', url: 'https://ptvicomo.net/medal.php' }
];
// Medal.php 404的站点,可能是站点没有勋章功能
const errorUrls = [
"https://52pt.site/medal.php",
"https://audiences.me/medal.php",
"https://pt.btschool.club/medal.php",
"https://ptchdbits.co/medal.php",
"https://www.dragonhd.xyz/medal.php",
"https://hdarea.club/medal.php",
"https://hdcity.city/medal.php",
"https://hdhome.org/medal.php",
"https://pt.hdpost.top/medal.php",
"https://hdsky.me/medal.php",
"https://hd-space.org/medal.php",
"https://pt.hdupt.com/medal.php",
"https://www.hitpt.com/medal.php",
"https://hudbt.hust.edu.cn/medal.php",
"https://www.haidan.video/medal.php",
"https://club.hares.top/medal.php",
"https://pt.itzmx.com/medal.php",
"https://www.joyhd.net/medal.php",
"https://nanyangpt.com/medal.php",
"https://ourbits.club/medal.php",
"https://www.pthome.net/medal.php",
"https://ptsbao.club/medal.php",
"https://et8.org/medal.php",
"https://totheglory.im/medal.php",
"https://monikadesign.uk/medal.php",
"https://www.htpt.cc/medal.php",
"https://fearnopeer.com/medal.php",
"https://star-space.net/medal.php",
"https://lemonhd.club/medal.php",
"https://pt.keepfrds.com/medal.php",
"https://u2.dmhy.org/medal.php",
"https://jpopsuki.eu/medal.php",
"https://pt.eastgame.org/medal.php",
"https://www.tjupt.org/medal.php",
"https://hdvideo.one/medal.php",
"https://byr.pt/medal.php",
"https://resource.xidian.edu.cn/medal.php",
"https://hawke.uno/medal.php"
];
const newSITES = SITES.filter(site => {
return!errorUrls.includes(site.url);
});
//console.log(newSITES);
// 在页面上展示表格
function displayTableOnPage(medals, siteName, siteUrl) {
// 创建表格容器
const container = document.createElement('div');
container.style.margin = '20px';
container.style.padding = '20px';
container.style.border = '1px solid #ccc';
container.style.borderRadius = '5px';
container.style.backgroundColor = '#f9f9f9';
// 创建站点标题
const siteHeader = document.createElement('h2');
siteHeader.textContent = `${siteName} (一共 ${medals.length} 个勋章)`;
// 创建链接
const link = document.createElement('a');
link.href = siteUrl; // 设置链接的 URL
link.textContent = '访问站点'; // 链接显示的文本
link.style.color = '#007bff'; // 设置链接颜色
link.style.textDecoration = 'none'; // 去掉下划线
link.style.marginLeft = '10px'; // 添加左边距
link.style.fontWeight = 'bold'; // 加粗字体
// 添加点击事件,确保链接在新标签页打开
link.addEventListener('click', function (event) {
event.preventDefault(); // 阻止默认行为
window.open(siteUrl, '_blank'); // 在新标签页打开链接
});
// 将链接添加到标题中
siteHeader.appendChild(link);
// 将标题添加到容器中
container.appendChild(siteHeader);
// 创建表格
const table = document.createElement('table');
table.style.width = '100%';
table.style.borderCollapse = 'collapse';
table.style.marginBottom = '20px';
// 创建表头
const thead = document.createElement('thead');
const headerRow = document.createElement('tr');
const headers = ['图片', '名称', '描述', '可购买时间', '有效期', '奖励倍数', '价格', '库存', '可购买', '手续费'];
headers.forEach(headerText => {
const th = document.createElement('th');
th.textContent = headerText;
th.style.border = '1px solid #ddd';
th.style.padding = '8px';
th.style.backgroundColor = '#f2f2f2';
headerRow.appendChild(th);
});
thead.appendChild(headerRow);
table.appendChild(thead);
// 创建表格内容
const tbody = document.createElement('tbody');
medals.forEach(medal => {
const row = document.createElement('tr');
// 如果勋章可购买,则高亮显示该行
if (medal.canBuy) {
row.style.backgroundColor = '#e6ffe6'; // 使用浅绿色背景高亮
row.style.fontWeight = 'bold'; // 加粗字体
}
// 检查可购买时间的开始点是否在未来
const availableTime = medal.availableTime;
if (availableTime && availableTime.trim() !== '不限 ~ 不限') {
const startTimeStr = availableTime.split('~')[0].trim(); // 获取开始时间部分
if (startTimeStr !== '不限') {
const startTime = new Date(startTimeStr); // 将字符串转换为日期对象
const now = new Date(); // 获取当前时间
if (startTime > now) { // 如果开始时间在未来
row.style.backgroundColor = '#ffe6cc'; // 使用橙色背景高亮
row.style.fontWeight = 'bold'; // 加粗字体
}
}
}
const columns = [
`<img src="${medal.image}" width="50" height="50">`,
medal.name,
medal.description,
medal.availableTime,
medal.validity,
medal.bonusMultiplier,
medal.price,
medal.stock,
medal.canBuy ? '是' : '否',
medal.giftFee
];
columns.forEach(columnText => {
const td = document.createElement('td');
td.innerHTML = columnText;
td.style.border = '1px solid #ddd';
td.style.padding = '8px';
row.appendChild(td);
});
tbody.appendChild(row);
});
table.appendChild(tbody);
// 将表格添加到容器中
container.appendChild(table);
// 将容器添加到页面中
document.body.appendChild(container);
}
// 从HTML字符串中提取勋章数据
function extractMedalsFromHtml(html, siteName, siteUrl) {
const parser = new DOMParser();
const doc = parser.parseFromString(html, 'text/html');
const medals = [];
// 查找包含勋章数据的表格
const tables = doc.querySelectorAll('table');
let medalTable = null;
// 查找包含正确列头的表格
for (const table of tables) {
const headers = table.querySelectorAll('.colhead');
if (headers.length >= 7) {
// 检查是否包含简体或繁体的“购买”或“价格”字样
const hasBuyOrPriceInHeader = Array.from(headers).some(header => {
const text = header.textContent;
return (text.includes('购买') || text.includes('購買') || text.includes('价格') || text.includes('價格'));
});
if (hasBuyOrPriceInHeader) {
medalTable = table;
break;
}
}
}
if (!medalTable) {
console.error(`[${siteName}] 未找到勋章表格`);
return medals;
}
// 处理勋章表格中的行
medalTable.querySelectorAll('tbody > tr').forEach(row => {
// Skip header row
if (!row.querySelector('.colhead')) {
// 确保行中有图片和数据
const imgElement = row.querySelector('td:nth-child(1) img');
const nameElement = row.querySelector('td:nth-child(2) h1');
if (imgElement && nameElement) {
const medal = {
site: siteName,
image: imgElement.src || '',
name: nameElement.textContent.trim() || '',
description: row.querySelector('td:nth-child(2)')?.textContent.trim().replace(nameElement.textContent, '').trim() || '',
availableTime: row.querySelector('td:nth-child(3)')?.textContent.trim().replace(/\s+/g, ' ') || '',
validity: row.querySelector('td:nth-child(4)')?.textContent.trim() || '',
bonusMultiplier: row.querySelector('td:nth-child(5)')?.textContent.trim() || '',
price: parseInt(row.querySelector('td:nth-child(6)')?.textContent.replace(/,/g, ''), 10) || 0,
stock: row.querySelector('td:nth-child(7)')?.textContent.trim() || '',
canBuy:!row.querySelector('td:nth-child(8) input')?.disabled,
giftFee: row.querySelector('td:nth-child(9) span.nowrap')?.textContent.replace('手續費: ', '').replace('手续费: ', '') || ''
};
medals.push(medal);
}
}
});
console.group(`${siteName} (一共 ${medals.length} 个勋章)`);
console.table(medals);
console.groupEnd();
// 在页面上展示表格
displayTableOnPage(medals,siteName,siteUrl);
return medals;
}
// 从单个站点抓取勋章
async function scrapeSite(site) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
method: "GET",
url: site.url,
headers: {
"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8",
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36"
},
onload: function (response) {
if (response.status === 200) {
const medals = extractMedalsFromHtml(response.responseText, site.name, site.url);
resolve(medals);
} else {
console.log(response);
reject(new Error(`HTTP error! status: ${response.status}`));
}
},
onerror: function (error) {
console.error(`[${site.name}] 抓取失败:`, error);
reject(error);
}
});
});
}
// 抓取所有站点的勋章
async function scrapeAllSites() {
try {
// 并行请求所有站点
const allMedals = await Promise.allSettled(
newSITES.map(site => scrapeSite(site))
);
// 合并所有结果
const flatMedals = allMedals.flat();
// 按站点分组显示结果
const groupedMedals = {};
flatMedals.forEach(medal => {
if (!groupedMedals[medal.site]) {
groupedMedals[medal.site] = [];
}
groupedMedals[medal.site].push(medal);
});
// 打印结果
for (const [site, medals] of Object.entries(groupedMedals)) {
console.group(`${site} (共有 ${medals.length} 个勋章)`);
console.table(medals);
console.groupEnd();
}
return flatMedals;
} catch (error) {
console.error('抓取过程出错:', error);
throw error;
}
}
window.addEventListener('load', function () {
// 创建按钮容器
const buttonContainer = document.createElement('div');
buttonContainer.style.position = 'fixed'; // 固定在页面显眼位置
buttonContainer.style.top = '20px';
buttonContainer.style.right = '20px';
buttonContainer.style.display = 'flex'; // 使用 Flex 布局
buttonContainer.style.gap = '10px'; // 按钮之间的间距
buttonContainer.style.zIndex = '1000'; // 确保按钮在最上层
// 创建“开始抓取勋章”按钮
const scrapeButton = document.createElement('button');
scrapeButton.textContent = '开始抓取勋章';
scrapeButton.style.padding = '10px 20px';
scrapeButton.style.backgroundColor = '#007bff';
scrapeButton.style.color = '#fff';
scrapeButton.style.border = 'none';
scrapeButton.style.borderRadius = '5px';
scrapeButton.style.cursor = 'pointer';
// 创建“隐藏无背景色行”按钮
const hideButton = document.createElement('button');
hideButton.textContent = '隐藏无背景色行';
hideButton.style.padding = '10px 20px';
hideButton.style.backgroundColor = '#28a745'; // 绿色背景
hideButton.style.color = '#fff';
hideButton.style.border = 'none';
hideButton.style.borderRadius = '5px';
hideButton.style.cursor = 'pointer';
// 将按钮添加到按钮容器中
buttonContainer.appendChild(scrapeButton);
buttonContainer.appendChild(hideButton);
// 将按钮容器添加到页面中
document.body.appendChild(buttonContainer);
// “开始抓取勋章”按钮点击事件
scrapeButton.addEventListener('click', function () {
scrapeButton.textContent = '抓取中...'; // 点击后改变按钮文字
scrapeButton.disabled = true; // 禁用按钮,防止重复点击
scrapeAllSites()
.then(medals => {
console.log('所有站点抓取完成,总共抓取到', medals.length, '个勋章');
scrapeButton.textContent = '抓取完成'; // 抓取完成后更新按钮文字
})
.catch(error => {
console.error('抓取过程出错:', error);
scrapeButton.textContent = '抓取出错,重试'; // 出错后更新按钮文字
scrapeButton.disabled = false; // 允许用户重试
});
});
// “隐藏无背景色行”按钮点击事件
hideButton.addEventListener('click', function () {
const tables = document.querySelectorAll('table'); // 获取页面中的所有表格
if (tables.length === 0) {
console.error('未找到表格');
return;
}
let isHidden = hideButton.textContent === '显示所有行'; // 当前是否为隐藏状态
tables.forEach(table => {
const rows = table.querySelectorAll('tbody tr'); // 获取当前表格的所有行
rows.forEach(row => {
const backgroundColor = row.style.backgroundColor;
// 如果没有背景色或背景色为默认值
if (!backgroundColor || backgroundColor === '' || backgroundColor === 'transparent') {
row.style.display = isHidden ? '' : 'none'; // 切换隐藏/显示
}
});
});
// 切换按钮文字
hideButton.textContent = isHidden ? '隐藏无背景色行' : '显示所有行';
});
});
})();