您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
支持BT4G/BTDig/BTSOW/Nyaa/GY/DMHY/SOBT/BTMulu等网站,可一键复制磁力链和推送到115网盘进行离线,支持打开磁力链,并支持通过脚本菜单控制各按钮的显示(推送离线任务需当前浏览器已登录(不可用)115会员账号)
当前为
// ==UserScript== // @name 一键复制磁力链和推送到115离线 // @author [email protected] // @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://gf.qytechs.cn/users/1453515 // @license MIT // ==/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: '<svg width="15" height="15" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="12" fill="#000"/><g transform="scale(0.8) translate(3,3)"><path d="M10 13C10.4295 13.5741 10.9774 14.0491 11.6066 14.3929C12.2357 14.7367 12.9315 14.9411 13.6466 14.9923C14.3618 15.0435 15.0796 14.9403 15.7513 14.6897C16.4231 14.4392 17.0331 14.047 17.54 13.54L20.54 10.54C21.4508 9.59695 21.9548 8.33394 21.9434 7.02296C21.932 5.71198 21.4061 4.45791 20.479 3.53087C19.5519 2.60383 18.2978 2.07799 16.9869 2.0666C15.6759 2.0552 14.4129 2.55916 13.47 3.46997L11.75 5.17997" stroke="#fff" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><path d="M14 11C13.5705 10.4259 13.0226 9.9508 12.3934 9.60705C11.7642 9.26329 11.0684 9.05889 10.3533 9.00766C9.63816 8.95643 8.92037 9.05963 8.24861 9.3102C7.57685 9.56077 6.96684 9.95296 6.45996 10.46L3.45996 13.46C2.54915 14.403 2.04518 15.666 2.05659 16.977C2.068 18.288 2.59383 19.542 3.52087 20.4691C4.44791 21.3961 5.70198 21.922 7.01296 21.9334C8.32394 21.9448 9.58695 21.4408 10.53 20.53L12.24 18.82" stroke="#fff" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></g></svg>', offline: '<img src="" style="width:15px;height:15px;">', open: '<svg width="15" height="15" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="9" stroke="#2563EB" stroke-width="2"/><path d="M3 12H21M12 3C14.501 6.738 15.922 10.592 16 12C15.922 13.408 14.501 17.262 12 21C9.499 17.262 8.078 13.408 8 12C8.078 10.592 9.499 6.738 12 3Z" stroke="#3B82F6" stroke-width="1.5"/></svg>' }; 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 = '<svg width="15" height="15" viewBox="0 0 14 14" style="display:inline-flex;align-items:center;justify-content:center;"><circle cx="7" cy="7" r="7" fill="#4caf50"/><polyline points="4,7 6,9 10,5" fill="none" stroke="#fff" stroke-width="1.5"/></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(); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址