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