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