// ==UserScript==
// @name 一键复制磁力链和推送到115离线
// @author wangzijian0@vip.qq.com
// @description 支持BT4G/BTDigg/BTSOW/Nyaa/DMHY(動漫花園)/GY(观影)/SeedHub/LongWangBT(梦幻天堂·龙网)/YuHuaGe(雨花阁)/SOBT/CLB(磁力宝)/BTMulu(BT目录)/ØMagnet(无极磁链)/磁力帝等磁力搜索网站,可一键复制磁力链和推送到115网盘进行离线下载,支持打开磁力链,支持批量操作;提供“菜单”和左下角悬浮“设置”入口(需已登录115会员账号才能推送离线任务成功)
// @version 1.1.6.20250913
// @icon 
// @include *://*bt4gprx.com/*
// @include *://*btdig.com/*
// @include *://*btsow.*/*
// @include *://*nyaa.si/*
// @include *://*dmhy.*/*
// @include *://*gying.*/*
// @include *://*gyg.*/*
// @include *://*seedhub.*/*
// @include *://*longwangbt.*/*
// @include *://*yuhuage.*/*
// @include *://*sobt*.*/*
// @include *://*clb*.*/*
// @include *://*btmulu.*/*
// @include *://*cili.*/*
// @include *://*mag.*/*
// @include *://*wuji.*/*
// @include *://*1122*.*/*
// @include *://*cld130.*/*
// @grant GM_setClipboard
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @connect 115.com
// @connect login.115.com
// @connect *
// @run-at document-end
// @namespace https://greasyfork.org/users/1453515
// @license MIT
// @downloadURL https://update.greasyfork.icu/scripts/531685/%E4%B8%80%E9%94%AE%E5%A4%8D%E5%88%B6%E7%A3%81%E5%8A%9B%E9%93%BE%E5%92%8C%E6%8E%A8%E9%80%81%E5%88%B0115%E7%A6%BB%E7%BA%BF.user.js
// @updateURL https://update.greasyfork.icu/scripts/531685/%E4%B8%80%E9%94%AE%E5%A4%8D%E5%88%B6%E7%A3%81%E5%8A%9B%E9%93%BE%E5%92%8C%E6%8E%A8%E9%80%81%E5%88%B0115%E7%A6%BB%E7%BA%BF.meta.js
// ==/UserScript==
(function() {
'use strict';
const isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);
const CONFIG = {
notificationTimeout: isMobile ? 5000 : 3000,
cookieRefreshInterval: 30 * 60 * 1000,
retryDelay: 2000,
maxRetries: 3,
defaultTimeout: 8000,
enableCopyButton: GM_getValue('enableCopyButton', true),
enableOfflineButton: GM_getValue('enableOfflineButton', true),
enableOpenButton: GM_getValue('enableOpenButton', true),
autoFetchEnabled: GM_getValue('autoFetchEnabled', true),
autoFetchSites: {
bt4g: GM_getValue('autoFetch_bt4g', false),
seedhub: GM_getValue('autoFetch_seedhub', false),
yuhuage: GM_getValue('autoFetch_yuhuage', true),
cilimag: GM_getValue('autoFetch_cilimag', true)
},
siteEnabled: {
bt4g: GM_getValue('site_bt4g', true),
btdig: GM_getValue('site_btdig', true),
btsow: GM_getValue('site_btsow', true),
nyaa: GM_getValue('site_nyaa', true),
dmhy: GM_getValue('site_dmhy', true),
seedhub: GM_getValue('site_seedhub', true),
longwangbt: GM_getValue('site_longwangbt', true),
yuhuage: GM_getValue('site_yuhuage', true),
sobt: GM_getValue('site_sobt', true),
clb: GM_getValue('site_clb', true),
btmulu: GM_getValue('site_btmulu', true),
cili_family: GM_getValue('site_cili_family', true),
gying_family: GM_getValue('site_gying_family', true),
cilidi: GM_getValue('site_cilidi', true)
},
enableFloatingSettingsBtn: GM_getValue('enableFloatingSettingsBtn', true)
};
const DEFAULT_BUTTONS_ORDER = ['copy', 'offline', 'open'];
function getButtonsOrder() {
try {
const saved = GM_getValue('buttonsOrder');
if (!saved) return DEFAULT_BUTTONS_ORDER.slice();
if (Array.isArray(saved)) {
const allow = new Set(DEFAULT_BUTTONS_ORDER);
const arr = saved.filter(x => allow.has(x));
return arr.length ? arr : DEFAULT_BUTTONS_ORDER.slice();
}
if (typeof saved === 'string') {
const parts = saved.split(/[|,\s]+/).filter(Boolean);
const allow = new Set(DEFAULT_BUTTONS_ORDER);
const arr = parts.filter(x => allow.has(x));
return arr.length ? arr : DEFAULT_BUTTONS_ORDER.slice();
}
} catch (_) {}
return DEFAULT_BUTTONS_ORDER.slice();
}
function setButtonsOrder(order) {
try { GM_setValue('buttonsOrder', Array.isArray(order) ? order : DEFAULT_BUTTONS_ORDER); } catch (_) {}
}
const SITES_LINKS = {
bt4g: {
sites: [
{ url: 'https://bt4gprx.com' }
]
},
btdig: {
sites: [
{ url: 'https://btdig.com' }
]
},
btsow: {
sites: [
{ url: 'https://btsow.com' }
]
},
nyaa: {
sites: [
{ url: 'https://nyaa.si' }
]
},
dmhy: {
sites: [
{ url: 'https://dmhy.org' }
]
},
gying_family: {
sites: [
{ url: 'www.gying.net' },
{ url: 'www.gying.org' },
{ url: 'www.gying.si' },
{ url: 'www.gying.in' },
{ url: 'www.gyg.la' },
{ url: 'www.gyg.si' }
]
},
seedhub: {
sites: [
{ url: 'https://www.seedhub.cc' },
{ url: 'https://www.seedhub.top' },
{ url: 'https://www.seedhub.icu' },
{ url: 'https://seedhub.pro', note: '移动可能不通' }
],
publish: [
{ url: 'https://workflowy.com/s/ff4ac3a19545/tEvTraNzl9fk1fJA' }
]
},
longwangbt: {
sites: [
{ url: 'http://www.longwangbt.com' }
]
},
yuhuage: {
sites: [
{ url: 'https://www.yuhuage.cc' }
]
},
sobt: {
sites: [
{ url: 'https://sobt.me' }
]
},
clb: {
sites: [
{ url: 'https://clb.im' },
{ url: 'https://cilibao.app' },
{ url: 'https://cilibao.top' }
]
},
btmulu: {
publish: [
{ url: 'https://cursor.vip/btmulu' }
]
},
cili_family: {
publish: [
{ url: 'https://CiLi.st' },
{ url: 'https://cili404.com' }
]
},
cilidi: {
sites: [
],
publish: [
{ url: 'https://cilidi.cyou' },
{ url: 'https://cldcld.cyou' },
{ url: 'https://cldcld.top' },
{ url: 'https://cldcld.com' }
]
}
};
const processedElements = new WeakSet();
function processElements(selector, processor, dataAttribute = 'buttonsAdded') {
document.querySelectorAll(selector).forEach(element => {
if (processedElements.has(element) || element.dataset[dataAttribute]) return;
const result = processor(element);
if (result !== false) {
processedElements.add(element);
element.dataset[dataAttribute] = 'true';
}
});
}
function handleCiLiDiSite() {
document.querySelectorAll('.ssbox .sbar').forEach(sbar => {
if (sbar.dataset.buttonsAdded) return;
const magnetA = sbar.querySelector('a[href^="magnet:"]');
if (!magnetA) return;
const btnContainer = createButtonContainer({ marginRight: '6px' });
const combinedBtn = createCombinedButtons(magnetA.href);
btnContainer.appendChild(combinedBtn);
const firstSpan = sbar.querySelector('span');
if (firstSpan) {
sbar.insertBefore(btnContainer, firstSpan);
} else {
sbar.insertBefore(btnContainer, sbar.firstChild);
}
sbar.dataset.buttonsAdded = true;
});
(() => {
const ssboxes = Array.from(document.querySelectorAll('.tbox .ssbox, .ssbox'));
if (!ssboxes.length) return;
const target = ssboxes.find(box => box.querySelector('.content a[href^="magnet:"]'));
if (!target) return;
const h3 = target.querySelector('.title h3');
if (!h3 || h3.dataset.buttonsAdded) return;
const magnetA = target.querySelector('.content a[href^="magnet:"]');
if (!magnetA) return;
const btnContainer = createButtonContainer({ marginLeft: '8px' });
const combinedBtn = createCombinedButtons(magnetA.href);
btnContainer.appendChild(combinedBtn);
h3.appendChild(btnContainer);
h3.dataset.buttonsAdded = true;
})();
}
function isAutoFetchEnabledFor(siteKey) {
try {
return !!(CONFIG.autoFetchEnabled && CONFIG.autoFetchSites && CONFIG.autoFetchSites[siteKey]);
} catch (_) {
return false;
}
}
async function retryOperation(operation, maxRetries = CONFIG.maxRetries, onRetry = null) {
for (let attempt = 0; attempt <= maxRetries; attempt++) {
try {
return await operation(attempt);
} catch (error) {
console.error(`Operation failed (attempt ${attempt + 1}/${maxRetries + 1}):`, error);
if (attempt === maxRetries) {
throw error;
}
if (onRetry) {
onRetry(attempt, maxRetries);
}
await new Promise(resolve => setTimeout(resolve, CONFIG.retryDelay * (attempt + 1)));
}
}
}
function setupRetryButton(button, retryFunction) {
setButtonError(button, '获取失败,点击重试');
button.style.cursor = 'pointer';
button.addEventListener('click', () => {
button.textContent = '重新获取中...';
button.style.color = '#666';
button.style.cursor = 'default';
retryFunction().then(success => {
if (!success) setButtonError(button, '获取失败');
}).catch(error => {
console.error('重试失败:', error);
setButtonError(button, '重试失败');
});
});
}
const ERROR_CODES = {
10008: '任务已存在,无需重复添加',
911: '需要账号验证,请确保已登录115会员账号',
990: '任务包含违规内容,无法添加',
991: '服务器繁忙,请稍后再试',
992: '离线下载配额已用完',
993: '当前账号无权使用离线下载功能',
994: '文件大小超过限制',
995: '不支持的链接类型',
996: '网络错误,请检查连接',
997: '服务器内部错误',
998: '请求超时',
999: '未知错误'
};
function initializeScript() {
addMenuCommands();
setupMutationObserver();
addActionButtons();
ensureModalStyles();
ensureFloatingSettingsButton();
}
function addMenuCommands() {
const menuCommands = [
{
name: "打开设置面板",
handler: () => openSettingsPanel()
},
{
name: "检查115登录状态",
handler: async () => {
try {
const isLoggedIn = await check115Login(true);
showNotification('115状态', isLoggedIn ? '已登录' : '未登录');
if (!isLoggedIn) {
setTimeout(() => {
if (confirm('需要登录115网盘,是否进入115网盘登录页面?')) {
window.open("https://115.com/?mode=login", "_blank");
}
}, 500);
}
} catch (error) {
showNotification('检查失败', error.message);
}
}
},
{
name: "打开115网盘",
handler: () => window.open("https://115.com/?cid=0&offset=0&mode=wangpan", "_blank")
}
];
menuCommands.forEach(({ name, handler }) => {
GM_registerMenuCommand(name, handler);
});
}
async function checkCookieRefresh() {
try {
await check115Login(true);
} catch (error) {
console.error('检查cookie刷新失败:', error);
}
}
function setupMutationObserver() {
let timeoutId;
const observer = new MutationObserver(() => {
clearTimeout(timeoutId);
timeoutId = setTimeout(() => {
addActionButtons();
ensureFloatingSettingsButton();
}, 100);
});
observer.observe(document, {
childList: true,
subtree: true
});
return observer;
}
function addActionButtons() {
const hostname = window.location.hostname;
const siteHandlers = {
'bt4gprx.com': handleBT4GSite,
'btdig.com': handleBTDigSite,
'nyaa.si': handleNyaaSite,
'dmhy.org': handleDMHYSite,
'seedhub': handleSeedhubSite
};
const patternHandlers = [
{ key: 'sobt', pattern: /sobt[^.]+\..+/, handler: handleSOBTSite },
{ key: 'clb', pattern: /clb[^.]+\..+/, handler: handleSOBTSite },
{ key: 'btsow', pattern: /(\.|^)btsow\./, handler: handleBtsowSite },
{ key: 'btmulu', pattern: /\.btmulu\./, handler: handleBTMULUSite },
{ key: 'cili_family', pattern: /cili|mag|wuji/, handler: handleCiliMagSite },
{ key: 'gying_family', pattern: /(\.gying|\.gyg)\..+/, handler: handleGyingGygSite },
{ key: 'yuhuage', pattern: /yuhuage\..+/, handler: handleYuhuageSite },
{ key: 'longwangbt', pattern: /longwangbt\..+/, handler: handleLongwangbtSite },
{ key: 'cilidi', pattern: /1122|cld130/, handler: handleCiLiDiSite }
];
const domainKeyMap = {
'bt4gprx.com': 'bt4g',
'btdig.com': 'btdig',
'nyaa.si': 'nyaa',
'dmhy.org': 'dmhy',
'seedhub': 'seedhub'
};
for (const [domain, handler] of Object.entries(siteHandlers)) {
if (hostname.includes(domain)) {
const key = domainKeyMap[domain];
if (!key || CONFIG.siteEnabled[key]) {
handler();
}
return;
}
}
for (const { key, pattern, handler } of patternHandlers) {
if (pattern.test(hostname)) {
if (!key || CONFIG.siteEnabled[key]) {
handler();
}
return;
}
}
}
const ICONS = {
copy: '',
offline: '
',
open: ''
};
function createButtonContainer(options = {}) {
const btnContainer = document.createElement(options.elementType || 'span');
btnContainer.className = 'magnet-action-buttons';
btnContainer.style.cssText = `
display: inline-block;
margin-right: ${options.marginRight || '5px'};
margin-left: ${options.marginLeft || '0'};
vertical-align: ${options.verticalAlign || 'middle'}
`;
if (options.customStyles) {
Object.assign(btnContainer.style, options.customStyles);
}
return btnContainer;
}
async function fetchWithRetry(url, options = {}, maxRetries = CONFIG.maxRetries) {
const normalizedUrl = /^https?:/.test(url) ? url : new URL(url, location.origin).href;
for (let attempt = 0; attempt <= maxRetries; attempt++) {
try {
const response = await fetch(normalizedUrl, {
credentials: 'omit',
...options
});
if (!response.ok) {
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
}
return await response.text();
} catch (error) {
console.error(`Fetch attempt ${attempt + 1}/${maxRetries + 1} failed for ${normalizedUrl}:`, error);
if (attempt === maxRetries) {
throw error;
}
await new Promise(resolve => setTimeout(resolve, CONFIG.retryDelay * (attempt + 1)));
}
}
}
function createCombinedButtons(magnetLinkOrElement) {
const combinedBtn = document.createElement('button');
combinedBtn.className = 'magnet-combined-button';
combinedBtn.style.display = 'inline-flex';
combinedBtn.style.alignItems = 'center';
combinedBtn.style.justifyContent = 'center';
combinedBtn.style.backgroundColor = 'transparent';
combinedBtn.style.border = '1px solid rgba(0,0,0,0.14)';
combinedBtn.style.borderRadius = '3px';
combinedBtn.style.padding = '2px';
combinedBtn.style.fontSize = '12px';
combinedBtn.style.cursor = 'pointer';
combinedBtn.style.transition = 'all 0.15s ease-in-out';
combinedBtn.style.userSelect = 'none';
combinedBtn.style.boxSizing = 'border-box';
combinedBtn.style.height = '26px';
const titles = { copy: '复制磁力链', offline: '推送到115离线', open: '打开磁力链' };
const createButtonPart = (type, icon) => {
const part = document.createElement('span');
part.className = `magnet-button-part ${type}-part`;
part.style.cssText = 'padding:0 6px;color:#333;transition:all 0.15s ease-in-out;display:inline-flex;align-items:center;justify-content:center;min-width:20px;height:22px;';
part.innerHTML = icon;
part.dataset.type = type;
part.title = titles[type] || '操作';
return part;
};
const order = getButtonsOrder();
const buttonParts = [];
for (const type of order) {
if (type === 'copy' && CONFIG.enableCopyButton) buttonParts.push(createButtonPart('copy', ICONS.copy));
if (type === 'offline' && CONFIG.enableOfflineButton) buttonParts.push(createButtonPart('offline', ICONS.offline));
if (type === 'open' && CONFIG.enableOpenButton) buttonParts.push(createButtonPart('open', ICONS.open));
}
if (buttonParts.length > 0) {
if (buttonParts.length === 1) {
buttonParts[0].style.borderRadius = '2px';
} else {
buttonParts[0].style.borderRadius = '2px 0 0 2px';
}
combinedBtn.appendChild(buttonParts[0]);
for (let i = 1; i < buttonParts.length; i++) {
const sep = document.createElement('span');
sep.style.cssText = 'padding: 0 2px; color: #999;';
sep.innerText = '|';
combinedBtn.appendChild(sep);
if (i === buttonParts.length - 1) {
buttonParts[i].style.borderRadius = '0 2px 2px 0';
} else {
buttonParts[i].style.borderRadius = '0';
}
combinedBtn.appendChild(buttonParts[i]);
}
}
const resolveMagnet = async () => {
try {
if (typeof magnetLinkOrElement === 'function') {
const res = await magnetLinkOrElement();
if (typeof res === 'string' && res.startsWith('magnet:')) return res;
}
if (typeof magnetLinkOrElement === 'string' && magnetLinkOrElement.startsWith('magnet:')) {
return magnetLinkOrElement;
}
const el = magnetLinkOrElement;
if (el && el.nodeType === 1) {
const a1 = el.closest('a[href^="magnet:"]') || el.querySelector?.('a[href^="magnet:"]');
if (a1?.href?.startsWith('magnet:')) return a1.href;
{
const linkEl = el.closest('a') || (el.tagName === 'A' ? el : null);
const href = linkEl?.href || el.getAttribute?.('href') || '';
const m = href.match(/\/(?:torrent|detail)\/([a-f0-9]+)\.html(?:$|\?)/i);
if (m && m[1]) {
const name = (linkEl?.textContent || el.textContent || '').trim();
let magnet = `magnet:?xt=urn:btih:${m[1]}`;
if (name) magnet += `&dn=${encodeURIComponent(name)}`;
return magnet;
}
}
const a2 = el.closest('a[href*="downloadtorrentfile.com/hash/"]') || el.querySelector?.('a[href*="downloadtorrentfile.com/hash/"]');
if (a2?.href) {
const m = a2.href.match(/hash\/([a-f0-9]{40})/i);
if (m) {
const nameMatch = a2.href.match(/[?&]name=([^&]+)/i);
let magnet = `magnet:?xt=urn:btih:${m[1]}`;
if (nameMatch?.[1]) magnet += `&dn=${nameMatch[1]}`;
return magnet;
}
}
if (/bt4g/.test(location.hostname) && el.closest('a')?.href) {
const href = el.closest('a').href;
if (typeof fetchBT4GMagnetFromDetail === 'function') {
const mg = await fetchBT4GMagnetFromDetail(href);
if (mg) return mg;
}
}
}
} catch (e) {
console.warn('解析磁力失败', e);
}
return null;
};
combinedBtn.addEventListener('click', async (ev) => {
const part = ev.target.closest?.('.magnet-button-part');
if (!part) return;
ev.preventDefault();
ev.stopPropagation();
const type = part.dataset.type;
const magnetLink = await resolveMagnet();
if (!magnetLink) {
showNotification('未获取到磁力链', '请稍后重试或手动打开详情页');
return;
}
if (type === 'copy') {
await handleCopyAction(combinedBtn, magnetLink);
} else if (type === 'offline') {
await handleOfflineAction(combinedBtn, magnetLink);
} else if (type === 'open') {
window.open(magnetLink, '_blank');
showButtonFeedback(combinedBtn, 'open');
}
});
return combinedBtn;
}
function applyButtonsOrderToExisting() {
const order = getButtonsOrder();
document.querySelectorAll('.magnet-combined-button').forEach(btn => {
const partsMap = new Map();
btn.querySelectorAll('.magnet-button-part').forEach(p => {
const t = p.dataset.type;
if (t) partsMap.set(t, p);
});
const existing = order.map(t => partsMap.get(t)).filter(Boolean);
if (!existing.length) return;
while (btn.firstChild) btn.removeChild(btn.firstChild);
existing.forEach((part, idx) => {
part.style.borderRadius = idx === 0 && existing.length === 1 ? '2px' : (idx === 0 ? '2px 0 0 2px' : (idx === existing.length - 1 ? '0 2px 2px 0' : '0'));
if (idx > 0) {
const sep = document.createElement('span');
sep.style.cssText = 'padding: 0 2px; color: #999;';
sep.innerText = '|';
btn.appendChild(sep);
}
btn.appendChild(part);
});
});
}
async function handleCopyAction(btn, magnetLink) {
try {
let decodedMagnetLink = magnetLink;
try {
decodedMagnetLink = decodeURIComponent(magnetLink);
} catch (e) {}
GM_setClipboard(decodedMagnetLink, 'text');
if (isMobile && navigator.clipboard?.writeText) {
try {
await navigator.clipboard.writeText(decodedMagnetLink);
} catch (clipboardError) {
console.log('使用navigator.clipboard失败:', clipboardError);
}
}
showNotification('磁力链已复制', decodedMagnetLink);
showButtonFeedback(btn, 'copy');
} catch (error) {
showNotification('复制失败', `请手动复制: ${magnetLink}`);
}
}
const SUCCESS_FEEDBACK_SVG = '';
function showButtonFeedback(btn, type = null) {
const clickedPart = btn.classList.contains('magnet-combined-button')
? btn.querySelector(type ? `.magnet-button-part[data-type="${type}"]` : '.magnet-button-part')
: btn;
if (!clickedPart) return;
const originalContent = clickedPart.innerHTML;
clickedPart.style.cssText += 'min-height:22px;display:inline-flex;align-items:center;justify-content:center;';
clickedPart.innerHTML = SUCCESS_FEEDBACK_SVG;
btn.disabled = true;
setTimeout(() => {
clickedPart.innerHTML = originalContent;
btn.disabled = false;
}, 2000);
}
async function handleOfflineAction(btn, magnetLink) {
await process115Offline(magnetLink);
showButtonFeedback(btn, 'offline');
}
function handleBT4GSite() {
try {
const searchForm = document.querySelector('form[action="/search"]');
const searchInput = document.getElementById('search');
if (searchForm && searchInput && !document.getElementById('bt4g-advanced-filter')) {
const keywordMaps = {
resolution: {
'SD': ['480p', '480P', '480i', '480I', '576p', '576P', '576i', '576I', 'SD', 'VCD', 'DVD', 'SDTV'],
'720p': ['720p', '720P', 'HD'],
'1080p': ['1080p', '1080P', 'HD1080P', 'FHD', 'FullHD', 'Full HD', '1920x1080'],
'1080i': ['1080i', '1080I', 'FHD', 'FullHD', 'Full HD', '1920x1080i'],
'1440p': ['1440p', '1440P', '2k', '2K', 'QHD', 'QuadHD', 'Quad HD', '2560x1440'],
'4K/UHD': ['2160p', '2160P', '4K', '4k', 'UHD', 'UltraHD', '3840x2160', '4096x2160'],
'8K': ['4320p', '4320P', '8K', 'FUHD', 'FUHD', '7680x4320']
},
hdr: {
'HDR': ['HDR'],
'HDR10': ['HDR10'],
'HDR10+': ['HDR10+', 'HDR10Plus'],
'HLG': ['HLG'],
'HDR Vivid': ['HDR Vivid', 'HDR-Vivid', 'Vivid'],
'Dolby Vision': ['DV', 'DoVi', 'DolbyVision', 'Dolby Vision']
},
codec: {
'H264/AVC': ['H264', '264', 'AVC', 'h264', 'MPEG4AVC', 'x264'],
'MPEG-2': ['MPEG-2', 'MPEG2', 'H262', 'H.262'],
'MPEG-4': ['MPEG-4', 'MPEG4', 'DivX', 'Xvid', 'XviD'],
'MPEG-5/EVC': ['MPEG-5', 'MPEG5', 'EVC'],
'H265/HEVC': ['H265', '265', 'HEVC', 'x265', 'h265'],
'AV1': ['AV1'],
'VC-1': ['VC-1', 'VC1'],
'VP8': ['VP8'],
'VP9': ['VP9']
},
mediaType: {
'BD': ['BD', 'BLURAY', 'BLU', 'RAY', 'BDMV', 'BDREMUX', 'REMUX'],
'BDrip/BRrip': ['BDrip', 'BRrip', 'BDRip', 'BRRip', 'BluRayRip'],
'WEB-DL': ['WEBDL', 'WEB-DL'],
'WEB': ['WEB', 'WEBRIP', 'WEBRip'],
'HDTV': ['HDTV', 'TV'],
'TS': ['TS', 'TS源'],
'DVD': ['DVD', 'DVDRIP', 'DVDRip'],
'TC': ['TC', '枪版', '抢版']
},
audio: {
'杜比': ['Dolby', 'DolbyDigital', 'DD'],
'杜比全景声': ['Atmos', 'DolbyAtmos'],
'Dolby Digital Plus': ['DD+', 'DDP', 'E-AC-3', 'EAC3', 'DolbyDigitalPlus', 'Dolby Digital Plus'],
'DTS': ['DTS', 'DTSHD', 'DTSHDMA', 'DTSX'],
'TrueHD': ['TrueHD', 'TRUEHD', 'TrueHD2', 'TrueHD.2', 'TrueHD.2.0', 'TrueHD5', 'TrueHD.5.1', 'TrueHD.5.1','TrueHD7', 'TrueHD.7', 'TrueHD.7.1'],
'通用': ['AAC', 'AC3', 'AC-3', 'MP3', 'LPCM', 'PCM', 'FLAC', 'Opus', 'OPUS']
}
};
const isDarkMode = document.body.classList.contains('dark-mode') ||
document.documentElement.classList.contains('dark') ||
document.documentElement.getAttribute('data-bs-theme') === 'dark';
const submitBtn = searchForm.querySelector('button[type="submit"], input[type="submit"]');
const filterBtn = document.createElement('button');
filterBtn.type = 'button';
filterBtn.textContent = '筛选';
filterBtn.className = 'btn btn-secondary btn-sm';
if (searchInput?.parentNode) {
searchInput.parentNode.insertBefore(filterBtn, searchInput.nextSibling);
} else if (submitBtn?.parentNode) {
submitBtn.parentNode.insertBefore(filterBtn, submitBtn);
} else {
searchForm.appendChild(filterBtn);
}
try {
searchInput.style.borderTopRightRadius = '0';
searchInput.style.borderBottomRightRadius = '0';
searchInput.style.position = 'relative';
searchInput.style.zIndex = '1';
filterBtn.style.margin = '0';
filterBtn.style.borderRadius = '0';
filterBtn.style.borderLeftWidth = '0';
filterBtn.style.position = 'relative';
filterBtn.style.zIndex = '2';
if (submitBtn) {
submitBtn.style.marginLeft = '-1px';
submitBtn.style.borderTopLeftRadius = '0';
submitBtn.style.borderBottomLeftRadius = '0';
submitBtn.style.position = 'relative';
submitBtn.style.zIndex = '3';
}
} catch (_) {}
const panel = document.createElement('div');
panel.id = 'bt4g-advanced-filter';
panel.style.display = 'none';
panel.className = 'advanced-search mb-3 mt-2';
updateFixedAdvancedSearchStyle(panel, isDarkMode);
searchForm.parentNode.insertBefore(panel, searchForm.nextSibling);
try {
localStorage.removeItem('bt4g_advanced_settings');
localStorage.removeItem('bt4g_filter_open');
localStorage.removeItem('bt4g_original_query');
} catch (_) {}
if (!document.getElementById('bt4g-advanced-style')) {
const style = document.createElement('style');
style.id = 'bt4g-advanced-style';
style.textContent = `
#bt4g-advanced-filter { font-size: 12px; line-height: 1.45; border-left: 3px solid rgba(13,110,253,.35); position: relative; padding-right: 40px; }
#bt4g-advanced-filter .bt4g-filter-row { padding: 6px 0; border-top: 1px dashed rgba(108,117,125,.25); display:flex; align-items:flex-start; }
#bt4g-advanced-filter .bt4g-filter-row:first-child { border-top: none; }
#bt4g-advanced-filter .bt4g-filter-row:hover { background: rgba(108,117,125,.05); border-radius: 5px; padding-left: 4px; margin-left: -4px; }
#bt4g-advanced-filter label.btn { border-radius: 8px; padding: 2px 10px; line-height: 1.3; border-width:1px; transition: all .12s ease-in-out; }
#bt4g-advanced-filter label.btn:hover { filter: brightness(0.97); transform: translateY(-0.5px); }
#bt4g-advanced-filter label.btn:active { transform: translateY(0); filter: brightness(0.95); }
#bt4g-advanced-filter .btn-check:focus + label,
#bt4g-advanced-filter label.btn:focus { box-shadow: 0 0 0 .12rem rgba(13,110,253,.15) !important; }
#bt4g-advanced-filter .bt4g-filter-row > span { width: 64px; flex: 0 0 64px; display:block; }
#bt4g-advanced-filter .bt4g-filter-row > div { display:flex; flex-wrap:wrap; gap:3px; flex:1 1 auto; min-width:0; }
form[action="/search"] { position: relative; }
form[action="/search"] #autocomplete-list,
form[action="/search"] .autocomplete-items { position: absolute !important; top: 100% !important; left: 0; right: 0; z-index: 1061 !important; }
form[action="/search"] { display:block; width:100%; }
#bt4g-advanced-filter { display:block; width:100% !important; max-width:none !important; flex:0 0 100%; align-self:stretch; clear:both; }
body:not(.dark):not(.dark-mode) #bt4g-advanced-filter label.btn.btn-outline-dark { background: rgba(108,117,125,.06); border-color: rgba(108,117,125,.35); color:#212529; }
body.dark-mode #bt4g-advanced-filter label.btn.btn-outline-light,
html.dark #bt4g-advanced-filter label.btn.btn-outline-light { background: rgba(255,255,255,.06); border-color: rgba(255,255,255,.25); color:#e9ecef; }
#bt4g-advanced-filter .bt4g-resolution .btn-check:checked + label.btn { background:#0d6efd !important; border-color:#0d6efd !important; color:#fff !important; }
#bt4g-advanced-filter .bt4g-hdr .btn-check:checked + label.btn { background:#6f42c1 !important; border-color:#6f42c1 !important; color:#fff !important; }
#bt4g-advanced-filter .bt4g-codec .btn-check:checked + label.btn { background:#198754 !important; border-color:#198754 !important; color:#fff !important; }
#bt4g-advanced-filter .bt4g-mediaType .btn-check:checked + label.btn { background:#20c997 !important; border-color:#20c997 !important; color:#fff !important; }
#bt4g-advanced-filter .bt4g-audio .btn-check:checked + label.btn { background:#d63384 !important; border-color:#d63384 !important; color:#fff !important; }
#bt4g-advanced-filter .bt4g-filter-row input[value=""]:checked + label.btn { background:#000 !important; border-color:#000 !important; color:#fff !important; }
#bt4g-advanced-filter .bt4g-reset-icon { position:absolute; top:6px; right:6px; width:24px; height:24px; border-radius:50%; display:flex; align-items:center; justify-content:center; cursor:pointer; z-index:5; background: transparent; border: 1px solid rgba(108,117,125,.35); color:#6c757d; padding:0; }
#bt4g-advanced-filter .bt4g-reset-icon:hover { background: rgba(108,117,125,.08); color:#495057; }
#bt4g-advanced-filter .bt4g-reset-icon:active { transform: scale(0.97); }
#bt4g-advanced-filter .bt4g-reset-icon svg { width:14px; height:14px; display:block; }
html.dark #bt4g-advanced-filter .bt4g-reset-icon,
body.dark-mode #bt4g-advanced-filter .bt4g-reset-icon { border-color: rgba(255,255,255,.25); color:#ced4da; }
html.dark #bt4g-advanced-filter .bt4g-reset-icon:hover,
body.dark-mode #bt4g-advanced-filter .bt4g-reset-icon:hover { background: rgba(255,255,255,.08); color:#e9ecef; }
@keyframes bt4g-spin-left { to { transform: rotate(-360deg); } }
#bt4g-advanced-filter .bt4g-reset-icon.spinning svg { animation: bt4g-spin-left .6s ease; }
@media (max-width: 576px) {
#bt4g-advanced-filter .bt4g-filter-row { padding: 4px 0; flex-direction: column; align-items: stretch; }
#bt4g-advanced-filter .bt4g-filter-row > span { width: 100%; flex: 0 0 100%; margin-bottom: 4px; }
#bt4g-advanced-filter .bt4g-filter-row > div { gap: 3px; width: 100%; flex: 0 0 100%; }
}
`;
document.head.appendChild(style);
}
function createOptionRow(name, label, choices, isDark) {
const row = document.createElement('div');
row.style.cssText = 'display:flex;align-items:center;margin-bottom:6px;width:100%';
row.classList.add('bt4g-filter-row', `bt4g-${name}`);
const labelEl = document.createElement('span');
labelEl.textContent = label;
labelEl.style.cssText = 'width:64px;margin-right:6px;white-space:nowrap;font-weight:bold;font-size:12px;';
labelEl.style.color = isDark ? '#e9ecef' : '#212529';
row.appendChild(labelEl);
const group = document.createElement('div');
group.style.cssText = 'display:flex;flex-wrap:wrap;gap:3px;';
choices.forEach((choice, idx) => {
const id = `${name}_${idx}`;
const radio = document.createElement('input');
radio.type = 'radio';
radio.name = name;
radio.id = id;
radio.value = choice.value;
radio.className = 'btn-check';
radio.checked = idx === 0;
const optLabel = document.createElement('label');
optLabel.className = isDark ? 'btn btn-outline-light btn-sm' : 'btn btn-outline-dark btn-sm';
optLabel.htmlFor = id;
optLabel.textContent = choice.label;
group.appendChild(radio);
group.appendChild(optLabel);
});
row.appendChild(group);
return row;
}
function updateFixedAdvancedSearchStyle(element, isDark) {
const backgroundColor = isDark ? '#212529' : '#f8f9fa';
const textColor = isDark ? '#e9ecef' : '#212529';
const borderColor = isDark ? '#3e444a' : '#dee2e6';
const shadow = isDark ? '0 2px 8px rgba(0,0,0,.25)' : '0 2px 10px rgba(0,0,0,.06)';
element.style.cssText = `display:flex;flex-direction:column;width:100%;padding:8px 10px;background-color:${backgroundColor};color:${textColor};border:1px solid ${borderColor};border-radius:6px;margin-bottom:10px;box-shadow:${shadow};`;
}
function setRadioValue(name, value) {
const radios = panel.querySelectorAll(`input[name="${name}"]`);
let found = false;
radios.forEach(r => { if (r.value === value) { r.checked = true; found = true; } });
if (!found && radios.length) radios[0].checked = true;
}
panel.appendChild(createOptionRow('resolution', '分辨规格:', [
{ value: '', label: '全部' },
{ value: 'SD', label: 'DVD/VCD' },
{ value: '720p', label: '720p' },
{ value: '1080p', label: '1080p' },
{ value: '1080i', label: '1080i' },
{ value: '1440p', label: '2K/QHD' },
{ value: '4K/UHD', label: '4K/UHD' },
{ value: '8K', label: '8K/FUHD' }
], isDarkMode));
panel.appendChild(createOptionRow('hdr', '高动态域:', [
{ value: '', label: '全部' },
{ value: 'HDR', label: 'HDR' },
{ value: 'HDR10', label: 'HDR10' },
{ value: 'HDR10+', label: 'HDR10+' },
{ value: 'HLG', label: 'HLG' },
{ value: 'HDR Vivid', label: 'HDR Vivid' },
{ value: 'Dolby Vision', label: 'Dolby Vision/DV' }
], isDarkMode));
panel.appendChild(createOptionRow('codec', '视频编码:', [
{ value: '', label: '全部' },
{ value: 'H264/AVC', label: 'H264/AVC/x264' },
{ value: 'MPEG-2', label: 'MPEG-2' },
{ value: 'MPEG-4', label: 'MPEG-4/DivX/Xvid' },
{ value: 'MPEG-5/EVC', label: 'MPEG-5/EVC' },
{ value: 'H265/HEVC', label: 'H265/HEVC/x265' },
{ value: 'AV1', label: 'AV1' },
{ value: 'VC-1', label: 'VC-1' },
{ value: 'VP8', label: 'VP8' },
{ value: 'VP9', label: 'VP9' }
], isDarkMode));
panel.appendChild(createOptionRow('mediaType', '媒体类型:', [
{ value: '', label: '全部' },
{ value: 'BD', label: 'BD/蓝光/REMUX' },
{ value: 'BDrip/BRrip', label: 'BDrip/BRrip' },
{ value: 'WEB-DL', label: 'WEB-DL' },
{ value: 'WEB', label: 'WEB/WEBRip' },
{ value: 'HDTV', label: 'HDTV' },
{ value: 'TS', label: 'TS' },
{ value: 'DVD', label: 'DVD/DVDRip' },
{ value: 'TC', label: 'TC' }
], isDarkMode));
const audioChoices = [
{ value: '', label: '全部' },
{ value: '杜比', label: '杜比/Dolby' },
{ value: '杜比全景声', label: '杜比全景声/Atmos' },
{ value: 'Dolby Digital Plus', label: 'Dolby Digital Plus/DD+/E-AC-3' },
{ value: 'DTS', label: 'DTS系列' },
{ value: 'TrueHD', label: 'TrueHD' },
{ value: '通用', label: '通用' }
];
panel.appendChild(createOptionRow('audio', '音频类型:', audioChoices, isDarkMode));
const resetBtn = document.createElement('button');
resetBtn.type = 'button';
resetBtn.className = 'bt4g-reset-icon';
resetBtn.title = '重置';
resetBtn.setAttribute('aria-label', '重置');
resetBtn.innerHTML = `
`;
resetBtn.addEventListener('click', () => {
resetBtn.classList.add('spinning');
const onAnimEnd = () => {
resetBtn.classList.remove('spinning');
resetBtn.removeEventListener('animationend', onAnimEnd, true);
};
resetBtn.addEventListener('animationend', onAnimEnd, true);
setTimeout(() => resetBtn.classList.remove('spinning'), 800);
['resolution','hdr','codec','mediaType','audio'].forEach(name => {
const first = panel.querySelector(`input[name="${name}"]`);
if (first) first.checked = true;
});
sessionStorage.setItem('bt4g_advanced_settings', JSON.stringify({resolution:'',hdr:'',codec:'',mediaType:'',audio:''}));
});
panel.appendChild(resetBtn);
filterBtn.addEventListener('click', () => {
const opened = panel.style.display !== 'none';
panel.style.display = opened ? 'none' : 'block';
sessionStorage.setItem('bt4g_filter_open', opened ? 'false' : 'true');
});
const urlParams = new URLSearchParams(window.location.search);
function storeAdvanced() {
const settings = {
resolution: panel.querySelector('input[name="resolution"]:checked')?.value || '',
hdr: panel.querySelector('input[name="hdr"]:checked')?.value || '',
codec: panel.querySelector('input[name="codec"]:checked')?.value || '',
mediaType: panel.querySelector('input[name="mediaType"]:checked')?.value || '',
audio: panel.querySelector('input[name="audio"]:checked')?.value || ''
};
sessionStorage.setItem('bt4g_advanced_settings', JSON.stringify(settings));
}
(function restoreAdvanced(){
try {
const s = JSON.parse(sessionStorage.getItem('bt4g_advanced_settings')) || {};
if (s.resolution) setRadioValue('resolution', s.resolution);
if (s.hdr) setRadioValue('hdr', s.hdr);
if (s.codec) setRadioValue('codec', s.codec);
if (s.mediaType) setRadioValue('mediaType', s.mediaType);
if (s.audio) setRadioValue('audio', s.audio);
const open = sessionStorage.getItem('bt4g_filter_open');
panel.style.display = open === 'true' ? 'block' : 'none';
} catch {}
})();
panel.addEventListener('change', (e) => {
if (e.target && e.target.matches('input[type="radio"]')) {
storeAdvanced();
}
});
function processSearch(e) {
if (e) e.preventDefault();
const baseQuery = searchInput.value.trim();
if (!baseQuery) { searchForm.submit(); return; }
const resolution = panel.querySelector('input[name="resolution"]:checked')?.value || '';
const hdr = panel.querySelector('input[name="hdr"]:checked')?.value || '';
const codec = panel.querySelector('input[name="codec"]:checked')?.value || '';
const mediaType = panel.querySelector('input[name="mediaType"]:checked')?.value || '';
const audio = panel.querySelector('input[name="audio"]:checked')?.value || '';
storeAdvanced();
const conds = [];
if (resolution && keywordMaps.resolution[resolution]) conds.push(`(${keywordMaps.resolution[resolution].join('|')})`);
if (hdr && keywordMaps.hdr[hdr]) conds.push(`(${keywordMaps.hdr[hdr].join('|')})`);
if (codec && keywordMaps.codec[codec]) conds.push(`(${keywordMaps.codec[codec].join('|')})`);
if (mediaType && keywordMaps.mediaType[mediaType]) conds.push(`(${keywordMaps.mediaType[mediaType].join('|')})`);
if (audio && keywordMaps.audio[audio]) conds.push(`(${keywordMaps.audio[audio].join('|')})`);
let finalQuery = baseQuery;
if (conds.length) finalQuery += ' ' + conds.join(' ');
sessionStorage.setItem('bt4g_original_query', baseQuery);
searchInput.value = finalQuery;
searchForm.submit();
}
searchForm.addEventListener('submit', processSearch, { capture: true });
searchInput.addEventListener('keydown', (e) => { if (e.key === 'Enter') processSearch(e); });
if (new URLSearchParams(window.location.search).has('q')) {
const originalQuery = sessionStorage.getItem('bt4g_original_query');
if (originalQuery) setTimeout(() => { searchInput.value = originalQuery; }, 100);
}
}
} catch (err) { console.warn('BT4G 筛选面板初始化失败:', err); }
processMagnetLinks({
selectors: '.result-item h5 > a[href^="/magnet/"]',
containerStyles: { marginRight: '8px' },
customProcessor: (titleA) => {
const btnContainer = createButtonContainer({ marginRight: '8px' });
titleA.parentNode.insertBefore(btnContainer, titleA);
if (isAutoFetchEnabledFor('bt4g')) {
const loadingBtn = createLoadingButton();
btnContainer.appendChild(loadingBtn);
processBT4GMagnetLink(titleA, btnContainer).then(success => {
if (!success) {
setupRetryButton(loadingBtn, () =>
processBT4GMagnetLink(titleA, btnContainer, 2, 6000)
);
}
}).catch(error => {
console.error('BT4G处理失败:', error);
setButtonError(loadingBtn, '处理失败');
});
} else {
const combinedBtn = createCombinedButtons(titleA);
btnContainer.appendChild(combinedBtn);
}
}
});
processElements('div.card-header', (headerEl) => {
const card = headerEl.closest('.card') || document;
const magnetBtn = card.querySelector('a[href*="downloadtorrentfile.com/hash/"]');
if (!magnetBtn) return false;
const btnContainer = createButtonContainer({ marginLeft: '8px' });
const combinedBtn = createCombinedButtons(magnetBtn);
btnContainer.appendChild(combinedBtn);
headerEl.appendChild(btnContainer);
return true;
});
}
async function fetchBT4GMagnetFromDetail(detailHref) {
try {
const html = await fetchWithRetry(detailHref);
const parser = new DOMParser();
const doc = parser.parseFromString(html, 'text/html');
const magnetA = doc.querySelector('a.btn.btn-primary.me-2[href*="downloadtorrentfile.com/hash/"]');
if (!magnetA) return null;
const href = magnetA.href;
const hashMatch = href.match(/hash\/([a-f0-9]{40})/i);
if (!hashMatch) return null;
const hash = hashMatch[1];
const nameMatch = href.match(/[?&]name=([^&]+)/i);
let magnet = `magnet:?xt=urn:btih:${hash}`;
if (nameMatch?.[1]) {
magnet += `&dn=${nameMatch[1]}`;
}
return magnet;
} catch (error) {
console.error('Failed to fetch BT4G magnet:', error);
return null;
}
}
async function processBT4GMagnetLink(linkElement, btnContainer, maxRetries = CONFIG.maxRetries, timeout = CONFIG.defaultTimeout) {
if (!linkElement?.href) return false;
return await retryOperation(async (attempt) => {
const magnetLink = await Promise.race([
fetchBT4GMagnetFromDetail(linkElement.href),
new Promise((_, reject) =>
setTimeout(() => reject(new Error('请求超时')), timeout)
)
]);
if (magnetLink) {
btnContainer.innerHTML = '';
btnContainer.appendChild(createCombinedButtons(magnetLink));
return true;
}
throw new Error('未获取到磁力链');
}, maxRetries, (attempt, maxRetries) => {
const loadingBtn = btnContainer.querySelector('.magnet-loading-btn');
if (loadingBtn) {
loadingBtn.textContent = `重试中(${attempt + 1}/${maxRetries})...`;
}
});
}
function handleBtsowSite() {
processMagnetLinks({
selectors: '.row.data-row .file',
containerStyles: { marginRight: '8px' },
customProcessor: (titleLink) => {
const magnetLink = extractBtsowMagnetLink(titleLink);
if (!magnetLink) return;
const btnContainer = createButtonContainer({ marginRight: '8px' });
const combinedBtn = createCombinedButtons(magnetLink);
btnContainer.appendChild(combinedBtn);
titleLink.parentNode.insertBefore(btnContainer, titleLink);
}
});
processMagnetLinks({
selectors: 'textarea.magnet-link[readonly]',
containerStyles: {
elementType: 'div',
marginLeft: '10px'
},
customProcessor: (textarea) => {
const magnetLink = textarea.value.trim();
if (!magnetLink?.startsWith('magnet:')) return;
const btnContainer = createButtonContainer({
elementType: 'div',
marginLeft: '10px'
});
const combinedBtn = createCombinedButtons(magnetLink);
btnContainer.appendChild(combinedBtn);
textarea.parentNode.insertBefore(btnContainer, textarea.nextSibling);
}
});
}
function handleBTMULUSite() {
processMagnetLinks({
selectors: 'div[style="overflow: hidden;"] a[href^="/hash/"] h4',
containerStyles: {
customStyles: { margin: '0 8px' }
},
customProcessor: (titleElement) => {
const titleLink = titleElement.closest('a[href^="/hash/"]');
if (!titleLink) return;
const labelElement = titleElement.querySelector('span.label');
if (!labelElement) return;
const hashMatch = titleLink.href.match(/\/hash\/([a-f0-9]{40})/i);
if (!hashMatch) return;
const hash = hashMatch[1];
const titleText = titleElement.textContent.replace(/^\s*\w+\s*/, '').trim();
const magnetLink = `magnet:?xt=urn:btih:${hash}&dn=${encodeURIComponent(titleText)}`;
const btnContainer = createButtonContainer({
customStyles: { margin: '0 8px' }
});
const combinedBtn = createCombinedButtons(magnetLink);
btnContainer.appendChild(combinedBtn);
if (labelElement.nextSibling) {
titleElement.insertBefore(btnContainer, labelElement.nextSibling);
} else {
titleElement.appendChild(btnContainer);
}
}
});
processElements('div.panel-heading > h3', (h3) => {
const panel = h3.closest('.panel') || h3.parentElement;
const magnetA = panel?.querySelector('div.media-body a[href^="magnet:"]');
if (!magnetA || !magnetA.href) return false;
const btnContainer = createButtonContainer({ marginLeft: '8px' });
const combinedBtn = createCombinedButtons(magnetA.href);
btnContainer.appendChild(combinedBtn);
h3.appendChild(btnContainer);
return true;
});
}
function extractBtsowMagnetLink(element) {
try {
const hashMatch = element.href.match(/detail\/(\w+)/i);
if (hashMatch && hashMatch[1]) {
const titleText = element.textContent.trim();
return `magnet:?xt=urn:btih:${hashMatch[1]}&dn=${encodeURIComponent(titleText)}`;
}
throw new Error('无法提取磁力链Hash');
} catch (error) {
return null;
}
}
function handleSOBTSite() {
processElements('h3 > a[href^="/torrent/"]', (titleLink) => {
const btnContainer = createButtonContainer();
const combinedBtn = createCombinedButtons(titleLink);
btnContainer.appendChild(combinedBtn);
titleLink.parentNode.insertBefore(btnContainer, titleLink);
return true;
});
processElements('.item-title h3 > a[href^="/detail/"]', (titleLink) => {
const btnContainer = createButtonContainer();
const combinedBtn = createCombinedButtons(titleLink);
btnContainer.appendChild(combinedBtn);
titleLink.parentNode.insertBefore(btnContainer, titleLink);
return true;
});
processElements('a.download[id="down-url"]', (openLinkBtn) => {
const btnContainer = createButtonContainer({ marginRight: '8px' });
const combinedBtn = createCombinedButtons(openLinkBtn.href);
btnContainer.appendChild(combinedBtn);
openLinkBtn.parentNode.insertBefore(btnContainer, openLinkBtn);
return true;
});
}
function handleBTDigSite() {
processElements('.torrent_name > a', (titleLink) => {
const resultDiv = titleLink.closest('.one_result');
const magnetLink = resultDiv?.querySelector('.torrent_magnet a[href^="magnet:"]');
if (!magnetLink) return false;
const btnContainer = createButtonContainer({ marginRight: '10px' });
const combinedBtn = createCombinedButtons(magnetLink);
btnContainer.appendChild(combinedBtn);
titleLink.parentNode.insertBefore(btnContainer, titleLink);
return true;
});
processElements('table tr', (tr) => {
const ths = tr.querySelectorAll('th');
if (!ths || ths.length < 2) return false;
const leftTh = ths[0];
const leftText = (leftTh.textContent || '').replace(/\s+/g, ' ').trim();
if (!/^Torrent\s*info$/i.test(leftText)) return false;
const rightTh = ths[1];
if (rightTh.dataset.buttonsAdded) return false;
const pageMagnet = document.querySelector('a[href^="magnet:"]');
const magnetHref = pageMagnet?.href;
if (!magnetHref) return false;
const btnContainer = createButtonContainer({ marginLeft: '8px' });
btnContainer.style.display = 'inline-flex';
btnContainer.style.verticalAlign = 'middle';
btnContainer.style.float = 'left';
const combinedBtn = createCombinedButtons(magnetHref);
btnContainer.appendChild(combinedBtn);
rightTh.appendChild(btnContainer);
rightTh.dataset.buttonsAdded = true;
return true;
});
}
function handleNyaaSite() {
processMagnetLinks({
selectors: 'td.text-center a[href^="magnet:"]',
containerStyles: {
marginRight: '6px',
customStyles: {
display: 'inline-flex',
alignItems: 'center'
}
},
customProcessor: (magnetLink) => {
const tr = magnetLink.closest('tr');
const downloadBtn = tr?.querySelector("a[href^='/download/']");
const btnContainer = createButtonContainer({
marginRight: '6px',
customStyles: {
display: 'inline-flex',
alignItems: 'center'
}
});
const combinedBtn = createCombinedButtons(magnetLink);
btnContainer.appendChild(combinedBtn);
if (downloadBtn) {
downloadBtn.parentNode.insertBefore(btnContainer, downloadBtn);
} else {
magnetLink.parentNode.insertBefore(btnContainer, magnetLink.nextSibling);
}
}
});
processMagnetLinks({
selectors: '.panel-footer .card-footer-item[href^="magnet:"]',
containerStyles: { marginLeft: '10px' },
insertPosition: 'after'
});
}
function processMagnetLinks({ selectors, containerStyles = { marginLeft: '5px' }, insertPosition = 'after', customProcessor }) {
if (!Array.isArray(selectors)) {
selectors = [selectors];
}
selectors.forEach(selector => {
document.querySelectorAll(selector).forEach(element => {
if (element.dataset.buttonsAdded) return;
element.dataset.buttonsAdded = true;
if (customProcessor && typeof customProcessor === 'function') {
customProcessor(element);
return;
}
const btnContainer = createButtonContainer(containerStyles);
const combinedBtn = createCombinedButtons(element);
btnContainer.appendChild(combinedBtn);
if (insertPosition === 'before') {
element.parentNode.insertBefore(btnContainer, element);
} else {
element.parentNode.insertBefore(btnContainer, element.nextSibling);
}
});
});
}
function handleDMHYSite() {
const magnetHeader = document.querySelector('#topic_list th:nth-child(4)');
if (magnetHeader) {
magnetHeader.style.width = '18%';
}
processMagnetLinks({
selectors: 'a.download-arrow.arrow-magnet',
containerStyles: { marginLeft: '5px' },
insertPosition: 'before'
});
processMagnetLinks({
selectors: ['#tabs-1 a.magnet', '#tabs-1 a#magnet2'],
containerStyles: { marginLeft: '5px' },
insertPosition: 'after'
});
}
function handleGyingGygSite() {
try {
const dlWrapper = document.querySelector('div.down-link');
const table = dlWrapper?.querySelector('table.bit_list');
if (table) {
try {
if (!table.dataset.gyStyled) {
table.style.tableLayout = 'auto';
const downloadTh = table.querySelector('thead th:nth-child(2)');
if (downloadTh && !downloadTh.dataset.gyStyled) {
downloadTh.style.width = 'auto';
downloadTh.style.whiteSpace = 'nowrap';
downloadTh.dataset.gyStyled = '1';
}
table.dataset.gyStyled = '1';
}
} catch (_) {}
table.querySelectorAll('tbody tr').forEach(tr => {
const downloadTd = tr.querySelector('td:nth-child(2)');
if (!downloadTd) return;
if (!downloadTd.dataset.gyStyled) {
downloadTd.style.whiteSpace = 'nowrap';
downloadTd.style.width = '1%';
downloadTd.dataset.gyStyled = '1';
}
const magnetA = downloadTd.querySelector('a[href^="magnet:"]');
if (!magnetA) return;
const containerBefore = magnetA.previousElementSibling;
if (containerBefore && containerBefore.classList?.contains('magnet-action-buttons')) {
magnetA.dataset.buttonsAdded = 'true';
return;
}
const existingContainer = downloadTd.querySelector('.magnet-action-buttons');
if (existingContainer) {
magnetA.parentNode.insertBefore(existingContainer, magnetA);
magnetA.dataset.buttonsAdded = 'true';
return;
}
const btnContainer = createButtonContainer({
marginRight: '8px',
customStyles: {
display: 'inline-flex',
alignItems: 'center'
}
});
const combinedBtn = createCombinedButtons(magnetA.href);
btnContainer.appendChild(combinedBtn);
magnetA.parentNode.insertBefore(btnContainer, magnetA);
magnetA.dataset.buttonsAdded = true;
});
return;
}
} catch (_) {}
}
async function extractMagnetLink(element) {
try {
if (typeof element === 'string') {
return element.startsWith('magnet:') ? element : null;
}
const href = element?.href;
if (!href) return null;
if (href.startsWith('magnet:')) return href;
const extractors = [
{ test: 'seedhub', handler: fetchSeedhubMagnetFromDetail },
{ test: '/magnet/', handler: fetchBT4GMagnetFromDetail },
{ test: '/torrent/', handler: (url) => {
const match = url.match(/\/torrent\/([a-f0-9]+)\.html$/i);
return match?.[1] ? `magnet:?xt=urn:btih:${match[1]}` : null;
}},
{ test: '/detail/', handler: (url) => {
const match = url.match(/\/detail\/([a-f0-9]+)\.html$/i);
return match?.[1] ? `magnet:?xt=urn:btih:${match[1]}` : null;
}},
{ test: 'downloadtorrentfile.com/hash/', handler: (url) => {
const hashMatch = url.match(/hash\/([a-f0-9]+)/i);
if (!hashMatch?.[1]) return null;
const nameMatch = url.match(/[?&]name=([^&]+)/i);
return `magnet:?xt=urn:btih:${hashMatch[1]}${nameMatch?.[1] ? `&dn=${nameMatch[1]}` : ''}`;
}},
{ test: '/hash/', handler: (url) => {
const match = url.match(/\/hash\/([a-f0-9]+)\.html$/i);
return match?.[1] ? `magnet:?xt=urn:btih:${match[1]}` : null;
}}
];
for (const { test, handler } of extractors) {
if (href.includes(test)) {
return await handler(href);
}
}
return null;
} catch (error) {
showNotification('错误', error.message);
return null;
}
}
async function check115Login(forceCheck = false) {
try {
const isValid = await validate115Cookies();
return isValid;
} catch (error) {
console.error('检查登录状态失败:', error);
return false;
}
}
function validate115Cookies() {
return new Promise((resolve) => {
GM_xmlhttpRequest({
url: 'https://115.com/web/lixian/?ct=lixian&ac=task_lists&t=' + Date.now(),
method: 'GET',
anonymous: false,
headers: {
'Accept': 'application/json, text/plain, */*',
'Referer': 'https://115.com/web/lixian/'
},
onload: function(response) {
try {
const finalUrl = (response.finalUrl || '').toLowerCase();
const status = response.status || 0;
const headers = response.responseHeaders || '';
const contentType = (/^content-type:\s*([^\n]+)/im.exec(headers)?.[1] || '').toLowerCase();
const text = response.responseText || '';
const redirectedToLogin = /login\.115\.com|passport\.115\.com|passportapi\.115\.com/.test(finalUrl);
if (redirectedToLogin) return resolve(false);
let json = null;
if (contentType.includes('application/json') || text.trim().startsWith('{')) {
try { json = JSON.parse(text); } catch (_) { json = null; }
}
if (json) {
const ok = (json.state === true) || (json.errno === 0) || (json.code === 0) || (json.data && json.data.state === true);
if (ok) return resolve(true);
return resolve(false);
}
const hasLoginHints = /登录|请先登录|账户|sign\s*in|login|passport\.115\.com|window\.location|top\.location/i.test(text);
if (hasLoginHints) return resolve(false);
if (status === 401 || status === 403) return resolve(false);
return resolve(false);
} catch (_) {
return resolve(false);
}
},
onerror: () => resolve(false)
});
});
}
async function process115Offline(magnetLink) {
const notificationId = Date.now();
try {
showNotification('115离线', '正在检查登录状态...', notificationId);
const isLoggedIn = await check115Login(true);
if (!isLoggedIn) {
throw new Error('请先登录115网盘');
}
showNotification('115离线', '正在提交离线任务...', notificationId);
const result = await submit115OfflineTask(magnetLink);
handleOfflineResult(result);
} catch (error) {
showNotification('115离线失败', error.message);
if (error.message.includes('登录')) {
setTimeout(() => {
if (confirm('需要登录115网盘,是否进入115网盘登录页面?')) {
window.open('https://115.com/?mode=login', '_blank');
}
}, 500);
}
}
}
function submit115OfflineTask(magnetLink) {
return new Promise((resolve, reject) => {
GM_xmlhttpRequest({
url: `https://115.com/web/lixian/?ct=lixian&ac=add_task_url&url=${encodeURIComponent(magnetLink)}`,
method: 'GET',
anonymous: false,
headers: {
'Referer': 'https://115.com/web/lixian/'
},
onload: function(response) {
const result = tryParseJson(response.responseText);
resolve(result);
},
onerror: function() {
reject(new Error('提交离线任务失败'));
}
});
});
}
function tryParseJson(text) {
try {
return JSON.parse(text);
} catch (e) {
return null;
}
}
function handleOfflineResult(result) {
if (!result) {
showNotification('115离线失败', '接口返回为空或解析失败');
return;
}
const success = result.state === true || result.errno === 0 || result.code === 0;
const message = result.message || result.msg || result.error || result.errmsg || result?.data?.errmsg || '';
const err = Number.isFinite(result.errno) ? result.errno : (Number.isFinite(result.code) ? result.code : null);
const rawText = typeof result === 'string' ? result : JSON.stringify(result);
const txt = [message, rawText].filter(Boolean).join(' ');
const duplicateMsgRe = /(已存在|已经存在|已添加|重复|已提交过|exist|exists|already)/i;
const duplicateFlag = !!(
duplicateMsgRe.test(txt) ||
result?.data?.exists === true ||
result?.exists === true ||
[911, 10008, 10009, 10010].includes(err)
);
if (duplicateFlag) {
showNotification('115离线', '任务已存在');
return;
}
if (success) {
showNotification('115离线成功', message || '任务已提交');
} else {
const text = message || rawText;
showNotification('115离线失败', text);
}
}
function showNotification(title, text, id = null) {
if (id) {
const existing = document.getElementById(`notification-${id}`);
if (existing) existing.remove();
}
const container = document.createElement('div');
container.className = 'custom-notification';
container.id = id ? `notification-${id}` : `notification-${Date.now()}`;
Object.assign(container.style, {
position: 'fixed',
bottom: '20px',
right: '20px',
padding: '12px 16px',
background: 'rgba(255, 255, 255, 0.95)',
color: '#333',
borderRadius: '8px',
boxShadow: '0 4px 12px rgba(0, 0, 0, 0.15)',
zIndex: '2147483647',
maxWidth: '300px',
wordWrap: 'break-word',
opacity: '0',
transform: 'translateY(20px)',
transition: 'opacity 0.3s ease, transform 0.3s ease',
fontFamily: 'system-ui, -apple-system, BlinkMacSystemFont, sans-serif',
backdropFilter: 'blur(10px)',
border: '1px solid rgba(255, 255, 255, 0.2)',
cursor: 'pointer'
});
const titleEl = document.createElement('div');
titleEl.textContent = title;
Object.assign(titleEl.style, {
fontWeight: 'bold',
marginBottom: '4px'
});
const textEl = document.createElement('div');
try {
textEl.textContent = (text?.includes('%') || text?.includes('magnet:'))
? decodeURIComponent(text) : text;
} catch (e) {
textEl.textContent = text;
}
textEl.style.fontSize = '14px';
container.append(titleEl, textEl);
document.body.appendChild(container);
requestAnimationFrame(() => {
Object.assign(container.style, {
opacity: '1',
transform: 'translateY(0)'
});
});
const removeNotification = () => {
Object.assign(container.style, {
opacity: '0',
transform: 'translateY(20px)'
});
setTimeout(() => container.remove(), 300);
};
const timeoutId = setTimeout(removeNotification, CONFIG.notificationTimeout);
container.addEventListener('click', () => {
clearTimeout(timeoutId);
removeNotification();
});
}
function ensureModalStyles() {
if (document.getElementById('magnet-script-modal-styles')) return;
const style = document.createElement('style');
style.id = 'magnet-script-modal-styles';
style.textContent = `
.modal-overlay{position:fixed;inset:0;background:rgba(0,0,0,.08);z-index:10000;display:flex;align-items:center;justify-content:center;padding:16px}
.modal-content{background:#fff;color:#333;border-radius:10px;box-shadow:0 12px 40px rgba(0,0,0,.08);width:760px;max-width:96vw;padding:16px 18px;font-family:system-ui,-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica,Arial,sans-serif}
.modal-header{display:flex;align-items:center;justify-content:space-between;margin:0;padding:0;min-height:32px}
.modal-header-colored{background:linear-gradient(90deg, rgba(66,133,244,.12), rgba(66,133,244,0));padding:2px 8px;border-radius:4px;margin:0 0 1px;display:flex;align-items:center;min-height:inherit}
.modal-title{margin:0;font-size:18px;font-weight:700;color:#333;line-height:1;padding:2px 0}
.modal-header-help{font-size:12px;color:#666;margin-left:8px;line-height:1}
.modal-desc{font-size:12px;color:#666;margin-bottom:12px}
.modal-section-title{font-weight:600;margin:12px 0 6px;padding:6px 10px;background:rgba(66,133,244,.08);border-left:3px solid #4285f4;border-radius:4px;color:#333;font-size:14px;display:flex;align-items:center;min-height:32px;line-height:1.2}
.modal-section-header{display:flex;align-items:center;justify-content:space-between;gap:8px;min-height:32px;padding:2px 0}
.vip-badge{display:inline-flex;align-items:center;gap:6px;margin-left:10px;padding:0 8px;height:18px;line-height:16px;font-size:11px;border-radius:999px;white-space:nowrap;border:1px solid #e5e7eb;color:#374151;background:#f9fafb}
.vip-badge .dot{width:6px;height:6px;border-radius:50%;background:#9ca3af}
/* VIP 渐变金 */
.vip-badge.vip{color:#7c2d12;background:linear-gradient(90deg,#fde68a,#ffc10778,#fcd34d);border-color:#f59e0b}
.vip-badge.vip .dot{background:#f59e0b}
/* 未登录浅红 */
.vip-badge.not-logged{color:#991b1b;background:#fef2f2;border-color:#fecaca}
.vip-badge.not-logged .dot{background:#ef4444}
/* 原石/原始/非会员浅灰 */
.vip-badge.non-vip{color:#374151;background:#f3f4f6;border-color:#e5e7eb}
.vip-badge.non-vip .dot{background:#9ca3af}
.vip-badge .vip-action{appearance:none;border:none;background:transparent;color:#b45309;cursor:pointer;padding:0 0 0 6px;margin:0;border-left:1px solid #fcd34d;font-size:11px;line-height:16px}
.vip-badge .vip-action:hover{color:#92400e}
.modal-form-group{display:flex;align-items:center;justify-content:space-between;gap:10px;margin:8px 0}
.modal-label{min-width:160px;font-size:13px;color:#555}
.modal-form-texts{display:flex;flex-direction:column;gap:4px;flex:1}
.modal-help{font-size:12px;color:#999}
.modal-control{margin-left:auto}
.modal-indent{margin-left:20px}
.modal-row-center{display:flex;justify-content:center;align-items:center;gap:12px;margin:8px 0}
.modal-two-col{display:grid;grid-template-columns:1fr 1fr;gap:12px;align-items:start}
.modal-three-col{display:grid;grid-template-columns:repeat(3,1fr);gap:10px}
.modal-four-col{display:grid;grid-template-columns:repeat(4,1fr);gap:10px}
.modal-tile{display:flex;align-items:center;justify-content:space-between;background:#f8f9fb;border:1px solid #eee;border-radius:8px;padding:8px 10px}
.modal-tile-label{font-size:13px;color:#555}
.modal-tip{font-size:12px;color:#8a6d3b;margin-bottom:8px;background:#fff6e5;border:1px solid #ffe5b7;border-radius:6px;padding:8px 10px;text-align:center}
.modal-input{width:auto}
.modal-btn{padding:8px 12px;font-size:13px;border:none;border-radius:6px;cursor:pointer}
.modal-btn-primary{background:#4285f4;color:#fff}
.modal-btn-primary:hover{background:#3367d6}
.modal-btn-secondary{background:#f5f5f5;color:#333}
.modal-btn-secondary:hover{background:#eaeaea}
.modal-btn-success{background:#16a34a;color:#fff}
.modal-btn-success:hover{background:#15803d}
.modal-btn-danger{background:#ef4444;color:#fff}
.modal-btn-danger:hover{background:#dc2626}
.modal-btn-warning{background:#f59e0b;color:#fff}
.modal-btn-warning:hover{background:#d97706}
.modal-footer{display:flex;justify-content:flex-end;gap:10px;margin-top:12px}
.toggle{position:relative;display:inline-block;width:44px;height:24px;vertical-align:middle}
.toggle input{opacity:0;width:0;height:0;position:absolute}
.toggle-slider{position:absolute;cursor:pointer;top:0;left:0;right:0;bottom:0;background:#e5e7eb;transition:.2s ease;border-radius:9999px;box-shadow:inset 0 0 0 1px rgba(0,0,0,.08)}
.toggle-slider:before{content:"";position:absolute;height:18px;width:18px;left:3px;top:3px;background:#fff;border-radius:50%;transition:.2s ease;box-shadow:0 1px 2px rgba(0,0,0,.25)}
.toggle input:checked + .toggle-slider{background:#4285f4}
.toggle input:checked + .toggle-slider:before{transform:translateX(20px)}
.toggle-disabled .toggle-slider{opacity:.55;cursor:not-allowed}
.floating-settings-btn{position:fixed;left:18px;bottom:18px;width:44px;height:44px;border-radius:50%;background:rgba(255,255,255,.6);backdrop-filter:saturate(180%) blur(12px);-webkit-backdrop-filter:saturate(180%) blur(12px);color:#333;display:flex;align-items:center;justify-content:center;box-shadow:0 8px 24px rgba(0,0,0,.15), inset 0 0 0 1px rgba(0,0,0,.06);cursor:grab;user-select:none;z-index:10000;transition:background .2s ease, box-shadow .2s ease, transform .1s ease}
.floating-settings-btn:hover{background:rgba(255,255,255,.78);box-shadow:0 10px 28px rgba(0,0,0,.18), inset 0 0 0 1px rgba(0,0,0,.08)}
.floating-settings-btn.dragging{cursor:grabbing;opacity:.95;transform:scale(.98)}
`;
document.head.appendChild(style);
}
function ensureFloatingSettingsButton() {
const existing = document.getElementById('magnet-floating-settings-btn');
if (!CONFIG.enableFloatingSettingsBtn) {
if (existing) existing.remove();
return;
}
if (existing) return;
const btn = document.createElement('div');
btn.id = 'magnet-floating-settings-btn';
btn.className = 'floating-settings-btn';
btn.title = '设置';
btn.textContent = '⚙️';
btn.style.lineHeight = '44px';
btn.style.fontSize = '18px';
try {
const pos = GM_getValue('floatingBtnPos', null);
if (pos && typeof pos.left === 'number' && typeof pos.top === 'number') {
const w = 44, h = 44;
const maxX = Math.max(0, window.innerWidth - w);
const maxY = Math.max(0, window.innerHeight - h);
const safeLeft = Math.min(Math.max(0, pos.left), maxX);
const safeTop = Math.min(Math.max(0, pos.top), maxY);
btn.style.left = safeLeft + 'px';
btn.style.top = safeTop + 'px';
btn.style.right = 'auto';
btn.style.bottom = 'auto';
}
} catch (_) {}
btn.addEventListener('click', (e) => {
e.stopPropagation();
openSettingsPanel({ suppress115Toast: true });
});
btn.addEventListener('mousedown', (e) => {
if (e.button !== 0) return;
e.preventDefault();
const rect = btn.getBoundingClientRect();
const offsetX = e.clientX - rect.left;
const offsetY = e.clientY - rect.top;
btn.classList.add('dragging');
const onMove = (ev) => {
const x = Math.min(Math.max(0, ev.clientX - offsetX), window.innerWidth - rect.width);
const y = Math.min(Math.max(0, ev.clientY - offsetY), window.innerHeight - rect.height);
btn.style.left = x + 'px';
btn.style.top = y + 'px';
btn.style.right = 'auto';
btn.style.bottom = 'auto';
};
const onUp = () => {
document.removeEventListener('mousemove', onMove);
document.removeEventListener('mouseup', onUp);
btn.classList.remove('dragging');
try {
const left = parseFloat(btn.style.left || '0');
const top = parseFloat(btn.style.top || '0');
GM_setValue('floatingBtnPos', { left, top });
} catch (_) {}
};
document.addEventListener('mousemove', onMove);
document.addEventListener('mouseup', onUp);
});
document.body.appendChild(btn);
}
function openSettingsPanel(options = {}) {
const suppress115Toast = !!options.suppress115Toast;
const existing = document.getElementById('magnet-script-settings-overlay');
if (existing) existing.remove();
ensureModalStyles();
const overlay = document.createElement('div');
overlay.id = 'magnet-script-settings-overlay';
overlay.className = 'modal-overlay';
const panel = document.createElement('div');
panel.className = 'modal-content';
if (!document.getElementById('magnet-close-style')) {
const style = document.createElement('style');
style.id = 'magnet-close-style';
style.textContent = `
.modal-close-btn {
appearance: none;
background: rgba(66,133,244, .15);
color: #4285f4;
border: none;
width: 28px; height: 28px;
padding: 0; margin: 0;
border-radius: 999px;
display: inline-flex; align-items: center; justify-content: center;
border: none;
}
.modal-close-btn:hover { background: rgba(66,133,244, .25); }
.modal-close-btn:focus-visible { outline: none; box-shadow: 0 0 0 3px rgba(66,133,244,.35); }
.modal-close-icon { width: 16px; height: 16px; display: inline-block; transition: transform .35s ease; }
.modal-close-btn:hover .modal-close-icon { transform: rotate(180deg); }
`;
document.head.appendChild(style);
}
const header = document.createElement('div');
header.className = 'modal-header modal-header-colored';
const titleEl = document.createElement('h3');
titleEl.className = 'modal-title';
titleEl.textContent = '设置';
const headerRight = document.createElement('div');
headerRight.style.marginLeft = 'auto';
const btnCloseX = document.createElement('button');
btnCloseX.type = 'button';
btnCloseX.title = '关闭';
btnCloseX.className = 'modal-close-btn';
btnCloseX.innerHTML = `
`;
btnCloseX.addEventListener('click', () => overlay.remove());
headerRight.appendChild(btnCloseX);
header.append(titleEl, headerRight);
const section = (title) => {
const s = document.createElement('div');
const t = document.createElement('div');
t.textContent = title;
t.className = 'modal-section-title';
s.appendChild(t);
return s;
};
const row = (labelText, inputEl, helpText = '') => {
const r = document.createElement('div');
r.className = 'modal-form-group';
const texts = document.createElement('div');
texts.className = 'modal-form-texts';
const l = document.createElement('label');
l.textContent = labelText;
l.className = 'modal-label';
texts.appendChild(l);
if (helpText) {
const h = document.createElement('div');
h.textContent = helpText;
h.className = 'modal-help';
texts.appendChild(h);
}
r.appendChild(texts);
inputEl.classList?.add('modal-input', 'modal-control');
r.appendChild(inputEl);
return r;
};
const mkSwitch = (checked, onChange) => {
const wrapper = document.createElement('label');
wrapper.className = 'toggle';
const input = document.createElement('input');
input.type = 'checkbox';
input.checked = !!checked;
input.addEventListener('change', () => onChange(!!input.checked));
const slider = document.createElement('span');
slider.className = 'toggle-slider';
wrapper.append(input, slider);
Object.defineProperty(wrapper, 'disabled', {
get() { return input.disabled; },
set(v) { input.disabled = !!v; wrapper.classList.toggle('toggle-disabled', !!v); }
});
return wrapper;
};
const sec115 = section('115功能');
const sec115TitleEl = sec115.querySelector?.('.modal-section-title');
if (sec115TitleEl) {
sec115TitleEl.classList.add('modal-section-header');
sec115TitleEl.style.padding = '4px 8px';
const vipBadge = document.createElement('span');
vipBadge.className = 'vip-badge';
vipBadge.innerHTML = 'VIP 信息加载中...';
sec115TitleEl.appendChild(vipBadge);
const applyBadgeStatus = (status) => {
vipBadge.classList.remove('vip', 'not-logged', 'non-vip');
if (status) vipBadge.classList.add(status);
};
const setVipBadge = (txt, opts = {}) => {
const { withLoginBtn = false, status = null } = opts;
const t = vipBadge.querySelector('.text');
if (t) t.textContent = txt;
applyBadgeStatus(status);
const oldBtn = vipBadge.querySelector('.vip-action');
if (oldBtn) oldBtn.remove();
if (withLoginBtn) {
const btn = document.createElement('button');
btn.type = 'button';
btn.className = 'vip-action';
btn.textContent = '去登录';
btn.addEventListener('click', (e) => {
e.stopPropagation();
window.open('https://115.com/?mode=login', '_blank');
});
vipBadge.appendChild(btn);
}
};
const CACHE_TTL = 10 * 60 * 1000;
const readCache = (key) => { try { return GM_getValue(key, null); } catch (_) { return null; } };
const writeCache = (key, data) => { try { GM_setValue(key, { t: Date.now(), data }); } catch (_) {} };
const useCacheValid = (key) => {
const c = readCache(key);
return c && (Date.now() - (c.t || 0) < CACHE_TTL) ? c.data : null;
};
let latestVip = null;
let vipFetched = false;
let latestQuota = null;
const updateBadge = () => {
const okVip = latestVip && latestVip.state === true;
const lvName = okVip ? (latestVip?.data?.user_limit?.level_name || '') : '';
const quotaOk = latestQuota && typeof latestQuota.surplus !== 'undefined' && typeof latestQuota.count !== 'undefined';
const isNonVip = lvName && /原石|原始|非会员/i.test(lvName);
const statusClass = !lvName ? null : (isNonVip ? 'non-vip' : 'vip');
if (!vipFetched) {
setVipBadge('VIP 信息加载中...');
return;
}
if (!okVip && !lvName) {
setVipBadge('未登录或无权限', { withLoginBtn: true, status: 'not-logged' });
return;
}
if (quotaOk) {
const text = (lvName ? `等级:${lvName},` : '') + `离线额度:${latestQuota.surplus}/${latestQuota.count}`;
setVipBadge(text, { status: statusClass });
} else if (lvName) {
setVipBadge(`等级:${lvName}`, { status: statusClass });
} else {
setVipBadge('未登录或无权限', { withLoginBtn: true, status: 'not-logged' });
}
};
const fetchVip = (bypassCache = false) => {
if (!bypassCache) {
const cached = useCacheValid('vip_info_cache_v2');
if (cached) { latestVip = cached; updateBadge(); }
}
GM_xmlhttpRequest({
url: 'https://webapi.115.com/user/vip_limit?feature=2',
method: 'GET',
anonymous: false,
headers: {
'Accept': 'application/json, text/plain, */*',
'Referer': 'https://115.com/web/lixian/'
},
onload: function (resp) {
try {
latestVip = JSON.parse(resp.responseText || 'null');
vipFetched = true;
writeCache('vip_info_cache_v2', latestVip);
updateBadge();
} catch (_) {
vipFetched = true;
updateBadge();
}
},
onerror: function () { vipFetched = true; updateBadge(); }
});
};
const fetchQuota = (bypassCache = false) => {
if (!bypassCache) {
const cached = useCacheValid('vip_quota_cache_v1');
if (cached) { latestQuota = cached; updateBadge(); }
}
GM_xmlhttpRequest({
url: 'https://115.com/web/lixian/?ct=lixian&ac=get_quota_package_info',
method: 'GET',
anonymous: false,
headers: {
'Accept': 'application/json, text/plain, */*',
'Referer': 'https://115.com/web/lixian/'
},
onload: function (resp) {
try {
const json = JSON.parse(resp.responseText || 'null');
const data = {
surplus: Number(json?.surplus ?? json?.package?.["1"]?.surplus ?? 0),
count: Number(json?.count ?? json?.package?.["1"]?.count ?? 0),
used: Number(json?.used ?? json?.package?.["1"]?.used ?? 0),
};
latestQuota = data;
writeCache('vip_quota_cache_v1', data);
updateBadge();
} catch (_) {
updateBadge();
}
},
onerror: function () { updateBadge(); }
});
};
fetchVip(false);
fetchQuota(false);
const origCheck = check115AndUpdate;
const bindRefresh = () => {
if (typeof check115AndUpdate === 'function' && check115AndUpdate !== origCheck) return;
if (typeof check115AndUpdate === 'function') {
const wrapped = async (notify = true) => {
const res = await origCheck(notify);
fetchVip(true);
fetchQuota(true);
return res;
};
window.check115AndUpdate = wrapped;
} else {
setTimeout(bindRefresh, 50);
}
};
bindRefresh();
}
const statusBtn = document.createElement('button');
statusBtn.textContent = '检测115登录状态';
statusBtn.className = 'modal-btn modal-btn-secondary';
const setStatus = (type) => {
statusBtn.className = 'modal-btn';
switch (type) {
case 'checking':
statusBtn.classList.add('modal-btn-warning');
statusBtn.textContent = '检测中...';
break;
case 'ok':
statusBtn.classList.add('modal-btn-success');
statusBtn.textContent = '账号已登录';
break;
case 'no':
statusBtn.classList.add('modal-btn-danger');
statusBtn.textContent = '当前浏览器未登录';
break;
default:
statusBtn.classList.add('modal-btn-warning');
statusBtn.textContent = '检测异常';
}
};
async function check115AndUpdate(notify = true) {
try {
setStatus('checking');
const ok = await check115Login(true);
setStatus(ok ? 'ok' : 'no');
if (notify) showNotification('115状态', ok ? '已登录' : '未登录');
} catch (e) {
console.error(e);
setStatus('error');
if (notify) showNotification('115状态', '检测异常');
}
}
window.check115AndUpdate = check115AndUpdate;
statusBtn.addEventListener('click', () => window.check115AndUpdate(true));
const btnOpen115 = document.createElement('button');
btnOpen115.textContent = '打开115网盘';
btnOpen115.className = 'modal-btn modal-btn-primary';
btnOpen115.addEventListener('click', () => window.open('https://115.com/?cid=0&offset=0&mode=wangpan', '_blank'));
const row115 = document.createElement('div');
row115.className = 'modal-row-center';
row115.append(statusBtn, btnOpen115);
sec115.appendChild(row115);
setTimeout(() => window.check115AndUpdate(!suppress115Toast), 100);
const secButtons = section('按钮显示');
const btnsTitleEl = secButtons.querySelector?.('.modal-section-title');
if (btnsTitleEl) {
btnsTitleEl.classList.add('modal-section-header');
btnsTitleEl.style.padding = '4px 8px';
const btnsTip = document.createElement('span');
btnsTip.className = 'modal-tip';
btnsTip.style.margin = '0 0 0 10px';
btnsTip.style.padding = '0 6px';
btnsTip.style.fontSize = '11px';
btnsTip.style.lineHeight = '16px';
btnsTip.style.height = '18px';
btnsTip.style.display = 'inline-flex';
btnsTip.style.alignItems = 'center';
btnsTip.style.boxSizing = 'border-box';
btnsTip.textContent = '提示:可通过拖拽右侧三项调整顺序;开关控制子按钮显示/隐藏,变更将即时应用到页面。';
btnsTitleEl.appendChild(btnsTip);
}
const swCopy = mkSwitch(CONFIG.enableCopyButton, (v) => { CONFIG.enableCopyButton = v; GM_setValue('enableCopyButton', v); showNotification('设置已保存', v ? '已启用“复制”按钮' : '已禁用“复制”按钮'); addActionButtons(); });
const swOffline = mkSwitch(CONFIG.enableOfflineButton, (v) => { CONFIG.enableOfflineButton = v; GM_setValue('enableOfflineButton', v); showNotification('设置已保存', v ? '已启用“离线”按钮' : '已禁用“离线”按钮'); addActionButtons(); });
const swOpen = mkSwitch(CONFIG.enableOpenButton, (v) => { CONFIG.enableOpenButton = v; GM_setValue('enableOpenButton', v); showNotification('设置已保存', v ? '已启用“打开”按钮' : '已禁用“打开”按钮'); addActionButtons(); });
const btnGrid = document.createElement('div');
btnGrid.className = 'modal-four-col';
const makeBtnTile = (label, sw) => {
const item = document.createElement('div');
item.className = 'modal-tile';
const name = document.createElement('span');
name.className = 'modal-tile-label';
name.textContent = label;
const right = document.createElement('div');
right.appendChild(sw);
item.append(name, right);
return item;
};
const tileCopy = makeBtnTile('复制按钮', swCopy); tileCopy.dataset.type = 'copy'; tileCopy.draggable = true;
const tileOffline = makeBtnTile('离线按钮', swOffline); tileOffline.dataset.type = 'offline'; tileOffline.draggable = true;
const tileOpen = makeBtnTile('打开按钮', swOpen); tileOpen.dataset.type = 'open'; tileOpen.draggable = true;
const tilesMap = { copy: tileCopy, offline: tileOffline, open: tileOpen };
const renderTilesByOrder = () => {
btnGrid.innerHTML = '';
const ord = getButtonsOrder();
ord.forEach(t => { const el = tilesMap[t]; if (el) btnGrid.appendChild(el); });
};
renderTilesByOrder();
let draggingEl = null;
btnGrid.addEventListener('dragstart', (e) => {
const tile = e.target.closest('.modal-tile');
if (!tile) return;
draggingEl = tile;
e.dataTransfer.effectAllowed = 'move';
try { e.dataTransfer.setData('text/plain', tile.dataset.type || ''); } catch (_) {}
tile.style.opacity = '0.6';
});
btnGrid.addEventListener('dragend', () => {
if (draggingEl) draggingEl.style.opacity = '';
draggingEl = null;
const newOrder = Array.from(btnGrid.querySelectorAll('.modal-tile')).map(el => el.dataset.type).filter(Boolean);
if (newOrder.length) {
setButtonsOrder(newOrder);
applyButtonsOrderToExisting();
}
});
btnGrid.addEventListener('dragover', (e) => {
e.preventDefault();
const tile = e.target.closest('.modal-tile');
if (!tile || !draggingEl || tile === draggingEl) return;
const rect = tile.getBoundingClientRect();
const before = (e.clientY - rect.top) < rect.height / 2;
btnGrid.insertBefore(draggingEl, before ? tile : tile.nextSibling);
});
secButtons.appendChild(btnGrid);
applyButtonsOrderToExisting();
const secFloat = section('悬浮设置按钮');
const swFloat = mkSwitch(CONFIG.enableFloatingSettingsBtn, (v) => {
CONFIG.enableFloatingSettingsBtn = v;
GM_setValue('enableFloatingSettingsBtn', v);
showNotification('设置已保存', v ? '已启用悬浮设置按钮' : '已关闭悬浮设置按钮');
ensureFloatingSettingsButton();
});
secFloat.appendChild(row('启用左下角悬浮设置按钮', swFloat, '可拖动,点击打开“设置”'));
const secAuto = document.createElement('div');
const secAutoHeader = document.createElement('div');
secAutoHeader.className = 'modal-section-title modal-section-header';
const secAutoTitle = document.createElement('span');
secAutoTitle.textContent = '自动异步获取磁力链';
secAutoHeader.appendChild(secAutoTitle);
secAutoHeader.style.padding = '4px 8px';
const riskTip = document.createElement('span');
riskTip.textContent = '提示:自动异步获取可能触发部分站点风控,按需开启。';
riskTip.className = 'modal-tip';
riskTip.style.margin = '0 0 0 10px';
riskTip.style.padding = '0 6px';
riskTip.style.fontSize = '11px';
riskTip.style.lineHeight = '16px';
riskTip.style.height = '18px';
riskTip.style.display = 'inline-flex';
riskTip.style.alignItems = 'center';
riskTip.style.boxSizing = 'border-box';
secAutoHeader.appendChild(riskTip);
const masterSwitch = mkSwitch(CONFIG.autoFetchEnabled, (v) => {
CONFIG.autoFetchEnabled = v; GM_setValue('autoFetchEnabled', v);
showNotification('设置已保存', v ? '已开启自动异步获取' : '已关闭自动异步获取');
[swBt4g, swSeedhub, swYhg, swCmg].forEach(sw => sw.disabled = !v);
addActionButtons();
});
secAutoHeader.appendChild(masterSwitch);
secAuto.appendChild(secAutoHeader);
const swBt4g = mkSwitch(CONFIG.autoFetchSites.bt4g, (v) => {
CONFIG.autoFetchSites.bt4g = v; GM_setValue('autoFetch_bt4g', v); addActionButtons();
});
const swSeedhub = mkSwitch(CONFIG.autoFetchSites.seedhub, (v) => {
CONFIG.autoFetchSites.seedhub = v; GM_setValue('autoFetch_seedhub', v); addActionButtons();
});
const swYhg = mkSwitch(CONFIG.autoFetchSites.yuhuage, (v) => {
CONFIG.autoFetchSites.yuhuage = v; GM_setValue('autoFetch_yuhuage', v); addActionButtons();
});
const swCmg = mkSwitch(CONFIG.autoFetchSites.cilimag, (v) => {
CONFIG.autoFetchSites.cilimag = v; GM_setValue('autoFetch_cilimag', v); addActionButtons();
});
[swBt4g, swSeedhub, swYhg, swCmg].forEach(sw => sw.disabled = !CONFIG.autoFetchEnabled);
const grid = document.createElement('div');
grid.className = 'modal-four-col';
const makeTile = (label, sw) => {
const item = document.createElement('div');
item.className = 'modal-tile';
const name = document.createElement('span');
name.className = 'modal-tile-label';
const m = label.match(/^(.*)\((.*)\)$/);
if (m) {
let left = (m[1] || '').trim();
let inside = (m[2] || '').trim();
const hasCn = (s) => /[\u4e00-\u9fff]/.test(s);
if (!hasCn(left) && hasCn(inside)) {
[left, inside] = [inside, left];
}
name.innerHTML = `${left}
(${inside})`;
} else {
name.textContent = label;
}
const right = document.createElement('div');
right.appendChild(sw);
item.append(name, right);
return item;
};
grid.append(
makeTile('BT4G', swBt4g),
makeTile('SeedHub', swSeedhub),
makeTile('雨花阁(YuHuaGe)', swYhg),
makeTile('ØMagnet(无极磁链)', swCmg)
);
secAuto.appendChild(grid);
const secSites = section('网站规则');
const sitesTitleEl = secSites.querySelector?.('.modal-section-title');
if (sitesTitleEl) {
sitesTitleEl.classList.add('modal-section-header');
sitesTitleEl.style.padding = '4px 8px';
}
const sitesTip = document.createElement('span');
sitesTip.className = 'modal-tip';
sitesTip.style.margin = '0 0 0 10px';
sitesTip.style.padding = '0 6px';
sitesTip.style.fontSize = '11px';
sitesTip.style.lineHeight = '16px';
sitesTip.style.height = '18px';
sitesTip.style.display = 'inline-flex';
sitesTip.style.alignItems = 'center';
sitesTip.style.boxSizing = 'border-box';
sitesTip.textContent = '测速提示:频繁测速请求可能导致站点风控,建议合理使用测速,必要时放慢操作节奏。';
const sitesHeader = secSites.querySelector?.('.modal-section-title') || secSites;
sitesHeader.appendChild(sitesTip);
const siteGrid = document.createElement('div');
siteGrid.className = 'modal-four-col';
if (!document.getElementById('magnet-site-links-icon-style')) {
const style = document.createElement('style');
style.id = 'magnet-site-links-icon-style';
style.textContent = `
.site-link-icon {\n\
background-image: url("data:image/svg+xml,%3Csvg width='14' height='14' viewBox='0 0 24 24' fill='none' stroke='currentColor' stroke-width='2' xmlns='http://www.w3.org/2000/svg'%3E%3Ccircle cx='12' cy='12' r='10'/%3E%3Cpath d='M2 12h20'/%3E%3Cpath d='M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z'/%3E%3C/svg%3E");\n\
background-repeat: no-repeat;\n\
background-size: contain;\n\
width: 14px;\n\
height: 14px;\n\
display: inline-block;\n\
vertical-align: -2px;\n\
}\n\
.modal-btn:hover .site-link-icon {\n\
background-image: url("data:image/svg+xml,%3Csvg width='14' height='14' viewBox='0 0 24 24' fill='none' stroke='%234caf50' stroke-width='2' xmlns='http://www.w3.org/2000/svg'%3E%3Ccircle cx='12' cy='12' r='10'/%3E%3Cpath d='M2 12h20'/%3E%3Cpath d='M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z'/%3E%3C/svg%3E");\n\
}\n `;
document.head.appendChild(style);
}
const showLinksPopover = (triggerEl, links) => {
document.querySelectorAll('.site-links-popover').forEach(p => p.remove());
const pop = document.createElement('div');
pop.className = 'site-links-popover';
pop.style.cssText = `
position: absolute;
z-index: 100000;
max-width: 420px;
padding: 10px;
background: #111;
color: #ddd;
border-radius: 6px;
box-shadow: 0 8px 20px rgba(0,0,0,.5);
line-height: 1.6;
`;
const list = document.createElement('div');
const tests = [];
const mkGroup = (title, arr, isPublish) => {
if (!arr || !arr.length) return;
const h = document.createElement('div');
h.textContent = title;
h.style.cssText = 'font-size:12px; color:#E5E7EB; background:#1f2937; text-align:center; padding:4px 8px; border-radius:4px; margin:8px 0 6px; border:1px solid #374151;';
list.appendChild(h);
arr.forEach((item, i) => {
const row = document.createElement('div');
row.style.cssText = 'display:flex; align-items:center; gap:8px; padding:2px 0;';
const idx = document.createElement('span');
idx.textContent = String(i + 1);
idx.style.cssText = 'flex:0 0 auto; width:18px; height:18px; display:inline-flex; align-items:center; justify-content:center; border-radius:50%; background:#334155; color:#E5E7EB; font-size:12px; border:1px solid #475569;';
const a = document.createElement('a');
a.href = item.url.startsWith('http') ? item.url : ('https://' + item.url.replace(/^\/+/, ''));
a.textContent = item.url + (item.note ? `(${item.note})` : '');
a.target = '_blank';
a.rel = 'noopener noreferrer';
a.style.cssText = `
flex: 1 1 auto; min-width:0; overflow:hidden; text-overflow:ellipsis; white-space:nowrap;
color:${isPublish ? '#F59E0B' : '#60A5FA'}; text-decoration:none;
`;
const badge = document.createElement('span');
badge.textContent = '测速中…';
badge.style.cssText = 'flex:0 0 auto; font-size:12px; color:#9CA3AF; border:1px solid #374151; padding:0 6px; border-radius:10px;';
row.appendChild(idx);
row.appendChild(a);
row.appendChild(badge);
list.appendChild(row);
tests.push(() => new Promise((resolve) => {
try {
const start = performance.now();
GM_xmlhttpRequest({
url: a.href,
method: 'GET',
headers: { 'Cache-Control': 'no-cache' },
timeout: 3000,
anonymous: true,
onload: () => {
const ms = Math.max(1, Math.round(performance.now() - start));
badge.textContent = ms + 'ms';
if (ms < 500) {
badge.style.color = '#10B981';
badge.style.borderColor = '#065F46';
} else if (ms < 1500) {
badge.style.color = '#F59E0B';
badge.style.borderColor = '#92400E';
} else {
badge.style.color = '#EF4444';
badge.style.borderColor = '#7F1D1D';
}
resolve();
},
onerror: () => {
badge.textContent = 'ERR';
badge.style.color = '#EF4444';
badge.style.borderColor = '#7F1D1D';
resolve();
},
ontimeout: () => {
badge.textContent = 'ERR';
badge.style.color = '#EF4444';
badge.style.borderColor = '#7F1D1D';
resolve();
}
});
} catch (_) {
badge.textContent = 'ERR';
badge.style.color = '#EF4444';
badge.style.borderColor = '#7F1D1D';
resolve();
}
}));
});
};
mkGroup('站点', links.sites || [], false);
mkGroup('发布页', links.publish || [], true);
if (!list.children.length) {
list.textContent = '暂无收录地址';
list.style.color = '#888';
}
pop.appendChild(list);
(async () => {
for (const job of tests) {
try { await job(); } catch (_) { /* ignore */ }
}
})();
document.body.appendChild(pop);
const rect = triggerEl.getBoundingClientRect();
pop.style.top = `${rect.bottom + window.scrollY + 6}px`;
pop.style.left = `${Math.min(rect.left + window.scrollX, window.scrollX + window.innerWidth - pop.offsetWidth - 12)}px`;
const close = (e) => {
if (!pop.contains(e.target) && e.target !== triggerEl) {
pop.remove();
document.removeEventListener('mousedown', close, true);
window.removeEventListener('scroll', close, true);
window.removeEventListener('resize', close, true);
}
};
document.addEventListener('mousedown', close, true);
window.addEventListener('scroll', close, true);
window.addEventListener('resize', close, true);
};
const addSiteTile = (label, key, gmKey) => {
const sw = mkSwitch(!!CONFIG.siteEnabled[key], (v) => {
CONFIG.siteEnabled[key] = v;
GM_setValue(gmKey, v);
showNotification('站点规则', `${label}已${v ? '启用' : '禁用'}`);
addActionButtons();
});
const item = document.createElement('div');
item.className = 'modal-tile';
const name = document.createElement('span');
name.className = 'modal-tile-label';
const m = label.match(/^(.*)\((.*)\)$/);
if (m) {
let left = (m[1] || '').trim();
let inside = (m[2] || '').trim();
const hasCn = (s) => /[\u4e00-\u9fff]/.test(s);
if (!hasCn(left) && hasCn(inside)) {
[left, inside] = [inside, left];
}
name.innerHTML = `${left}
(${inside})`;
} else {
name.textContent = label;
}
const right = document.createElement('div');
right.style.display = 'flex';
right.style.alignItems = 'center';
right.style.gap = '6px';
const linkBtn = document.createElement('button');
linkBtn.innerHTML = '';
linkBtn.className = 'modal-btn';
linkBtn.style.cssText = 'padding:2px 6px;font-size:12px;background:transparent;color:#6B7280;border:none;border-radius:4px;';
linkBtn.title = '地址';
linkBtn.addEventListener('click', (e) => {
e.stopPropagation();
const links = SITES_LINKS[key] || {};
showLinksPopover(linkBtn, links);
});
right.appendChild(linkBtn);
right.appendChild(sw);
item.append(name, right);
siteGrid.appendChild(item);
};
addSiteTile('BT4G', 'bt4g', 'site_bt4g');
addSiteTile('BTDigg', 'btdig', 'site_btdig');
addSiteTile('BTSOW', 'btsow', 'site_btsow');
addSiteTile('Nyaa', 'nyaa', 'site_nyaa');
addSiteTile('動漫花園(DMHY)', 'dmhy', 'site_dmhy');
addSiteTile('观影', 'gying_family', 'site_gying_family');
addSiteTile('SeedHub', 'seedhub', 'site_seedhub');
addSiteTile('梦幻天堂·龙网(LongWangBT)', 'longwangbt', 'site_longwangbt');
addSiteTile('雨花阁(YuHuaGe)', 'yuhuage', 'site_yuhuage');
addSiteTile('SOBT', 'sobt', 'site_sobt');
addSiteTile('CLB(磁力宝)', 'clb', 'site_clb');
addSiteTile('BT目录(BTMulu)', 'btmulu', 'site_btmulu');
addSiteTile('ØMagnet(无极磁链)', 'cili_family', 'site_cili_family');
addSiteTile('磁力帝', 'cilidi', 'site_cilidi');
secSites.appendChild(siteGrid);
const topRow = document.createElement('div');
topRow.className = 'modal-two-col';
topRow.append(sec115, secFloat);
panel.append(header, topRow, secButtons, secSites, secAuto);
overlay.appendChild(panel);
document.body.appendChild(overlay);
}
function handleSeedhubSite() {
processElements('.seeds a', (linkElement) => {
const btnContainer = createButtonContainer({
marginRight: '8px',
customStyles: {
display: 'inline-block',
verticalAlign: 'middle'
}
});
if (isAutoFetchEnabledFor('seedhub')) {
const loadingBtn = createLoadingButton();
btnContainer.appendChild(loadingBtn);
processSeedhubMagnetLink(linkElement, btnContainer).then(success => {
if (!success) {
setupRetryButton(loadingBtn, () =>
processSeedhubMagnetLink(linkElement, btnContainer, 2, 6000)
);
}
}).catch(error => {
console.error('SeedHub处理失败:', error);
setButtonError(loadingBtn, '处理失败');
});
} else {
const combinedBtn = createCombinedButtons(async () => {
if (linkElement?.href?.startsWith('magnet:')) return linkElement.href;
return await fetchSeedhubMagnetFromDetail(linkElement.href);
});
btnContainer.appendChild(combinedBtn);
}
linkElement.parentNode.insertBefore(btnContainer, linkElement);
return true;
});
}
async function processSeedhubMagnetLink(linkElement, btnContainer, maxRetries = CONFIG.maxRetries, timeout = CONFIG.defaultTimeout) {
if (!linkElement?.href) return false;
return await retryOperation(async (attempt) => {
const magnetLink = await Promise.race([
fetchSeedhubMagnetFromDetail(linkElement.href),
new Promise((_, reject) =>
setTimeout(() => reject(new Error('请求超时')), timeout)
)
]);
if (magnetLink) {
btnContainer.innerHTML = '';
btnContainer.appendChild(createCombinedButtons(magnetLink));
return true;
}
throw new Error('未获取到磁力链');
}, maxRetries, (attempt, maxRetries) => {
const loadingBtn = btnContainer.querySelector('.magnet-loading-btn');
if (loadingBtn) {
loadingBtn.textContent = `重试中(${attempt + 1}/${maxRetries})...`;
}
});
}
function handleYuhuageSite() {
processElements('.search-item .item-title h3 > a[href^="/hash/"]', (titleLink) => {
const btnContainer = createButtonContainer({ marginRight: '8px' });
titleLink.parentNode.insertBefore(btnContainer, titleLink);
if (isAutoFetchEnabledFor('yuhuage')) {
const loadingBtn = createLoadingButton();
btnContainer.appendChild(loadingBtn);
processYuhuageMagnetLink(titleLink, btnContainer).then(success => {
if (!success) {
setupRetryButton(loadingBtn, () =>
processYuhuageMagnetLink(titleLink, btnContainer, 2, 6000)
);
}
}).catch(error => {
console.error('Yuhuage处理失败:', error);
setButtonError(loadingBtn, '处理失败');
});
} else {
const combinedBtn = createCombinedButtons(async () => {
return await fetchYuhuageMagnetFromDetail(titleLink.href);
});
btnContainer.appendChild(combinedBtn);
}
return true;
}, 'yuhuageButtonsAdded');
processElements('.detail-panel .panel-header', (panelHeader) => {
const magnetIcon = panelHeader.querySelector('i.fa.fa-magnet');
if (!magnetIcon) return false;
const panelBody = panelHeader.nextElementSibling;
const magnetLink = panelBody?.querySelector('a.download[href^="magnet:"]');
if (!magnetLink) return false;
const btnContainer = createButtonContainer({
marginLeft: '10px',
customStyles: {
display: 'inline-flex',
alignItems: 'center'
}
});
const combinedBtn = createCombinedButtons(magnetLink.href);
btnContainer.appendChild(combinedBtn);
panelHeader.appendChild(btnContainer);
return true;
}, 'yuhuagePanelProcessed');
}
async function fetchSeedhubMagnetFromDetail(detailHref) {
try {
const html = await fetchWithRetry(detailHref);
const encodedMatch = html.match(/data = "([a-zA-Z0-9]+)"/);
if (encodedMatch?.[1]) {
const magnetLink = atob(encodedMatch[1]);
if (magnetLink?.startsWith('magnet:')) {
return magnetLink;
}
}
return null;
} catch (error) {
console.error('获取Seedhub磁力链失败:', error);
return null;
}
}
async function fetchYuhuageMagnetFromDetail(detailHref) {
try {
const html = await fetchWithRetry(detailHref);
const parser = new DOMParser();
const doc = parser.parseFromString(html, 'text/html');
const magnetLink = doc.querySelector('.detail-panel .panel-body a.download[href^="magnet:"]');
if (magnetLink?.href) {
return magnetLink.href.trim();
}
const magnetMatch = html.match(/magnet:\?xt=urn:btih:[a-f0-9]+[^"'>\s]*/i);
if (magnetMatch?.[0]) {
return magnetMatch[0].trim();
}
return null;
} catch (error) {
console.error('获取Yuhuage磁力链失败:', error);
return null;
}
}
async function processYuhuageMagnetLink(linkElement, btnContainer, maxRetries = CONFIG.maxRetries, timeout = CONFIG.defaultTimeout) {
if (!linkElement?.href) return false;
return await retryOperation(async (attempt) => {
const magnetLink = await Promise.race([
fetchYuhuageMagnetFromDetail(linkElement.href),
new Promise((_, reject) =>
setTimeout(() => reject(new Error('请求超时')), timeout)
)
]);
if (magnetLink) {
btnContainer.innerHTML = '';
btnContainer.appendChild(createCombinedButtons(magnetLink));
return true;
}
throw new Error('未获取到磁力链');
}, maxRetries, (attempt, maxRetries) => {
const loadingBtn = btnContainer.querySelector('.magnet-loading-btn');
if (loadingBtn) {
loadingBtn.textContent = `重试中(${attempt + 1}/${maxRetries})...`;
}
});
}
function createLoadingButton() {
const loadingBtn = document.createElement('span');
loadingBtn.className = 'magnet-loading-btn';
loadingBtn.textContent = '获取中...';
loadingBtn.style.cssText = 'font-size:12px;color:#666;padding:2px 6px;border:1px solid rgba(0,0,0,0.14);border-radius:4px;background-color:transparent;';
return loadingBtn;
}
function setButtonError(button, message = '获取失败') {
if (!button) return;
button.textContent = message;
button.style.color = '#ff4d4f';
}
async function processMagnetLink(linkElement, btnContainer, maxRetries = CONFIG.maxRetries, timeout = CONFIG.defaultTimeout) {
if (!linkElement?.href) return false;
return await retryOperation(async (attempt) => {
const magnetLink = await Promise.race([
fetchMagnetFromDetailPage(linkElement.href),
new Promise((_, reject) =>
setTimeout(() => reject(new Error('请求超时')), timeout)
)
]);
if (magnetLink) {
btnContainer.innerHTML = '';
btnContainer.appendChild(createCombinedButtons(magnetLink));
return true;
}
throw new Error('未获取到磁力链');
}, maxRetries, (attempt, maxRetries) => {
const loadingBtn = btnContainer.querySelector('.magnet-loading-btn');
if (loadingBtn) {
loadingBtn.textContent = `重试中(${attempt + 1}/${maxRetries})...`;
}
});
}
function handleCiliMagSite() {
processElements('table.table.table-hover.file-list tbody tr', (row) => {
const linkElement = row.querySelector('td a[href^="/"]');
if (!linkElement) return false;
const btnContainer = createButtonContainer({ marginRight: '8px' });
linkElement.parentNode.insertBefore(btnContainer, linkElement);
if (isAutoFetchEnabledFor('cilimag')) {
const loadingBtn = createLoadingButton();
btnContainer.appendChild(loadingBtn);
processMagnetLink(linkElement, btnContainer).then(success => {
if (!success) {
setupRetryButton(loadingBtn, () =>
processMagnetLink(linkElement, btnContainer, 2, 6000)
);
}
}).catch(error => {
console.error('CiliMag处理失败:', error);
setButtonError(loadingBtn, '处理失败');
});
} else {
const combinedBtn = createCombinedButtons(async () => {
return await fetchMagnetFromDetailPage(linkElement.href);
});
btnContainer.appendChild(combinedBtn);
}
return true;
}, 'ciliMagProcessed');
processElements('div.input-group.magnet-box', (magnetBox) => {
const magnetInput = magnetBox.querySelector('input[id="input-magnet"][value^="magnet:"]');
const addonElement = magnetBox.querySelector('.input-group-addon');
if (!magnetInput?.value.trim() || !addonElement) return false;
if (addonElement.classList.contains('magnet-prefix')) {
addonElement.style.padding = '2px 5px';
}
const btnContainer = createButtonContainer({
marginLeft: '5px',
customStyles: { display: 'inline-flex', alignItems: 'center' }
});
const combinedBtn = createCombinedButtons(magnetInput.value.trim());
btnContainer.appendChild(combinedBtn);
addonElement.appendChild(btnContainer);
return true;
}, 'magnetBoxProcessed');
}
async function fetchMagnetFromDetailPage(detailHref) {
try {
const html = await fetchWithRetry(detailHref, {
headers: { 'User-Agent': navigator.userAgent }
});
const parser = new DOMParser();
const doc = parser.parseFromString(html, 'text/html');
const magnetInput = doc.querySelector('input[id="input-magnet"][value^="magnet:"]');
if (magnetInput?.value) {
return magnetInput.value.trim();
}
const magnetLink = doc.querySelector('a[href^="magnet:"]');
if (magnetLink?.href) {
return magnetLink.href.trim();
}
const magnetMatch = html.match(/magnet:\?xt=urn:btih:[a-f0-9]+[^"'>]+/i);
if (magnetMatch?.[0]) {
return magnetMatch[0].trim();
}
return null;
} catch (error) {
console.error('从详情页获取磁力链失败:', error);
return null;
}
}
function handleLongwangbtSite() {
processElements('td.text_left a[href^="show.php?hash="]', (titleLink) => {
const hashMatch = titleLink.href.match(/hash=([a-f0-9]{40})/i);
if (!hashMatch) return false;
const hash = hashMatch[1];
const titleText = titleLink.textContent.trim();
const magnetLink = `magnet:?xt=urn:btih:${hash}&dn=${encodeURIComponent(titleText)}`;
const btnContainer = createButtonContainer({ marginRight: '8px' });
const combinedBtn = createCombinedButtons(magnetLink);
btnContainer.appendChild(combinedBtn);
titleLink.parentNode.insertBefore(btnContainer, titleLink);
return true;
});
}
initializeScript();
})();