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