// ==UserScript== // @name 一键复制磁力链和推送到115离线 // @author wangzijian0@vip.qq.com // @description 支持BT4G/BTDig/BTSOW/Nyaa/GY/DMHY/SOBT/BTMulu等网站,可一键复制磁力链和推送到115网盘进行离线,支持打开磁力链,并支持通过脚本菜单控制各按钮的显示(推送离线任务需当前浏览器已登录115会员账号) // @version 1.1.2.20250819 // @icon  // @include *://bt4gprx.com/* // @include *://*btdig.com/* // @include *://*btsow.*/* // @include *://nyaa.si/* // @include *://*gying.*/* // @include *://*gyg.*/* // @include *://*dmhy.*/* // @include *://sobt*.*/* // @include *://*btmulu.*/* // @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, enableCopyButton: GM_getValue('enableCopyButton', true), enableOfflineButton: GM_getValue('enableOfflineButton', true), enableOpenButton: GM_getValue('enableOpenButton', true) }; 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() { GM_registerMenuCommand("检查115登录状态", async () => { const isLoggedIn = await check115Login(true); showNotification('115状态', isLoggedIn ? '已登录' : '未登录'); if (!isLoggedIn) { setTimeout(() => { if (confirm('需要登录115网盘,是否进入115网盘登录页面?')) { window.open("https://115.com/?mode=login", "_blank"); } }, 500); } }); GM_registerMenuCommand("打开115网盘", () => window.open("https://115.com/?cid=0&offset=0&mode=wangpan", "_blank")); const toggleCopyButtonText = CONFIG.enableCopyButton ? "禁用复制按钮" : "启用复制按钮"; GM_registerMenuCommand(toggleCopyButtonText, () => { const newState = !CONFIG.enableCopyButton; CONFIG.enableCopyButton = newState; GM_setValue('enableCopyButton', newState); showNotification('设置已保存', newState ? '已启用"复制"按钮' : '已禁用"复制"按钮'); addActionButtons(); }); const toggleOfflineButtonText = CONFIG.enableOfflineButton ? "禁用离线按钮" : "启用离线按钮"; GM_registerMenuCommand(toggleOfflineButtonText, () => { const newState = !CONFIG.enableOfflineButton; CONFIG.enableOfflineButton = newState; GM_setValue('enableOfflineButton', newState); showNotification('设置已保存', newState ? '已启用"离线"按钮' : '已禁用"离线"按钮'); addActionButtons(); }); const toggleOpenButtonText = CONFIG.enableOpenButton ? "禁用打开按钮" : "启用打开按钮"; GM_registerMenuCommand(toggleOpenButtonText, () => { const newState = !CONFIG.enableOpenButton; CONFIG.enableOpenButton = newState; GM_setValue('enableOpenButton', newState); showNotification('设置已保存', newState ? '已启用"打开"按钮' : '已禁用"打开"按钮'); addActionButtons(); }); } async function checkCookieRefresh() { try { await check115Login(true); } catch (error) { console.error('检查cookie刷新失败:', error); } } function setupMutationObserver() { const observer = new MutationObserver((mutations) => { for (const mutation of mutations) { if (mutation.addedNodes.length) { addActionButtons(); } } }); observer.observe(document, { childList: true, subtree: true }); } function addActionButtons() { if (window.location.host.includes('bt4gprx.com')) { handleBT4GSite(); } else { handleCommonSites(); } } function createButton(type, element, icon = null, noDefaultClick = false) { const btn = document.createElement('button'); btn.className = `${type}-magnet-btn`; const buttonIcon = icon || ICONS[type] || ICONS.offline; applyButtonStyle(btn, type, buttonIcon); if (!noDefaultClick) { setupButtonClickHandler(btn, type, element); } return btn; } const ICONS = { copy: '', offline: '', open: '' }; function applyButtonStyle(btn, type, icon) { Object.assign(btn.style, { cursor: 'pointer', backgroundColor: 'transparent', color: '#555', border: '1px solid #ddd', borderRadius: '4px', padding: '2px 6px', fontSize: '12px', marginRight: '5px', transition: 'all 0.15s ease-in-out', fontWeight: '400', lineHeight: '1.5', verticalAlign: 'middle', touchAction: 'manipulation', width: '30px', height: '26px', minWidth: '30px', minHeight: '26px', boxSizing: 'border-box', }); const titles = { copy: '复制磁力链', offline: '推送到115离线', open: '打开磁力链' }; btn.title = titles[type] || '操作'; btn.innerHTML = icon || ICONS[type] || ICONS.offline; btn.addEventListener('mouseenter', () => { btn.style.backgroundColor = '#f0f0f0'; btn.style.borderColor = '#ccc'; }); btn.addEventListener('mouseleave', () => { btn.style.backgroundColor = 'transparent'; btn.style.borderColor = '#ddd'; }); btn.addEventListener('touchstart', () => { btn.style.backgroundColor = '#f0f0f0'; btn.style.borderColor = '#ccc'; }); btn.addEventListener('touchend', () => { btn.style.backgroundColor = 'transparent'; btn.style.borderColor = '#ddd'; }); } function createButtonContainer(options = {}) { const elementType = options.elementType || 'span'; const btnContainer = document.createElement(elementType); btnContainer.className = 'magnet-action-buttons'; const styles = { display: 'inline-block', marginRight: options.marginRight || '5px', marginLeft: options.marginLeft || '0px', verticalAlign: options.verticalAlign || 'middle', ...options.customStyles }; Object.assign(btnContainer.style, styles); return btnContainer; } 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 createButtonPart = (type, icon, isLast = false) => { const part = document.createElement('span'); part.className = `magnet-button-part ${type}-part`; part.style.padding = '0 6px'; part.style.color = '#333'; part.style.transition = 'all 0.15s ease-in-out'; part.style.display = 'inline-flex'; part.style.alignItems = 'center'; part.style.justifyContent = 'center'; part.style.minWidth = '20px'; part.style.height = '22px'; part.innerHTML = icon; part.dataset.type = type; const titles = { copy: '复制磁力链', offline: '推送到115离线', open: '打开磁力链' }; part.title = titles[type] || '操作'; return part; }; const copyPart = createButtonPart('copy', ICONS.copy); const offlinePart = createButtonPart('offline', ICONS.offline); let openPart = null; if (CONFIG.enableOpenButton) { openPart = createButtonPart('open', ICONS.open); } const buttonParts = []; if (CONFIG.enableCopyButton) { buttonParts.push(copyPart); } if (CONFIG.enableOfflineButton) { buttonParts.push(offlinePart); } if (CONFIG.enableOpenButton && openPart) { buttonParts.push(openPart); } if (buttonParts.length > 0) { 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.padding = '0 2px'; sep.style.color = '#999'; sep.innerText = '|'; combinedBtn.appendChild(sep); combinedBtn.appendChild(buttonParts[i]); } buttonParts[buttonParts.length - 1].style.borderRadius = '0 2px 2px 0'; } combinedBtn.addEventListener('mouseenter', () => { combinedBtn.style.backgroundColor = '#f5f5f5'; combinedBtn.style.borderColor = '#ccc'; }); combinedBtn.addEventListener('mouseleave', () => { combinedBtn.style.backgroundColor = 'transparent'; combinedBtn.style.borderColor = '#ddd'; }); 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; } function setupButtonClickHandler(btn, type, element) { const handleClick = async (e) => { e.preventDefault(); e.stopPropagation(); const magnetLink = typeof element === 'string' ? element : await extractMagnetLink(element); if (!magnetLink) return; if (type === 'copy') { await handleCopyAction(btn, magnetLink); } else { await handleOfflineAction(btn, magnetLink); } }; btn.addEventListener('click', handleClick); btn.addEventListener('touchend', handleClick); } async function handleCopyAction(btn, magnetLink) { try { let decodedMagnetLink = magnetLink; try { decodedMagnetLink = decodeURIComponent(magnetLink); } catch (e) { } GM_setClipboard(decodedMagnetLink, 'text'); if (isMobile && navigator.clipboard && navigator.clipboard.writeText) { try { await navigator.clipboard.writeText(decodedMagnetLink); } catch (clipboardError) { console.log('使用navigator.clipboard失败:', clipboardError); } } let displayText = decodedMagnetLink; showNotification('磁力链已复制', displayText); showButtonFeedback(btn, 'copy'); } catch (error) { showNotification('复制失败', `请手动复制: ${magnetLink}`); } } const SUCCESS_FEEDBACK_SVG = ''; function showButtonFeedback(btn, type = null) { const feedbackHTML = SUCCESS_FEEDBACK_SVG; if (btn.classList.contains('magnet-combined-button')) { let clickedPart; if (type) { clickedPart = btn.querySelector(`.magnet-button-part[data-type="${type}"]`); } else { clickedPart = btn.querySelector('.magnet-button-part'); } if (clickedPart) { const originalContent = clickedPart.innerHTML; clickedPart.style.minHeight = '22px'; clickedPart.style.display = 'inline-flex'; clickedPart.style.alignItems = 'center'; clickedPart.style.justifyContent = 'center'; clickedPart.innerHTML = feedbackHTML; btn.disabled = true; setTimeout(() => { clickedPart.innerHTML = originalContent; btn.disabled = false; }, 2000); } } else { const originalHTML = btn.innerHTML; btn.style.minHeight = '26px'; btn.innerHTML = feedbackHTML; btn.disabled = true; setTimeout(() => { btn.innerHTML = originalHTML; btn.disabled = false; }, 2000); } } async function handleOfflineAction(btn, magnetLink) { await process115Offline(magnetLink); showButtonFeedback(btn, 'offline'); } function handleBT4GSite() { document.querySelectorAll('.result-item h5 > a[href^="/magnet/"]').forEach(titleA => { if (titleA.dataset.bt4gButtonsAdded) return; titleA.dataset.bt4gButtonsAdded = 'true'; const btnContainer = createButtonContainer({ marginRight: '8px' }); const combinedBtn = createCombinedButtons(titleA); btnContainer.appendChild(combinedBtn); titleA.parentNode.insertBefore(btnContainer, titleA); }); document.querySelectorAll('.card-body').forEach(cardBody => { if (cardBody.dataset.buttonsAdded) return; const magnetBtn = cardBody.querySelector('a[href*="downloadtorrentfile.com/hash/"]'); if (!magnetBtn) return; cardBody.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ elementType: 'div', marginRight: '10px' }); const combinedBtn = createCombinedButtons(magnetBtn); btnContainer.appendChild(combinedBtn); magnetBtn.parentNode.insertBefore(btnContainer, magnetBtn); }); } async function fetchBT4GMagnetFromDetail(detailHref) { try { let url = detailHref; if (!/^https?:/.test(url)) { url = location.origin + url; } const resp = await fetch(url, { credentials: 'omit' }); const html = await resp.text(); 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 magnetLink = `magnet:?xt=urn:btih:${hash}`; if (nameMatch && nameMatch[1]) { const name = nameMatch[1]; magnetLink += `&dn=${name}`; } return magnetLink; } catch (e) { return null; } } function handleCommonSites() { if (/sobt[^.]+\..+/.test(window.location.host)) { handleSOBTSite(); } else if (window.location.host.endsWith('btdig.com')) { handleBTDigSite(); } else if (window.location.host.includes('nyaa.si')) { handleNyaaSite(); } else if (window.location.host.includes('dmhy.org')) { handleDMHYSite(); } else if (/(\.gying|\.gyg)\..+/.test(window.location.host)) { handleGyingGygSite(); } else if (/(\.|^)btsow\./.test(window.location.host)) { handleBtsowSite(); } else if (/\.btmulu\./.test(window.location.host)) { handleBTMULUSite(); } } function handleBtsowSite() { document.querySelectorAll('.row.data-row .file').forEach(titleLink => { if (titleLink.dataset.buttonsAdded) return; titleLink.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginRight: '8px' }); const magnetLink = extractBtsowMagnetLink(titleLink); if (!magnetLink) return; const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); titleLink.parentNode.insertBefore(btnContainer, titleLink); }); document.querySelectorAll('textarea.magnet-link[readonly]').forEach(textarea => { if (textarea.dataset.buttonsAdded) return; textarea.dataset.buttonsAdded = true; const magnetLink = textarea.value.trim(); if (!magnetLink || !magnetLink.startsWith('magnet:')) return; const btnContainer = document.createElement('div'); btnContainer.className = 'magnet-action-buttons'; btnContainer.style.display = 'inline-block'; btnContainer.style.marginLeft = '10px'; btnContainer.style.verticalAlign = 'middle'; const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); textarea.parentNode.insertBefore(btnContainer, textarea.nextSibling); }); } async function fetchBTMULUMagnetFromDetail(detailHref) { try { let url = detailHref; if (!/^https?:/.test(url)) { url = location.origin + url; } const resp = await fetch(url, { credentials: 'omit' }); const html = await resp.text(); const parser = new DOMParser(); const doc = parser.parseFromString(html, 'text/html'); const magnetA = doc.querySelector('div.media-body a[href^="magnet:"]'); if (!magnetA) return null; return magnetA.href; } catch (e) { console.error('获取BTMULU磁力链失败:', e); return null; } } function handleBTMULUSite() { document.querySelectorAll('div[style="overflow: hidden;"] a[href^="/hash/"] h4').forEach(titleElement => { const titleLink = titleElement.closest('a[href^="/hash/"]'); if (!titleLink || titleLink.dataset.buttonsAdded) return; titleLink.dataset.buttonsAdded = true; const labelElement = titleElement.querySelector('span.label'); if (!labelElement) return; const btnContainer = createButtonContainer({ customStyles: { margin: '0 8px' } }); (async () => { try { const magnetLink = await fetchBTMULUMagnetFromDetail(titleLink.href); if (magnetLink) { const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); if (labelElement.nextSibling) { titleElement.insertBefore(btnContainer, labelElement.nextSibling); } else { titleElement.appendChild(btnContainer); } } } catch (e) { console.error('处理BTMULU磁力链失败:', e); } })(); }); document.querySelectorAll('div.media-body a[href^="magnet:"]').forEach(magnetLink => { if (magnetLink.dataset.buttonsAdded) return; magnetLink.dataset.buttonsAdded = true; 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() { document.querySelectorAll('h3 > a[href^="/torrent/"]').forEach(titleLink => { if (titleLink.dataset.buttonsAdded) return; titleLink.dataset.buttonsAdded = true; const btnContainer = document.createElement('span'); btnContainer.className = 'magnet-action-buttons'; btnContainer.style.display = 'inline-block'; btnContainer.style.marginRight = '5px'; const combinedBtn = createCombinedButtons(titleLink); btnContainer.appendChild(combinedBtn); titleLink.parentNode.insertBefore(btnContainer, titleLink); }); document.querySelectorAll('a.download[id="down-url"]').forEach(openLinkBtn => { if (openLinkBtn.dataset.buttonsAdded) return; openLinkBtn.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginRight: '8px' }); const magnetLink = openLinkBtn.href; const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); openLinkBtn.parentNode.insertBefore(btnContainer, openLinkBtn); }); } function handleBTDigSite() { document.querySelectorAll('.torrent_name > a').forEach(titleLink => { if (titleLink.dataset.buttonsAdded) return; titleLink.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginRight: '10px' }); let resultDiv = titleLink.closest('.one_result'); let magnetLink = resultDiv ? resultDiv.querySelector('.torrent_magnet a[href^="magnet:"]') : null; if (!magnetLink) return; const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); titleLink.parentNode.insertBefore(btnContainer, titleLink); }); document.querySelectorAll('tr td div.fa.fa-magnet a[href^="magnet:"]').forEach(magnetLink => { if (magnetLink.dataset.buttonsAdded) return; magnetLink.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginLeft: '10px' }); const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); magnetLink.parentNode.appendChild(btnContainer); }); } function handleNyaaSite() { document.querySelectorAll('td.text-center a[href^="magnet:"]').forEach(magnetLink => { if (magnetLink.dataset.buttonsAdded) return; magnetLink.dataset.buttonsAdded = true; let tr = magnetLink.closest('tr'); let downloadBtn = tr ? tr.querySelector("a[href^='/download/']") : null; 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); } }); document.querySelectorAll('.panel-footer .card-footer-item[href^="magnet:"]').forEach(magnetLink => { if (magnetLink.dataset.buttonsAdded) return; magnetLink.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginLeft: '10px' }); const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); magnetLink.parentNode.insertBefore(btnContainer, magnetLink.nextSibling); }); } function handleDMHYSite() { const magnetHeader = document.querySelector('#topic_list th:nth-child(4)'); if (magnetHeader) { magnetHeader.style.width = '18%'; } document.querySelectorAll('a.download-arrow.arrow-magnet').forEach(magnetLink => { if (magnetLink.dataset.buttonsAdded) return; magnetLink.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginLeft: '5px' }); const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); magnetLink.parentNode.insertBefore(btnContainer, magnetLink); }); document.querySelectorAll('#tabs-1 a.magnet, #tabs-1 a#magnet2').forEach(magnetLink => { if (magnetLink.dataset.buttonsAdded) return; magnetLink.dataset.buttonsAdded = true; const btnContainer = createButtonContainer({ marginLeft: '5px' }); const combinedBtn = createCombinedButtons(magnetLink); btnContainer.appendChild(combinedBtn); magnetLink.parentNode.insertBefore(btnContainer, magnetLink.nextSibling); }); } 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 = document.createElement('span'); btnContainer.className = 'magnet-action-buttons'; btnContainer.style.display = 'inline-block'; btnContainer.style.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 || !magnetLink.startsWith('magnet:')) return; const newLi = document.createElement('li'); newLi.className = 'magnet-script-custom-li'; Object.assign(newLi.style, { display: 'inline-flex', alignItems: 'center', marginRight: '8px', verticalAlign: 'middle', padding: '0', backgroundColor: 'transparent', border: 'none', boxShadow: 'none', listStyle: 'none', fontSize: '12px', lineHeight: '1.5', fontFamily: 'Arial, sans-serif', color: '#333', textDecoration: 'none', userSelect: 'none', float: 'none', clear: 'none', width: 'auto', height: 'auto', minWidth: '0', minHeight: '0', maxWidth: 'none', maxHeight: 'none', position: 'relative', overflow: 'visible', zIndex: '100', whiteSpace: 'nowrap', transform: 'translateY(-8.5px)' }); const combinedBtn = createCombinedButtons(magnetLink); combinedBtn.className = 'magnet-combined-button'; newLi.appendChild(combinedBtn); list.insertBefore(newLi, magnetItem); }); } async function extractMagnetLink(element) { try { if (typeof element === 'string') { if (element.startsWith('magnet:')) { return element; } throw new Error('无法提取磁力链Hash'); } if (element.href) { if (element.href.startsWith('magnet:')) { return element.href; } if (element.href.includes('/magnet/')) { return await fetchBT4GMagnetFromDetail(element.href); } if (element.href.includes('/torrent/')) { const hashMatch = element.href.match(/\/torrent\/([a-f0-9]+)\.html$/i); if (hashMatch && hashMatch[1]) { return `magnet:?xt=urn:btih:${hashMatch[1]}`; } } if (element.href.includes('downloadtorrentfile.com/hash/')) { const hashMatch = element.href.match(/hash\/([a-f0-9]+)/i); if (hashMatch && hashMatch[1]) { const hash = hashMatch[1]; const nameMatch = element.href.match(/[?&]name=([^&]+)/i); let magnetLink = `magnet:?xt=urn:btih:${hash}`; if (nameMatch && nameMatch[1]) { magnetLink += `&dn=${nameMatch[1]}`; } return magnetLink; } } if (element.href.includes('/hash/')) { const hashMatch = element.href.match(/\/hash\/([a-f0-9]+)\.html$/i); if (hashMatch && hashMatch[1]) { return `magnet:?xt=urn:btih:${hashMatch[1]}`; } } } throw new Error('无法提取磁力链Hash'); } 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(); if (!cookies) { GM_setValue('115_cookies', ''); GM_setValue('115_last_cookie_refresh', 0); return false; } const isValid = await validate115Cookies(cookies); if (isValid) { GM_setValue('115_cookies', cookies); GM_setValue('115_last_cookie_refresh', Date.now()); return true; } GM_setValue('115_cookies', ''); GM_setValue('115_last_cookie_refresh', 0); return false; } 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) { const notificationContainer = document.createElement('div'); notificationContainer.className = 'custom-notification'; notificationContainer.id = id ? `notification-${id}` : `notification-${Date.now()}`; Object.assign(notificationContainer.style, { position: 'fixed', bottom: '20px', right: '20px', padding: '12px 16px', backgroundColor: 'rgba(255, 255, 255, 0.8)', 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)', WebkitBackdropFilter: 'blur(10px)', border: '1px solid rgba(255, 255, 255, 0.2)' }); const titleElement = document.createElement('div'); titleElement.textContent = title; titleElement.style.fontWeight = 'bold'; titleElement.style.marginBottom = '4px'; const textElement = document.createElement('div'); try { if (text && (text.includes('%') || text.includes('magnet:'))) { textElement.textContent = decodeURIComponent(text); } else { textElement.textContent = text; } } catch (e) { textElement.textContent = text; } textElement.style.fontSize = '14px'; notificationContainer.appendChild(titleElement); notificationContainer.appendChild(textElement); document.body.appendChild(notificationContainer); setTimeout(() => { notificationContainer.style.opacity = '1'; notificationContainer.style.transform = 'translateY(0)'; }, 10); const timeoutId = setTimeout(() => { notificationContainer.style.opacity = '0'; notificationContainer.style.transform = 'translateY(20px)'; setTimeout(() => { if (document.body.contains(notificationContainer)) { document.body.removeChild(notificationContainer); } }, 300); }, CONFIG.notificationTimeout); notificationContainer.addEventListener('click', () => { clearTimeout(timeoutId); notificationContainer.style.opacity = '0'; notificationContainer.style.transform = 'translateY(20px)'; setTimeout(() => { if (document.body.contains(notificationContainer)) { document.body.removeChild(notificationContainer); } }, 300); }); } initializeScript(); })();