// ==UserScript== // @name 一键复制磁力链和推送到115离线 // @author wangzijian0@vip.qq.com // @description 支持BT4G/BTDig/BTSOW/Nyaa/GY/DMHY/SOBT/BTMulu等网站,可一键复制磁力链和推送到115网盘进行离线,支持打开磁力链,并支持通过脚本菜单控制各按钮的显示(推送离线任务需当前浏览器已登录115会员账号) // @version 1.1.3.20250819 // @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.*/* // @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 none // ==/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) }; 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'; } }); } 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(); setInterval(checkCookieRefresh, 5 * 60 * 1000); setupMutationObserver(); addActionButtons(); } function addMenuCommands() { const menuCommands = [ { 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") } ]; const buttonConfigs = [ { key: 'enableCopyButton', name: '复制' }, { key: 'enableOfflineButton', name: '离线' }, { key: 'enableOpenButton', name: '打开' } ]; buttonConfigs.forEach(({ key, name }) => { const isEnabled = CONFIG[key]; const toggleText = isEnabled ? `禁用${name}按钮` : `启用${name}按钮`; menuCommands.push({ name: toggleText, handler: () => { const newState = !CONFIG[key]; CONFIG[key] = newState; GM_setValue(key, newState); showNotification('设置已保存', newState ? `已启用"${name}"按钮` : `已禁用"${name}"按钮`); addActionButtons(); } }); }); 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(); }, 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 = [ { pattern: /sobt[^.]+\..+|clb[^.]+\..+/, handler: handleSOBTSite }, { pattern: /(\.|^)btsow\./, handler: handleBtsowSite }, { pattern: /\.btmulu\./, handler: handleBTMULUSite }, { pattern: /cili|mag|wuji/, handler: handleCiliMagSite }, { pattern: /(\.gying|\.gyg)\..+/, handler: handleGyingGygSite }, { pattern: /yuhuage\..+/, handler: handleYuhuageSite }, { pattern: /longwangbt\..+/, handler: handleLongwangbtSite } ]; for (const [domain, handler] of Object.entries(siteHandlers)) { if (hostname.includes(domain)) { handler(); return; } } for (const { pattern, handler } of patternHandlers) { if (pattern.test(hostname)) { 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 #ddd'; 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 buttonParts = [ CONFIG.enableCopyButton && createButtonPart('copy', ICONS.copy), CONFIG.enableOfflineButton && createButtonPart('offline', ICONS.offline), CONFIG.enableOpenButton && createButtonPart('open', ICONS.open) ].filter(Boolean); if (buttonParts.length > 0) { buttonParts[0].style.borderRadius = '2px 0 0 2px'; combinedBtn.appendChild(buttonParts[0]); buttonParts.slice(1).forEach((part, index) => { const sep = document.createElement('span'); sep.style.cssText = 'padding: 0 2px; color: #999;'; sep.innerText = '|'; combinedBtn.append(sep, part); }); if (buttonParts.length > 1) { buttonParts[buttonParts.length - 1].style.borderRadius = '0 2px 2px 0'; } } ['mouseenter', 'mouseleave'].forEach((event, i) => { combinedBtn.addEventListener(event, () => { combinedBtn.style.backgroundColor = i ? 'transparent' : '#f5f5f5'; combinedBtn.style.borderColor = i ? '#ddd' : '#ccc'; }); }); combinedBtn.addEventListener('click', async (e) => { e.preventDefault(); e.stopPropagation(); const clickedPart = e.target.closest('.magnet-button-part'); if (!clickedPart) return; const type = clickedPart.dataset.type; const magnetLink = typeof magnetLinkOrElement === 'string' ? magnetLinkOrElement : await extractMagnetLink(magnetLinkOrElement); if (!magnetLink) return; if (type === 'copy') { await handleCopyAction(combinedBtn, magnetLink); } else if (type === 'offline') { await handleOfflineAction(combinedBtn, magnetLink); } else if (type === 'open') { window.open(magnetLink, '_blank'); showNotification('已打开磁力链', '磁力链已在新标签页打开'); showButtonFeedback(combinedBtn, 'open'); } }); return combinedBtn; } 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() { processMagnetLinks({ selectors: '.result-item h5 > a[href^="/magnet/"]', containerStyles: { marginRight: '8px' }, customProcessor: (titleA) => { const btnContainer = createButtonContainer({ marginRight: '8px' }); const loadingBtn = createLoadingButton(); btnContainer.appendChild(loadingBtn); titleA.parentNode.insertBefore(btnContainer, titleA); processBT4GMagnetLink(titleA, btnContainer).then(success => { if (!success) { setupRetryButton(loadingBtn, () => processBT4GMagnetLink(titleA, btnContainer, 2, 6000) ); } }).catch(error => { console.error('BT4G处理失败:', error); setButtonError(loadingBtn, '处理失败'); }); } }); processMagnetLinks({ selectors: '.card-body', customProcessor: (cardBody) => { const magnetBtn = cardBody.querySelector('a[href*="downloadtorrentfile.com/hash/"]'); if (!magnetBtn) return; const btnContainer = createButtonContainer({ elementType: 'div', marginRight: '10px' }); const combinedBtn = createCombinedButtons(magnetBtn); btnContainer.appendChild(combinedBtn); magnetBtn.parentNode.insertBefore(btnContainer, magnetBtn); } }); } 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); const magnetUrl = new URL(`magnet:?xt=urn:btih:${hash}`); if (nameMatch?.[1]) { magnetUrl.searchParams.set('dn', nameMatch[1]); } return magnetUrl.toString(); } 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); } } }); processMagnetLinks({ selectors: 'div.media-body a[href^="magnet:"]', containerStyles: { elementType: 'div', customStyles: { display: 'block', marginTop: '10px' } }, customProcessor: (magnetLink) => { const btnContainer = createButtonContainer({ elementType: 'div', customStyles: { display: 'block', marginTop: '10px' } }); const combinedBtn = createCombinedButtons(magnetLink.href); btnContainer.appendChild(combinedBtn); magnetLink.parentNode.insertBefore(btnContainer, magnetLink.nextSibling); } }); } 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('tr td div.fa.fa-magnet a[href^="magnet:"]', (magnetLink) => { const btnContainer = createButtonContainer({ marginLeft: '10px' }); const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); magnetLink.parentNode.appendChild(btnContainer); 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() { document.querySelectorAll('li.down-list2').forEach(item => { const magnetLink = item.querySelector('a.torrent[href^="magnet:"]'); const detailLink = item.querySelector('a[href^="/bt/"]'); if (!magnetLink || !detailLink || detailLink.dataset.buttonsAdded) return; detailLink.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginRight: '8px' }); const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); detailLink.parentNode.insertBefore(btnContainer, detailLink); }); document.querySelectorAll('div.alert-info ul.down123').forEach(list => { const magnetItem = list.querySelector('li[data-clipboard-text^="magnet:"]'); if (!magnetItem || magnetItem.dataset.buttonsAdded) return; magnetItem.dataset.buttonsAdded = true; const magnetLink = magnetItem.getAttribute('data-clipboard-text'); if (!magnetLink?.startsWith('magnet:')) return; const newLi = document.createElement('li'); newLi.className = 'magnet-script-custom-li'; newLi.style.cssText = ` display: inline-flex; align-items: center; margin-right: 8px; vertical-align: middle; padding: 0; background: transparent; border: none; list-style: none; font-size: 12px; transform: translateY(-8.5px); z-index: 100 `; const combinedBtn = createCombinedButtons(magnetLink); newLi.appendChild(combinedBtn); list.insertBefore(newLi, magnetItem); }); } 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 lastRefresh = GM_getValue('115_last_cookie_refresh', 0); const currentCookies = GM_getValue('115_cookies', ''); if (!forceCheck && currentCookies && Date.now() - lastRefresh < CONFIG.cookieRefreshInterval) { return true; } const cookies = await getCurrent115Cookies(); const isValid = cookies && await validate115Cookies(cookies); GM_setValue('115_cookies', isValid ? cookies : ''); GM_setValue('115_last_cookie_refresh', isValid ? Date.now() : 0); return isValid; } catch (error) { console.error('检查登录状态失败:', error); return false; } } function getCurrent115Cookies() { return new Promise((resolve) => { GM_xmlhttpRequest({ url: 'https://115.com/', method: 'GET', anonymous: true, onload: function(response) { const cookieHeader = response.responseHeaders .split('\n') .find(row => row.toLowerCase().startsWith('set-cookie:')); if (cookieHeader) { const cookies = cookieHeader.replace(/^set-cookie:\s*/i, '').split(';')[0]; resolve(cookies); } else { if (response.finalUrl.includes('login.115.com')) { resolve(''); } else { const savedCookies = GM_getValue('115_cookies', ''); resolve(savedCookies); } } }, onerror: () => resolve('') }); }); } function validate115Cookies(cookies) { return new Promise((resolve) => { GM_xmlhttpRequest({ url: 'https://115.com/web/lixian/', method: 'GET', headers: { 'Cookie': cookies }, onload: function(response) { resolve(!response.finalUrl.includes('login.115.com')); }, 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); } } } async function submit115OfflineTask(magnetLink) { const cookies = GM_getValue('115_cookies', ''); if (!cookies) { throw new Error('未检测到有效的登录状态'); } const response = await fetch115Api( `https://115.com/web/lixian/?ct=lixian&ac=add_task_url&url=${encodeURIComponent(magnetLink)}`, { headers: { 'Cookie': cookies } } ); return tryParseJson(response); } function handleOfflineResult(result) { if (!result) { throw new Error('无效的响应'); } if (result.state) { showNotification('115离线成功', '任务已成功添加到离线下载列表'); return; } const errorMsg = ERROR_CODES[result.errcode] || result.error_msg || '未知错误'; throw new Error(errorMsg); } function fetch115Api(url, options = {}) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ url: url, method: options.method || 'GET', headers: { 'User-Agent': navigator.userAgent, 'Origin': 'https://115.com', ...(options.headers || {}) }, data: options.body, onload: function(response) { if (response.status >= 200 && response.status < 300) { resolve(response.responseText); } else { reject(new Error(`请求失败: ${response.status}`)); } }, onerror: reject }); }); } function tryParseJson(text) { try { return JSON.parse(text); } catch (e) { return null; } } 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: '9999', 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 handleSeedhubSite() { processElements('.seeds a', (linkElement) => { const btnContainer = createButtonContainer({ marginRight: '8px', customStyles: { display: 'inline-block', verticalAlign: 'middle' } }); const combinedBtn = createCombinedButtons(linkElement); btnContainer.appendChild(combinedBtn); linkElement.parentNode.insertBefore(btnContainer, linkElement); return true; }); } function handleYuhuageSite() { processElements('.search-item .item-title h3 > a[href^="/hash/"]', (titleLink) => { const btnContainer = createButtonContainer({ marginRight: '8px' }); const loadingBtn = createLoadingButton(); btnContainer.appendChild(loadingBtn); titleLink.parentNode.insertBefore(btnContainer, titleLink); processYuhuageMagnetLink(titleLink, btnContainer).then(success => { if (!success) { setupRetryButton(loadingBtn, () => processYuhuageMagnetLink(titleLink, btnContainer, 2, 6000) ); } }).catch(error => { console.error('Yuhuage处理失败:', error); setButtonError(loadingBtn, '处理失败'); }); 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 #ddd;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' }); const loadingBtn = createLoadingButton(); btnContainer.appendChild(loadingBtn); linkElement.parentNode.insertBefore(btnContainer, linkElement); processMagnetLink(linkElement, btnContainer).then(success => { if (!success) { setupRetryButton(loadingBtn, () => processMagnetLink(linkElement, btnContainer, 2, 6000) ); } }).catch(error => { console.error('CiliMag处理失败:', error); setButtonError(loadingBtn, '处理失败'); }); 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(); })();