一键复制磁力链和推送到115离线

支持BT4G/BTDig/BTSOW/Nyaa/GY/DMHY/SOBT/BTMulu等网站,可一键复制磁力链和推送到115网盘进行离线,支持打开磁力链,并支持通过脚本菜单控制各按钮的显示(推送离线任务需当前浏览器已登录(不可用)115会员账号)

目前為 2025-08-19 提交的版本,檢視 最新版本

// ==UserScript==
// @name         一键复制磁力链和推送到115离线
// @author       [email protected]
// @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://gf.qytechs.cn/users/1453515
// @license      MIT
// ==/UserScript==

(function() {
    'use strict';

    const isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);

    const CONFIG = {
        notificationTimeout: isMobile ? 5000 : 3000,
        cookieRefreshInterval: 30 * 60 * 1000,
        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: '<svg width="15" height="15" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="12" fill="#000"/><g transform="scale(0.8) translate(3,3)"><path d="M10 13C10.4295 13.5741 10.9774 14.0491 11.6066 14.3929C12.2357 14.7367 12.9315 14.9411 13.6466 14.9923C14.3618 15.0435 15.0796 14.9403 15.7513 14.6897C16.4231 14.4392 17.0331 14.047 17.54 13.54L20.54 10.54C21.4508 9.59695 21.9548 8.33394 21.9434 7.02296C21.932 5.71198 21.4061 4.45791 20.479 3.53087C19.5519 2.60383 18.2978 2.07799 16.9869 2.0666C15.6759 2.0552 14.4129 2.55916 13.47 3.46997L11.75 5.17997" stroke="#fff" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/><path d="M14 11C13.5705 10.4259 13.0226 9.9508 12.3934 9.60705C11.7642 9.26329 11.0684 9.05889 10.3533 9.00766C9.63816 8.95643 8.92037 9.05963 8.24861 9.3102C7.57685 9.56077 6.96684 9.95296 6.45996 10.46L3.45996 13.46C2.54915 14.403 2.04518 15.666 2.05659 16.977C2.068 18.288 2.59383 19.542 3.52087 20.4691C4.44791 21.3961 5.70198 21.922 7.01296 21.9334C8.32394 21.9448 9.58695 21.4408 10.53 20.53L12.24 18.82" stroke="#fff" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/></g></svg>',
        offline: '<img src="" style="width:15px;height:15px;">',
        open: '<svg width="15" height="15" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg"><circle cx="12" cy="12" r="9" stroke="#2563EB" stroke-width="2"/><path d="M3 12H21M12 3C14.501 6.738 15.922 10.592 16 12C15.922 13.408 14.501 17.262 12 21C9.499 17.262 8.078 13.408 8 12C8.078 10.592 9.499 6.738 12 3Z" stroke="#3B82F6" stroke-width="1.5"/></svg>'
    };

    function 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 = '<svg width="15" height="15" viewBox="0 0 14 14" style="display:inline-flex;align-items:center;justify-content:center;"><circle cx="7" cy="7" r="7" fill="#4caf50"/><polyline points="4,7 6,9 10,5" fill="none" stroke="#fff" stroke-width="1.5"/></svg>';

    function showButtonFeedback(btn, type = null) {
        const 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();
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址