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

支持BT4G/BTDigg/BTSOW/Nyaa/DMHY(動漫花園)/GY(观影)/SeedHub/LongWangBT(梦幻天堂·龙网)/YuHuaGe(雨花阁)/SOBT/CLB(磁力宝)/BTMulu(BT目录)/ØMagnet(无极磁链)/磁力帝等磁力搜索网站,可一键复制磁力链和推送到115网盘进行离线下载,支持打开磁力链,支持批量操作;提供“菜单”和左下角悬浮“设置”入口(需已登录(不可用)115会员账号才能推送离线任务成功)

// ==UserScript==
// @name         一键复制磁力链和推送到115离线
// @author       [email protected]
// @description  支持BT4G/BTDigg/BTSOW/Nyaa/DMHY(動漫花園)/GY(观影)/SeedHub/LongWangBT(梦幻天堂·龙网)/YuHuaGe(雨花阁)/SOBT/CLB(磁力宝)/BTMulu(BT目录)/ØMagnet(无极磁链)/磁力帝等磁力搜索网站,可一键复制磁力链和推送到115网盘进行离线下载,支持打开磁力链,支持批量操作;提供“菜单”和左下角悬浮“设置”入口(需已登录(不可用)115会员账号才能推送离线任务成功)
// @version      1.1.4.20250822
// @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.*/*
// @include      *://*1122*.*/*
// @include      *://*cld130.*/*
// @grant        GM_setClipboard
// @grant        GM_xmlhttpRequest
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// @connect      115.com
// @connect      login.115.com
// @connect      *
// @run-at       document-end
// @namespace    https://gf.qytechs.cn/users/1453515
// @license      MIT
// ==/UserScript==

(function() {
    'use strict';

    const isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);
    const CONFIG = {
        notificationTimeout: isMobile ? 5000 : 3000,
        cookieRefreshInterval: 30 * 60 * 1000,
        retryDelay: 2000,
        maxRetries: 3,
        defaultTimeout: 8000,
        enableCopyButton: GM_getValue('enableCopyButton', true),
        enableOfflineButton: GM_getValue('enableOfflineButton', true),
        enableOpenButton: GM_getValue('enableOpenButton', true),
        autoFetchEnabled: GM_getValue('autoFetchEnabled', true),
        autoFetchSites: {
            bt4g: GM_getValue('autoFetch_bt4g', false),
            seedhub: GM_getValue('autoFetch_seedhub', false),
            yuhuage: GM_getValue('autoFetch_yuhuage', true),
            cilimag: GM_getValue('autoFetch_cilimag', true)
        },
        siteEnabled: {
            bt4g: GM_getValue('site_bt4g', true),
            btdig: GM_getValue('site_btdig', true),
            btsow: GM_getValue('site_btsow', true),
            nyaa: GM_getValue('site_nyaa', true),
            dmhy: GM_getValue('site_dmhy', true),
            seedhub: GM_getValue('site_seedhub', true),
            longwangbt: GM_getValue('site_longwangbt', true),
            yuhuage: GM_getValue('site_yuhuage', true),
            sobt: GM_getValue('site_sobt', true),
            clb: GM_getValue('site_clb', true),
            btmulu: GM_getValue('site_btmulu', true),
            cili_family: GM_getValue('site_cili_family', true),
            gying_family: GM_getValue('site_gying_family', true),
            cilidi: GM_getValue('site_cilidi', true)
        },
        enableFloatingSettingsBtn: GM_getValue('enableFloatingSettingsBtn', true)
    };


  const DEFAULT_BUTTONS_ORDER = ['copy', 'offline', 'open'];
  function getButtonsOrder() {
      try {
          const saved = GM_getValue('buttonsOrder');
          if (!saved) return DEFAULT_BUTTONS_ORDER.slice();
          if (Array.isArray(saved)) {
              const allow = new Set(DEFAULT_BUTTONS_ORDER);
              const arr = saved.filter(x => allow.has(x));
              return arr.length ? arr : DEFAULT_BUTTONS_ORDER.slice();
          }
          if (typeof saved === 'string') {
              const parts = saved.split(/[|,\s]+/).filter(Boolean);
              const allow = new Set(DEFAULT_BUTTONS_ORDER);
              const arr = parts.filter(x => allow.has(x));
              return arr.length ? arr : DEFAULT_BUTTONS_ORDER.slice();
          }
      } catch (_) {}
      return DEFAULT_BUTTONS_ORDER.slice();
  }
  function setButtonsOrder(order) {
      try { GM_setValue('buttonsOrder', Array.isArray(order) ? order : DEFAULT_BUTTONS_ORDER); } catch (_) {}
  }

    const SITES_LINKS = {
        bt4g: {
            sites: [
                { url: 'https://bt4gprx.com' }
            ]
        },
        btdig: {
            sites: [
                { url: 'https://btdig.com' }
            ]
        },
        btsow: {
            sites: [
                { url: 'https://btsow.com' }
            ]
        },
        nyaa: {
            sites: [
                { url: 'https://nyaa.si' }
            ]
        },
        dmhy: {
            sites: [
                { url: 'https://dmhy.org' }
            ]
        },
        gying_family: {
            sites: [
                { url: 'www.gying.net' },
                { url: 'www.gying.org' },
                { url: 'www.gying.si' },
                { url: 'www.gying.in' },
                { url: 'www.gyg.la' },
                { url: 'www.gyg.si' }
            ]
        },
        seedhub: {
            sites: [
                { url: 'https://www.seedhub.cc' },
                { url: 'https://www.seedhub.top' },
                { url: 'https://www.seedhub.icu' },
                { url: 'https://seedhub.pro', note: '移动可能不通' }
            ],
            publish: [
                { url: 'https://workflowy.com/s/ff4ac3a19545/tEvTraNzl9fk1fJA' }
            ]
        },
        longwangbt: {
            sites: [
                { url: 'http://www.longwangbt.com' }
            ]
        },
        yuhuage: {
            sites: [
                { url: 'https://www.yuhuage.cc' }
            ]
        },
        sobt: {
            sites: [
                { url: 'https://sobt.me' }
            ]
        },
        clb: {
            sites: [
                { url: 'https://clb.im' },
                { url: 'https://cilibao.app' },
                { url: 'https://cilibao.top' }
            ]
        },
        btmulu: {
            publish: [
                { url: 'https://cursor.vip/btmulu' }
            ]
        },
        cili_family: {
            publish: [
                { url: 'https://CiLi.st' },
                { url: 'https://cili404.com' }
            ]
        },
        cilidi: {
            sites: [
            ],
            publish: [
                { url: 'https://cilidi.cyou' },
                { url: 'https://cldcld.cyou' },
                { url: 'https://cldcld.top' },
                { url: 'https://cldcld.com' }
            ]
        }
    };
    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';
            }
        });
    }

    function handleCiLiDiSite() {
        document.querySelectorAll('.ssbox .sbar').forEach(sbar => {
            if (sbar.dataset.buttonsAdded) return;
            const magnetA = sbar.querySelector('a[href^="magnet:"]');
            if (!magnetA) return;

            const btnContainer = createButtonContainer({ marginRight: '6px' });
            const combinedBtn = createCombinedButtons(magnetA.href);
            btnContainer.appendChild(combinedBtn);

            const firstSpan = sbar.querySelector('span');
            if (firstSpan) {
                sbar.insertBefore(btnContainer, firstSpan);
            } else {
                sbar.insertBefore(btnContainer, sbar.firstChild);
            }
            sbar.dataset.buttonsAdded = true;
        });

        (() => {
            const ssboxes = Array.from(document.querySelectorAll('.tbox .ssbox, .ssbox'));
            if (!ssboxes.length) return;

            const target = ssboxes.find(box => box.querySelector('.content a[href^="magnet:"]'));
            if (!target) return;

            const h3 = target.querySelector('.title h3');
            if (!h3 || h3.dataset.buttonsAdded) return;

            const magnetA = target.querySelector('.content a[href^="magnet:"]');
            if (!magnetA) return;

            const btnContainer = createButtonContainer({ marginLeft: '8px' });
            const combinedBtn = createCombinedButtons(magnetA.href);
            btnContainer.appendChild(combinedBtn);
            h3.appendChild(btnContainer);
            h3.dataset.buttonsAdded = true;
        })();
    }
    
    function isAutoFetchEnabledFor(siteKey) {
        try {
            return !!(CONFIG.autoFetchEnabled && CONFIG.autoFetchSites && CONFIG.autoFetchSites[siteKey]);
        } catch (_) {
            return false;
        }
    }
    
    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();
        setupMutationObserver();
        addActionButtons();
        ensureModalStyles();
        ensureFloatingSettingsButton();
    }

    function addMenuCommands() {
        const menuCommands = [
            {
                name: "打开设置面板",
                handler: () => openSettingsPanel()
            },
            {
                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")
            }
        ];

        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();
                ensureFloatingSettingsButton();
            }, 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 = [
            { key: 'sobt', pattern: /sobt[^.]+\..+/, handler: handleSOBTSite },
            { key: 'clb', pattern: /clb[^.]+\..+/, handler: handleSOBTSite },
            { key: 'btsow', pattern: /(\.|^)btsow\./, handler: handleBtsowSite },
            { key: 'btmulu', pattern: /\.btmulu\./, handler: handleBTMULUSite },
            { key: 'cili_family', pattern: /cili|mag|wuji/, handler: handleCiliMagSite },
            { key: 'gying_family', pattern: /(\.gying|\.gyg)\..+/, handler: handleGyingGygSite },
            { key: 'yuhuage', pattern: /yuhuage\..+/, handler: handleYuhuageSite },
            { key: 'longwangbt', pattern: /longwangbt\..+/, handler: handleLongwangbtSite },
            { key: 'cilidi', pattern: /1122|cld130/, handler: handleCiLiDiSite }
        ];
        
        const domainKeyMap = {
            'bt4gprx.com': 'bt4g',
            'btdig.com': 'btdig',
            'nyaa.si': 'nyaa',
            'dmhy.org': 'dmhy',
            'seedhub': 'seedhub'
        };
        for (const [domain, handler] of Object.entries(siteHandlers)) {
            if (hostname.includes(domain)) {
                const key = domainKeyMap[domain];
                if (!key || CONFIG.siteEnabled[key]) {
                    handler();
                }
                return;
            }
        }
        
        for (const { key, pattern, handler } of patternHandlers) {
            if (pattern.test(hostname)) {
                if (!key || CONFIG.siteEnabled[key]) {
                    handler();
                }
                return;
            }
        }
    }


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


    function createButtonContainer(options = {}) {
        const btnContainer = document.createElement(options.elementType || 'span');
        btnContainer.className = 'magnet-action-buttons';
        
        btnContainer.style.cssText = `
            display: inline-block;
            margin-right: ${options.marginRight || '5px'};
            margin-left: ${options.marginLeft || '0'};
            vertical-align: ${options.verticalAlign || 'middle'}
        `;
        
        if (options.customStyles) {
            Object.assign(btnContainer.style, options.customStyles);
        }
        
        return btnContainer;
    }
    
    async function fetchWithRetry(url, options = {}, maxRetries = CONFIG.maxRetries) {
        const normalizedUrl = /^https?:/.test(url) ? url : new URL(url, location.origin).href;
        
        for (let attempt = 0; attempt <= maxRetries; attempt++) {
            try {
                const response = await fetch(normalizedUrl, {
                    credentials: 'omit',
                    ...options
                });
                
                if (!response.ok) {
                    throw new Error(`HTTP ${response.status}: ${response.statusText}`);
                }
                
                return await response.text();
            } catch (error) {
                console.error(`Fetch attempt ${attempt + 1}/${maxRetries + 1} failed for ${normalizedUrl}:`, error);
                
                if (attempt === maxRetries) {
                    throw error;
                }
                
                await new Promise(resolve => setTimeout(resolve, CONFIG.retryDelay * (attempt + 1)));
            }
        }
    }

    function createCombinedButtons(magnetLinkOrElement) {
        const combinedBtn = document.createElement('button');
        combinedBtn.className = 'magnet-combined-button';
        combinedBtn.style.display = 'inline-flex';
        combinedBtn.style.alignItems = 'center';
        combinedBtn.style.justifyContent = 'center';
        combinedBtn.style.backgroundColor = 'transparent';
        combinedBtn.style.border = '1px solid rgba(0,0,0,0.14)';
        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 order = getButtonsOrder();
        const buttonParts = [];
        for (const type of order) {
            if (type === 'copy' && CONFIG.enableCopyButton) buttonParts.push(createButtonPart('copy', ICONS.copy));
            if (type === 'offline' && CONFIG.enableOfflineButton) buttonParts.push(createButtonPart('offline', ICONS.offline));
            if (type === 'open' && CONFIG.enableOpenButton) buttonParts.push(createButtonPart('open', ICONS.open));
        }

        if (buttonParts.length > 0) {
            if (buttonParts.length === 1) {
                buttonParts[0].style.borderRadius = '2px';
            } else {
                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.cssText = 'padding: 0 2px; color: #999;';
                sep.innerText = '|';
                combinedBtn.appendChild(sep);

                if (i === buttonParts.length - 1) {
                    buttonParts[i].style.borderRadius = '0 2px 2px 0';
                } else {
                    buttonParts[i].style.borderRadius = '0';
                }
                combinedBtn.appendChild(buttonParts[i]);
            }
        }

        const resolveMagnet = async () => {
            try {
                if (typeof magnetLinkOrElement === 'function') {
                    const res = await magnetLinkOrElement();
                    if (typeof res === 'string' && res.startsWith('magnet:')) return res;
                }
                if (typeof magnetLinkOrElement === 'string' && magnetLinkOrElement.startsWith('magnet:')) {
                    return magnetLinkOrElement;
                }
                const el = magnetLinkOrElement;
                if (el && el.nodeType === 1) {
                    const a1 = el.closest('a[href^="magnet:"]') || el.querySelector?.('a[href^="magnet:"]');
                    if (a1?.href?.startsWith('magnet:')) return a1.href;

                    {
                        const linkEl = el.closest('a') || (el.tagName === 'A' ? el : null);
                        const href = linkEl?.href || el.getAttribute?.('href') || '';
                        const m = href.match(/\/(?:torrent|detail)\/([a-f0-9]+)\.html(?:$|\?)/i);
                        if (m && m[1]) {
                            const name = (linkEl?.textContent || el.textContent || '').trim();
                            let magnet = `magnet:?xt=urn:btih:${m[1]}`;
                            if (name) magnet += `&dn=${encodeURIComponent(name)}`;
                            return magnet;
                        }
                    }
                    const a2 = el.closest('a[href*="downloadtorrentfile.com/hash/"]') || el.querySelector?.('a[href*="downloadtorrentfile.com/hash/"]');
                    if (a2?.href) {
                        const m = a2.href.match(/hash\/([a-f0-9]{40})/i);
                        if (m) {
                            const nameMatch = a2.href.match(/[?&]name=([^&]+)/i);
                            let magnet = `magnet:?xt=urn:btih:${m[1]}`;
                            if (nameMatch?.[1]) magnet += `&dn=${nameMatch[1]}`;
                            return magnet;
                        }
                    }
                    if (/bt4g/.test(location.hostname) && el.closest('a')?.href) {
                        const href = el.closest('a').href;
                        if (typeof fetchBT4GMagnetFromDetail === 'function') {
                            const mg = await fetchBT4GMagnetFromDetail(href);
                            if (mg) return mg;
                        }
                    }
                }
            } catch (e) {
                console.warn('解析磁力失败', e);
            }
            return null;
        };
        combinedBtn.addEventListener('click', async (ev) => {
            const part = ev.target.closest?.('.magnet-button-part');
            if (!part) return;
            ev.preventDefault();
            ev.stopPropagation();
            const type = part.dataset.type;
            const magnetLink = await resolveMagnet();
            if (!magnetLink) {
                showNotification('未获取到磁力链', '请稍后重试或手动打开详情页');
                return;
            }
            if (type === 'copy') {
                await handleCopyAction(combinedBtn, magnetLink);
            } else if (type === 'offline') {
                await handleOfflineAction(combinedBtn, magnetLink);
            } else if (type === 'open') {
                window.open(magnetLink, '_blank');
                showButtonFeedback(combinedBtn, 'open');
            }
        });

        return combinedBtn;
    }


    function applyButtonsOrderToExisting() {
        const order = getButtonsOrder();
        document.querySelectorAll('.magnet-combined-button').forEach(btn => {
            const partsMap = new Map();
            btn.querySelectorAll('.magnet-button-part').forEach(p => {
                const t = p.dataset.type;
                if (t) partsMap.set(t, p);
            });
            const existing = order.map(t => partsMap.get(t)).filter(Boolean);
            if (!existing.length) return;
            while (btn.firstChild) btn.removeChild(btn.firstChild);
            existing.forEach((part, idx) => {
                part.style.borderRadius = idx === 0 && existing.length === 1 ? '2px' : (idx === 0 ? '2px 0 0 2px' : (idx === existing.length - 1 ? '0 2px 2px 0' : '0'));
                if (idx > 0) {
                    const sep = document.createElement('span');
                    sep.style.cssText = 'padding: 0 2px; color: #999;';
                    sep.innerText = '|';
                    btn.appendChild(sep);
                }
                btn.appendChild(part);
            });
        });
    }


    async function handleCopyAction(btn, magnetLink) {
        try {
            let decodedMagnetLink = magnetLink;
            try {
                decodedMagnetLink = decodeURIComponent(magnetLink);
            } catch (e) {}

            GM_setClipboard(decodedMagnetLink, 'text');

            if (isMobile && navigator.clipboard?.writeText) {
                try {
                    await navigator.clipboard.writeText(decodedMagnetLink);
                } catch (clipboardError) {
                    console.log('使用navigator.clipboard失败:', clipboardError);
                }
            }

            showNotification('磁力链已复制', decodedMagnetLink);
            showButtonFeedback(btn, 'copy');
        } catch (error) {
            showNotification('复制失败', `请手动复制: ${magnetLink}`);
        }
    }

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

    function showButtonFeedback(btn, type = null) {
        const clickedPart = btn.classList.contains('magnet-combined-button') 
            ? btn.querySelector(type ? `.magnet-button-part[data-type="${type}"]` : '.magnet-button-part')
            : btn;
            
        if (!clickedPart) return;
        
        const originalContent = clickedPart.innerHTML;
        clickedPart.style.cssText += 'min-height:22px;display:inline-flex;align-items:center;justify-content:center;';
        clickedPart.innerHTML = SUCCESS_FEEDBACK_SVG;
        btn.disabled = true;
        
        setTimeout(() => {
            clickedPart.innerHTML = originalContent;
            btn.disabled = false;
        }, 2000);
    }

    async function handleOfflineAction(btn, magnetLink) {
        await process115Offline(magnetLink);
        showButtonFeedback(btn, 'offline');
    }

    function handleBT4GSite() {
        try {
            const searchForm = document.querySelector('form[action="/search"]');
            const searchInput = document.getElementById('search');
            if (searchForm && searchInput && !document.getElementById('bt4g-advanced-filter')) {
                const keywordMaps = {
                    resolution: {
                        'SD': ['480p', '480P', '480i', '480I', '576p', '576P', '576i', '576I', 'SD', 'VCD', 'DVD', 'SDTV'],
                        '720p': ['720p', '720P', 'HD'],
                        '1080p': ['1080p', '1080P', 'HD1080P', 'FHD', 'FullHD', 'Full HD', '1920x1080'],
                        '1080i': ['1080i', '1080I', 'FHD', 'FullHD', 'Full HD', '1920x1080i'],
                        '1440p': ['1440p', '1440P', '2k', '2K', 'QHD', 'QuadHD', 'Quad HD', '2560x1440'],
                        '4K/UHD': ['2160p', '2160P', '4K', '4k', 'UHD', 'UltraHD', '3840x2160', '4096x2160'],
                        '8K': ['4320p', '4320P', '8K', 'FUHD', 'FUHD', '7680x4320']
                    },
                    hdr: {
                        'HDR': ['HDR'],
                        'HDR10': ['HDR10'],
                        'HDR10+': ['HDR10+', 'HDR10Plus'],
                        'HLG': ['HLG'],
                        'HDR Vivid': ['HDR Vivid', 'HDR-Vivid', 'Vivid'],
                        'Dolby Vision': ['DV', 'DoVi', 'DolbyVision', 'Dolby Vision']
                    },
                    codec: {
                        'H264/AVC': ['H264', '264', 'AVC', 'h264', 'MPEG4AVC', 'x264'],
                        'MPEG-2': ['MPEG-2', 'MPEG2', 'H262', 'H.262'],
                        'MPEG-4': ['MPEG-4', 'MPEG4', 'DivX', 'Xvid', 'XviD'],
                        'MPEG-5/EVC': ['MPEG-5', 'MPEG5', 'EVC'],
                        'H265/HEVC': ['H265', '265', 'HEVC', 'x265', 'h265'],
                        'AV1': ['AV1'],
                        'VC-1': ['VC-1', 'VC1'],
                        'VP8': ['VP8'],
                        'VP9': ['VP9']
                    },
                    mediaType: {
                        'BD': ['BD', 'BLURAY', 'BLU', 'RAY', 'BDMV', 'BDREMUX', 'REMUX'],
                        'BDrip/BRrip': ['BDrip', 'BRrip', 'BDRip', 'BRRip', 'BluRayRip'],
                        'WEB-DL': ['WEBDL', 'WEB-DL'],
                        'WEB': ['WEB', 'WEBRIP', 'WEBRip'],
                        'HDTV': ['HDTV', 'TV'],
                        'TS': ['TS', 'TS源'],
                        'DVD': ['DVD', 'DVDRIP', 'DVDRip'],
                        'TC': ['TC', '枪版', '抢版']
                    },
                    audio: {
                        '杜比': ['Dolby', 'DolbyDigital', 'DD'],
                        '杜比全景声': ['Atmos', 'DolbyAtmos'],
                        'Dolby Digital Plus': ['DD+', 'DDP', 'E-AC-3', 'EAC3', 'DolbyDigitalPlus', 'Dolby Digital Plus'],
                        'DTS': ['DTS', 'DTSHD', 'DTSHDMA', 'DTSX'],
                        'TrueHD': ['TrueHD', 'TRUEHD', 'TrueHD2', 'TrueHD.2', 'TrueHD.2.0', 'TrueHD5', 'TrueHD.5.1', 'TrueHD.5.1','TrueHD7', 'TrueHD.7', 'TrueHD.7.1'],
                        '通用': ['AAC', 'AC3', 'AC-3', 'MP3', 'LPCM', 'PCM', 'FLAC', 'Opus', 'OPUS']
                    }
                };

                const isDarkMode = document.body.classList.contains('dark-mode') ||
                    document.documentElement.classList.contains('dark') ||
                    document.documentElement.getAttribute('data-bs-theme') === 'dark';

                const submitBtn = searchForm.querySelector('button[type="submit"], input[type="submit"]');
                const filterBtn = document.createElement('button');
                filterBtn.type = 'button';
                filterBtn.textContent = '筛选';
                filterBtn.className = 'btn btn-secondary btn-sm';
                if (searchInput?.parentNode) {
                    searchInput.parentNode.insertBefore(filterBtn, searchInput.nextSibling);
                } else if (submitBtn?.parentNode) {
                    submitBtn.parentNode.insertBefore(filterBtn, submitBtn);
                } else {
                    searchForm.appendChild(filterBtn);
                }
                try {
                    searchInput.style.borderTopRightRadius = '0';
                    searchInput.style.borderBottomRightRadius = '0';
                    searchInput.style.position = 'relative';
                    searchInput.style.zIndex = '1';
                    filterBtn.style.margin = '0';
                    filterBtn.style.borderRadius = '0';
                    filterBtn.style.borderLeftWidth = '0';
                    filterBtn.style.position = 'relative';
                    filterBtn.style.zIndex = '2';
                    if (submitBtn) {
                        submitBtn.style.marginLeft = '-1px';
                        submitBtn.style.borderTopLeftRadius = '0';
                        submitBtn.style.borderBottomLeftRadius = '0';
                        submitBtn.style.position = 'relative';
                        submitBtn.style.zIndex = '3';
                    }
                } catch (_) {}

                const panel = document.createElement('div');
                panel.id = 'bt4g-advanced-filter';
                panel.style.display = 'none';
                panel.className = 'advanced-search mb-3 mt-2';
                updateFixedAdvancedSearchStyle(panel, isDarkMode);
                searchForm.parentNode.insertBefore(panel, searchForm.nextSibling);

                if (!document.getElementById('bt4g-advanced-style')) {
                    const style = document.createElement('style');
                    style.id = 'bt4g-advanced-style';
                    style.textContent = `
                    #bt4g-advanced-filter { font-size: 12px; line-height: 1.45; border-left: 3px solid rgba(13,110,253,.35); position: relative; padding-right: 40px; }
                    #bt4g-advanced-filter .bt4g-filter-row { padding: 6px 0; border-top: 1px dashed rgba(108,117,125,.25); display:flex; align-items:flex-start; }
                    #bt4g-advanced-filter .bt4g-filter-row:first-child { border-top: none; }
                    #bt4g-advanced-filter .bt4g-filter-row:hover { background: rgba(108,117,125,.05); border-radius: 5px; padding-left: 4px; margin-left: -4px; }
                    #bt4g-advanced-filter label.btn { border-radius: 8px; padding: 2px 10px; line-height: 1.3; border-width:1px; transition: all .12s ease-in-out; }
                    #bt4g-advanced-filter label.btn:hover { filter: brightness(0.97); transform: translateY(-0.5px); }
                    #bt4g-advanced-filter label.btn:active { transform: translateY(0); filter: brightness(0.95); }
                    #bt4g-advanced-filter .btn-check:focus + label,
                    #bt4g-advanced-filter label.btn:focus { box-shadow: 0 0 0 .12rem rgba(13,110,253,.15) !important; }
                    #bt4g-advanced-filter .bt4g-filter-row > span { width: 64px; flex: 0 0 64px; display:block; }
                    #bt4g-advanced-filter .bt4g-filter-row > div { display:flex; flex-wrap:wrap; gap:3px; flex:1 1 auto; min-width:0; }

                    form[action="/search"] { position: relative; }
                    form[action="/search"] #autocomplete-list,
                    form[action="/search"] .autocomplete-items { position: absolute !important; top: 100% !important; left: 0; right: 0; z-index: 1061 !important; }
                    form[action="/search"] { display:block; width:100%; }
                    #bt4g-advanced-filter { display:block; width:100% !important; max-width:none !important; flex:0 0 100%; align-self:stretch; clear:both; }
                    body:not(.dark):not(.dark-mode) #bt4g-advanced-filter label.btn.btn-outline-dark { background: rgba(108,117,125,.06); border-color: rgba(108,117,125,.35); color:#212529; }
                    body.dark-mode #bt4g-advanced-filter label.btn.btn-outline-light,
                    html.dark #bt4g-advanced-filter label.btn.btn-outline-light { background: rgba(255,255,255,.06); border-color: rgba(255,255,255,.25); color:#e9ecef; }

                    #bt4g-advanced-filter .bt4g-resolution .btn-check:checked + label.btn { background:#0d6efd !important; border-color:#0d6efd !important; color:#fff !important; }
                    #bt4g-advanced-filter .bt4g-hdr .btn-check:checked + label.btn { background:#6f42c1 !important; border-color:#6f42c1 !important; color:#fff !important; }
                    #bt4g-advanced-filter .bt4g-codec .btn-check:checked + label.btn { background:#198754 !important; border-color:#198754 !important; color:#fff !important; }
                    #bt4g-advanced-filter .bt4g-mediaType .btn-check:checked + label.btn { background:#20c997 !important; border-color:#20c997 !important; color:#fff !important; }
                    #bt4g-advanced-filter .bt4g-audio .btn-check:checked + label.btn { background:#d63384 !important; border-color:#d63384 !important; color:#fff !important; }

                    #bt4g-advanced-filter .bt4g-filter-row input[value=""]:checked + label.btn { background:#000 !important; border-color:#000 !important; color:#fff !important; }

                    #bt4g-advanced-filter .bt4g-reset-icon { position:absolute; top:6px; right:6px; width:24px; height:24px; border-radius:50%; display:flex; align-items:center; justify-content:center; cursor:pointer; z-index:5; background: transparent; border: 1px solid rgba(108,117,125,.35); color:#6c757d; padding:0; }
                    #bt4g-advanced-filter .bt4g-reset-icon:hover { background: rgba(108,117,125,.08); color:#495057; }
                    #bt4g-advanced-filter .bt4g-reset-icon:active { transform: scale(0.97); }
                    #bt4g-advanced-filter .bt4g-reset-icon svg { width:14px; height:14px; display:block; }
                    html.dark #bt4g-advanced-filter .bt4g-reset-icon,
                    body.dark-mode #bt4g-advanced-filter .bt4g-reset-icon { border-color: rgba(255,255,255,.25); color:#ced4da; }
                    html.dark #bt4g-advanced-filter .bt4g-reset-icon:hover,
                    body.dark-mode #bt4g-advanced-filter .bt4g-reset-icon:hover { background: rgba(255,255,255,.08); color:#e9ecef; }

                    @keyframes bt4g-spin-left { to { transform: rotate(-360deg); } }
                    #bt4g-advanced-filter .bt4g-reset-icon.spinning svg { animation: bt4g-spin-left .6s ease; }

                    @media (max-width: 576px) {
                      #bt4g-advanced-filter .bt4g-filter-row { padding: 4px 0; flex-direction: column; align-items: stretch; }
                      #bt4g-advanced-filter .bt4g-filter-row > span { width: 100%; flex: 0 0 100%; margin-bottom: 4px; }
                      #bt4g-advanced-filter .bt4g-filter-row > div { gap: 3px; width: 100%; flex: 0 0 100%; }
                    }
                    `;
                    document.head.appendChild(style);
                }

                function createOptionRow(name, label, choices, isDark) {
                    const row = document.createElement('div');
                    row.style.cssText = 'display:flex;align-items:center;margin-bottom:6px;width:100%';
                    row.classList.add('bt4g-filter-row', `bt4g-${name}`);
                    const labelEl = document.createElement('span');
                    labelEl.textContent = label;
                    labelEl.style.cssText = 'width:64px;margin-right:6px;white-space:nowrap;font-weight:bold;font-size:12px;';
                    labelEl.style.color = isDark ? '#e9ecef' : '#212529';
                    row.appendChild(labelEl);
                    const group = document.createElement('div');
                    group.style.cssText = 'display:flex;flex-wrap:wrap;gap:3px;';
                    choices.forEach((choice, idx) => {
                        const id = `${name}_${idx}`;
                        const radio = document.createElement('input');
                        radio.type = 'radio';
                        radio.name = name;
                        radio.id = id;
                        radio.value = choice.value;
                        radio.className = 'btn-check';
                        radio.checked = idx === 0;
                        const optLabel = document.createElement('label');
                        optLabel.className = isDark ? 'btn btn-outline-light btn-sm' : 'btn btn-outline-dark btn-sm';
                        optLabel.htmlFor = id;
                        optLabel.textContent = choice.label;
                        group.appendChild(radio);
                        group.appendChild(optLabel);
                    });
                    row.appendChild(group);
                    return row;
                }

                function updateFixedAdvancedSearchStyle(element, isDark) {
                    const backgroundColor = isDark ? '#212529' : '#f8f9fa';
                    const textColor = isDark ? '#e9ecef' : '#212529';
                    const borderColor = isDark ? '#3e444a' : '#dee2e6';
                    const shadow = isDark ? '0 2px 8px rgba(0,0,0,.25)' : '0 2px 10px rgba(0,0,0,.06)';
                    element.style.cssText = `display:flex;flex-direction:column;width:100%;padding:8px 10px;background-color:${backgroundColor};color:${textColor};border:1px solid ${borderColor};border-radius:6px;margin-bottom:10px;box-shadow:${shadow};`;
                }

                function setRadioValue(name, value) {
                    const radios = panel.querySelectorAll(`input[name="${name}"]`);
                    let found = false;
                    radios.forEach(r => { if (r.value === value) { r.checked = true; found = true; } });
                    if (!found && radios.length) radios[0].checked = true;
                }

                panel.appendChild(createOptionRow('resolution', '分辨规格:', [
                    { value: '', label: '全部' },
                    { value: 'SD', label: 'DVD/VCD' },
                    { value: '720p', label: '720p' },
                    { value: '1080p', label: '1080p' },
                    { value: '1080i', label: '1080i' },
                    { value: '1440p', label: '2K/QHD' },
                    { value: '4K/UHD', label: '4K/UHD' },
                    { value: '8K', label: '8K/FUHD' }
                ], isDarkMode));

                panel.appendChild(createOptionRow('hdr', '高动态域:', [
                    { value: '', label: '全部' },
                    { value: 'HDR', label: 'HDR' },
                    { value: 'HDR10', label: 'HDR10' },
                    { value: 'HDR10+', label: 'HDR10+' },
                    { value: 'HLG', label: 'HLG' },
                    { value: 'HDR Vivid', label: 'HDR Vivid' },
                    { value: 'Dolby Vision', label: 'Dolby Vision/DV' }
                ], isDarkMode));

                panel.appendChild(createOptionRow('codec', '视频编码:', [
                    { value: '', label: '全部' },
                    { value: 'H264/AVC', label: 'H264/AVC/x264' },
                    { value: 'MPEG-2', label: 'MPEG-2' },
                    { value: 'MPEG-4', label: 'MPEG-4/DivX/Xvid' },
                    { value: 'MPEG-5/EVC', label: 'MPEG-5/EVC' },
                    { value: 'H265/HEVC', label: 'H265/HEVC/x265' },
                    { value: 'AV1', label: 'AV1' },
                    { value: 'VC-1', label: 'VC-1' },
                    { value: 'VP8', label: 'VP8' },
                    { value: 'VP9', label: 'VP9' }
                ], isDarkMode));

                panel.appendChild(createOptionRow('mediaType', '媒体类型:', [
                    { value: '', label: '全部' },
                    { value: 'BD', label: 'BD/蓝光/REMUX' },
                    { value: 'BDrip/BRrip', label: 'BDrip/BRrip' },
                    { value: 'WEB-DL', label: 'WEB-DL' },
                    { value: 'WEB', label: 'WEB/WEBRip' },
                    { value: 'HDTV', label: 'HDTV' },
                    { value: 'TS', label: 'TS' },
                    { value: 'DVD', label: 'DVD/DVDRip' },
                    { value: 'TC', label: 'TC' }
                ], isDarkMode));

                const audioChoices = [
                    { value: '', label: '全部' },
                    { value: '杜比', label: '杜比/Dolby' },
                    { value: '杜比全景声', label: '杜比全景声/Atmos' },
                    { value: 'Dolby Digital Plus', label: 'Dolby Digital Plus/DD+/E-AC-3' },
                    { value: 'DTS', label: 'DTS系列' },
                    { value: 'TrueHD', label: 'TrueHD' },
                    { value: '通用', label: '通用' }
                ];
                panel.appendChild(createOptionRow('audio', '音频类型:', audioChoices, isDarkMode));

                const resetBtn = document.createElement('button');
                resetBtn.type = 'button';
                resetBtn.className = 'bt4g-reset-icon';
                resetBtn.title = '重置';
                resetBtn.setAttribute('aria-label', '重置');
                resetBtn.innerHTML = `
                    <svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true">
                        <polyline points="1 4 1 10 7 10" stroke="currentColor" stroke-width="1.6" stroke-linecap="round" stroke-linejoin="round"/>
                        <path d="M3.51 15a9 9 0 1 0 2.13-9.36L1 10" stroke="currentColor" stroke-width="1.6" stroke-linecap="round" stroke-linejoin="round"/>
                    </svg>`;
                resetBtn.addEventListener('click', () => {
                    resetBtn.classList.add('spinning');
                    const onAnimEnd = () => {
                        resetBtn.classList.remove('spinning');
                        resetBtn.removeEventListener('animationend', onAnimEnd, true);
                    };
                    resetBtn.addEventListener('animationend', onAnimEnd, true);
                    setTimeout(() => resetBtn.classList.remove('spinning'), 800);
                    ['resolution','hdr','codec','mediaType','audio'].forEach(name => {
                        const first = panel.querySelector(`input[name="${name}"]`);
                        if (first) first.checked = true;
                    });
                    localStorage.setItem('bt4g_advanced_settings', JSON.stringify({resolution:'',hdr:'',codec:'',mediaType:'',audio:''}));
                });
                panel.appendChild(resetBtn);

                filterBtn.addEventListener('click', () => {
                    const opened = panel.style.display !== 'none';
                    panel.style.display = opened ? 'none' : 'block';
                    localStorage.setItem('bt4g_filter_open', opened ? 'false' : 'true');
                });

                const urlParams = new URLSearchParams(window.location.search);
                function storeAdvanced() {
                    const settings = {
                        resolution: panel.querySelector('input[name="resolution"]:checked')?.value || '',
                        hdr: panel.querySelector('input[name="hdr"]:checked')?.value || '',
                        codec: panel.querySelector('input[name="codec"]:checked')?.value || '',
                        mediaType: panel.querySelector('input[name="mediaType"]:checked')?.value || '',
                        audio: panel.querySelector('input[name="audio"]:checked')?.value || ''
                    };
                    localStorage.setItem('bt4g_advanced_settings', JSON.stringify(settings));
                }
                (function restoreAdvanced(){
                    try {
                        const s = JSON.parse(localStorage.getItem('bt4g_advanced_settings')) || {};
                        if (s.resolution) setRadioValue('resolution', s.resolution);
                        if (s.hdr) setRadioValue('hdr', s.hdr);
                        if (s.codec) setRadioValue('codec', s.codec);
                        if (s.mediaType) setRadioValue('mediaType', s.mediaType);
                        if (s.audio) setRadioValue('audio', s.audio);
                        const open = localStorage.getItem('bt4g_filter_open');
                        panel.style.display = open === 'true' ? 'block' : 'none';
                    } catch {}
                })();

                panel.addEventListener('change', (e) => {
                    if (e.target && e.target.matches('input[type="radio"]')) {
                        storeAdvanced();
                    }
                });

                function processSearch(e) {
                    if (e) e.preventDefault();
                    const baseQuery = searchInput.value.trim();
                    if (!baseQuery) { searchForm.submit(); return; }
                    const resolution = panel.querySelector('input[name="resolution"]:checked')?.value || '';
                    const hdr = panel.querySelector('input[name="hdr"]:checked')?.value || '';
                    const codec = panel.querySelector('input[name="codec"]:checked')?.value || '';
                    const mediaType = panel.querySelector('input[name="mediaType"]:checked')?.value || '';
                    const audio = panel.querySelector('input[name="audio"]:checked')?.value || '';
                    storeAdvanced();
                    const conds = [];
                    if (resolution && keywordMaps.resolution[resolution]) conds.push(`(${keywordMaps.resolution[resolution].join('|')})`);
                    if (hdr && keywordMaps.hdr[hdr]) conds.push(`(${keywordMaps.hdr[hdr].join('|')})`);
                    if (codec && keywordMaps.codec[codec]) conds.push(`(${keywordMaps.codec[codec].join('|')})`);
                    if (mediaType && keywordMaps.mediaType[mediaType]) conds.push(`(${keywordMaps.mediaType[mediaType].join('|')})`);
                    if (audio && keywordMaps.audio[audio]) conds.push(`(${keywordMaps.audio[audio].join('|')})`);
                    let finalQuery = baseQuery;
                    if (conds.length) finalQuery += ' ' + conds.join(' ');
                    localStorage.setItem('bt4g_original_query', baseQuery);
                    searchInput.value = finalQuery;
                    searchForm.submit();
                }
                searchForm.addEventListener('submit', processSearch, { capture: true });
                searchInput.addEventListener('keydown', (e) => { if (e.key === 'Enter') processSearch(e); });

                if (new URLSearchParams(window.location.search).has('q')) {
                    const originalQuery = localStorage.getItem('bt4g_original_query');
                    if (originalQuery) setTimeout(() => { searchInput.value = originalQuery; }, 100);
                }
            }
        } catch (err) { console.warn('BT4G 筛选面板初始化失败:', err); }

        processMagnetLinks({
            selectors: '.result-item h5 > a[href^="/magnet/"]',
            containerStyles: { marginRight: '8px' },
            customProcessor: (titleA) => {
                const btnContainer = createButtonContainer({ marginRight: '8px' });
                titleA.parentNode.insertBefore(btnContainer, titleA);

                if (isAutoFetchEnabledFor('bt4g')) {
                    const loadingBtn = createLoadingButton();
                    btnContainer.appendChild(loadingBtn);
                    processBT4GMagnetLink(titleA, btnContainer).then(success => {
                        if (!success) {
                            setupRetryButton(loadingBtn, () => 
                                processBT4GMagnetLink(titleA, btnContainer, 2, 6000)
                            );
                        }
                    }).catch(error => {
                        console.error('BT4G处理失败:', error);
                        setButtonError(loadingBtn, '处理失败');
                    });
                } else {
                    const combinedBtn = createCombinedButtons(titleA);
                    btnContainer.appendChild(combinedBtn);
                }
            }
        });

        processElements('div.card-header', (headerEl) => {
            const card = headerEl.closest('.card') || document;
            const magnetBtn = card.querySelector('a[href*="downloadtorrentfile.com/hash/"]');
            if (!magnetBtn) return false;

            const btnContainer = createButtonContainer({ marginLeft: '8px' });
            const combinedBtn = createCombinedButtons(magnetBtn);
            btnContainer.appendChild(combinedBtn);
            headerEl.appendChild(btnContainer);
            return true;
        });
    }

    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);
            
            let magnet = `magnet:?xt=urn:btih:${hash}`;
            if (nameMatch?.[1]) {
                magnet += `&dn=${nameMatch[1]}`;
            }
            return magnet;
        } 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);
                }
            }
        });

        processElements('div.panel-heading > h3', (h3) => {
            const panel = h3.closest('.panel') || h3.parentElement;
            const magnetA = panel?.querySelector('div.media-body a[href^="magnet:"]');
            if (!magnetA || !magnetA.href) return false;

            const btnContainer = createButtonContainer({ marginLeft: '8px' });
            const combinedBtn = createCombinedButtons(magnetA.href);
            btnContainer.appendChild(combinedBtn);
            h3.appendChild(btnContainer);
            return true;
        });
    }

    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('table tr', (tr) => {
            const ths = tr.querySelectorAll('th');
            if (!ths || ths.length < 2) return false;

            const leftTh = ths[0];
            const leftText = (leftTh.textContent || '').replace(/\s+/g, ' ').trim();
            if (!/^Torrent\s*info$/i.test(leftText)) return false;

            const rightTh = ths[1];
            if (rightTh.dataset.buttonsAdded) return false;

            const pageMagnet = document.querySelector('a[href^="magnet:"]');
            const magnetHref = pageMagnet?.href;
            if (!magnetHref) return false;

            const btnContainer = createButtonContainer({ marginLeft: '8px' });
            btnContainer.style.display = 'inline-flex';
            btnContainer.style.verticalAlign = 'middle';
            btnContainer.style.float = 'left';

            const combinedBtn = createCombinedButtons(magnetHref);
            btnContainer.appendChild(combinedBtn);
            rightTh.appendChild(btnContainer);
            rightTh.dataset.buttonsAdded = true;
            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 isValid = await validate115Cookies();
return isValid;
} catch (error) {
console.error('检查登录(不可用)状态失败:', error);
return false;
}
}


function validate115Cookies() {
    return new Promise((resolve) => {
        GM_xmlhttpRequest({
            url: 'https://115.com/web/lixian/?ct=lixian&ac=task_lists&t=' + Date.now(),
            method: 'GET',
            anonymous: false,
            headers: {
                'Accept': 'application/json, text/plain, */*',
                'Referer': 'https://115.com/web/lixian/'
            },
            onload: function(response) {
                try {
                    const finalUrl = (response.finalUrl || '').toLowerCase();
                    const status = response.status || 0;
                    const headers = response.responseHeaders || '';
                    const contentType = (/^content-type:\s*([^\n]+)/im.exec(headers)?.[1] || '').toLowerCase();
                    const text = response.responseText || '';

                    const redirectedToLogin = /login\.115\.com|passport\.115\.com|passportapi\.115\.com/.test(finalUrl);
                    if (redirectedToLogin) return resolve(false);

                    let json = null;
                    if (contentType.includes('application/json') || text.trim().startsWith('{')) {
                        try { json = JSON.parse(text); } catch (_) { json = null; }
                    }
                    if (json) {
                        const ok = (json.state === true) || (json.errno === 0) || (json.code === 0) || (json.data && json.data.state === true);
                        if (ok) return resolve(true);

                        return resolve(false);
                    }

                    const hasLoginHints = /登录(不可用)|请先登录(不可用)|账户|sign\s*in|login|passport\.115\.com|window\.location|top\.location/i.test(text);
                    if (hasLoginHints) return resolve(false);

                    if (status === 401 || status === 403) return resolve(false);

                    return resolve(false);
                } catch (_) {
                    return resolve(false);
                }
            },
            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);
        }
    }
}

function submit115OfflineTask(magnetLink) {
    return new Promise((resolve, reject) => {
        GM_xmlhttpRequest({
            url: `https://115.com/web/lixian/?ct=lixian&ac=add_task_url&url=${encodeURIComponent(magnetLink)}`,
            method: 'GET',
            anonymous: false,
            headers: {
                'Referer': 'https://115.com/web/lixian/'
            },
            onload: function(response) {
                const result = tryParseJson(response.responseText);
                resolve(result);
            },
            onerror: function() {
                reject(new Error('提交离线任务失败'));
            }
        });
    });
}

function tryParseJson(text) {
    try {
        return JSON.parse(text);
    } catch (e) {
        return null;
    }
}

function handleOfflineResult(result) {
    if (!result) {
        showNotification('115离线失败', '接口返回为空或解析失败');
        return;
    }

    const success = result.state === true || result.errno === 0 || result.code === 0;
    const message = result.message || result.msg || result.error || result.errmsg || result?.data?.errmsg || '';

    const err = Number.isFinite(result.errno) ? result.errno : (Number.isFinite(result.code) ? result.code : null);
    const rawText = typeof result === 'string' ? result : JSON.stringify(result);
    const txt = [message, rawText].filter(Boolean).join(' ');

    const duplicateMsgRe = /(已存在|已经存在|已添加|重复|已提交过|exist|exists|already)/i;
    const duplicateFlag = !!(
        duplicateMsgRe.test(txt) ||
        result?.data?.exists === true ||
        result?.exists === true ||
        [911, 10008, 10009, 10010].includes(err)
    );

    if (duplicateFlag) {
        showNotification('115离线', '任务已存在');
        return;
    }

    if (success) {
        showNotification('115离线成功', message || '任务已提交');
    } else {
        const text = message || rawText;
        showNotification('115离线失败', text);
    }
}

    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: '2147483647',
            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 ensureModalStyles() {
        if (document.getElementById('magnet-script-modal-styles')) return;
        const style = document.createElement('style');
        style.id = 'magnet-script-modal-styles';
        style.textContent = `
        .modal-overlay{position:fixed;inset:0;background:rgba(0,0,0,.08);z-index:10000;display:flex;align-items:center;justify-content:center;padding:16px}
        .modal-content{background:#fff;color:#333;border-radius:10px;box-shadow:0 12px 40px rgba(0,0,0,.08);width:760px;max-width:96vw;padding:16px 18px;font-family:system-ui,-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica,Arial,sans-serif}
        .modal-header{display:flex;align-items:center;justify-content:space-between;margin:0;padding:0;min-height:32px}
        .modal-header-colored{background:linear-gradient(90deg, rgba(66,133,244,.12), rgba(66,133,244,0));padding:2px 8px;border-radius:4px;margin:0 0 1px;display:flex;align-items:center;min-height:inherit}
        .modal-title{margin:0;font-size:18px;font-weight:700;color:#333;line-height:1;padding:2px 0}
        .modal-header-help{font-size:12px;color:#666;margin-left:8px;line-height:1}
        .modal-desc{font-size:12px;color:#666;margin-bottom:12px}
        .modal-section-title{font-weight:600;margin:12px 0 6px;padding:6px 10px;background:rgba(66,133,244,.08);border-left:3px solid #4285f4;border-radius:4px;color:#333;font-size:14px;display:flex;align-items:center;min-height:32px;line-height:1.2}
        .modal-section-header{display:flex;align-items:center;justify-content:space-between;gap:8px;min-height:32px;padding:2px 0}
        .vip-badge{display:inline-flex;align-items:center;gap:6px;margin-left:10px;padding:0 8px;height:18px;line-height:16px;font-size:11px;border-radius:999px;white-space:nowrap;border:1px solid #e5e7eb;color:#374151;background:#f9fafb}
        .vip-badge .dot{width:6px;height:6px;border-radius:50%;background:#9ca3af}
        /* VIP 渐变金 */
        .vip-badge.vip{color:#7c2d12;background:linear-gradient(90deg,#fde68a,#ffc10778,#fcd34d);border-color:#f59e0b}
        .vip-badge.vip .dot{background:#f59e0b}
        /* 未登录(不可用)浅红 */
        .vip-badge.not-logged{color:#991b1b;background:#fef2f2;border-color:#fecaca}
        .vip-badge.not-logged .dot{background:#ef4444}
        /* 原石/原始/非会员浅灰 */
        .vip-badge.non-vip{color:#374151;background:#f3f4f6;border-color:#e5e7eb}
        .vip-badge.non-vip .dot{background:#9ca3af}
        .vip-badge .vip-action{appearance:none;border:none;background:transparent;color:#b45309;cursor:pointer;padding:0 0 0 6px;margin:0;border-left:1px solid #fcd34d;font-size:11px;line-height:16px}
        .vip-badge .vip-action:hover{color:#92400e}
        .modal-form-group{display:flex;align-items:center;justify-content:space-between;gap:10px;margin:8px 0}
        .modal-label{min-width:160px;font-size:13px;color:#555}
        .modal-form-texts{display:flex;flex-direction:column;gap:4px;flex:1}
        .modal-help{font-size:12px;color:#999}
        .modal-control{margin-left:auto}
        .modal-indent{margin-left:20px}
        .modal-row-center{display:flex;justify-content:center;align-items:center;gap:12px;margin:8px 0}
        .modal-two-col{display:grid;grid-template-columns:1fr 1fr;gap:12px;align-items:start}
        .modal-three-col{display:grid;grid-template-columns:repeat(3,1fr);gap:10px}
        .modal-four-col{display:grid;grid-template-columns:repeat(4,1fr);gap:10px}
        .modal-tile{display:flex;align-items:center;justify-content:space-between;background:#f8f9fb;border:1px solid #eee;border-radius:8px;padding:8px 10px}
        .modal-tile-label{font-size:13px;color:#555}
        .modal-tip{font-size:12px;color:#8a6d3b;margin-bottom:8px;background:#fff6e5;border:1px solid #ffe5b7;border-radius:6px;padding:8px 10px;text-align:center}
        .modal-input{width:auto}
        .modal-btn{padding:8px 12px;font-size:13px;border:none;border-radius:6px;cursor:pointer}
        .modal-btn-primary{background:#4285f4;color:#fff}
        .modal-btn-primary:hover{background:#3367d6}
        .modal-btn-secondary{background:#f5f5f5;color:#333}
        .modal-btn-secondary:hover{background:#eaeaea}
        .modal-btn-success{background:#16a34a;color:#fff}
        .modal-btn-success:hover{background:#15803d}
        .modal-btn-danger{background:#ef4444;color:#fff}
        .modal-btn-danger:hover{background:#dc2626}
        .modal-btn-warning{background:#f59e0b;color:#fff}
        .modal-btn-warning:hover{background:#d97706}
        .modal-footer{display:flex;justify-content:flex-end;gap:10px;margin-top:12px}
        .toggle{position:relative;display:inline-block;width:44px;height:24px;vertical-align:middle}
        .toggle input{opacity:0;width:0;height:0;position:absolute}
        .toggle-slider{position:absolute;cursor:pointer;top:0;left:0;right:0;bottom:0;background:#e5e7eb;transition:.2s ease;border-radius:9999px;box-shadow:inset 0 0 0 1px rgba(0,0,0,.08)}
        .toggle-slider:before{content:"";position:absolute;height:18px;width:18px;left:3px;top:3px;background:#fff;border-radius:50%;transition:.2s ease;box-shadow:0 1px 2px rgba(0,0,0,.25)}
        .toggle input:checked + .toggle-slider{background:#4285f4}
        .toggle input:checked + .toggle-slider:before{transform:translateX(20px)}
        .toggle-disabled .toggle-slider{opacity:.55;cursor:not-allowed}
        .floating-settings-btn{position:fixed;left:18px;bottom:18px;width:44px;height:44px;border-radius:50%;background:rgba(255,255,255,.6);backdrop-filter:saturate(180%) blur(12px);-webkit-backdrop-filter:saturate(180%) blur(12px);color:#333;display:flex;align-items:center;justify-content:center;box-shadow:0 8px 24px rgba(0,0,0,.15), inset 0 0 0 1px rgba(0,0,0,.06);cursor:grab;user-select:none;z-index:10000;transition:background .2s ease, box-shadow .2s ease, transform .1s ease}
        .floating-settings-btn:hover{background:rgba(255,255,255,.78);box-shadow:0 10px 28px rgba(0,0,0,.18), inset 0 0 0 1px rgba(0,0,0,.08)}
        .floating-settings-btn.dragging{cursor:grabbing;opacity:.95;transform:scale(.98)}
        `;
        document.head.appendChild(style);
    }

    function ensureFloatingSettingsButton() {
        const existing = document.getElementById('magnet-floating-settings-btn');
        if (!CONFIG.enableFloatingSettingsBtn) {
            if (existing) existing.remove();
            return;
        }
        if (existing) return;

        const btn = document.createElement('div');
        btn.id = 'magnet-floating-settings-btn';
        btn.className = 'floating-settings-btn';
        btn.title = '设置';
        btn.textContent = '⚙️';
        btn.style.lineHeight = '44px';
        btn.style.fontSize = '18px';

        try {
            const pos = GM_getValue('floatingBtnPos', null);
            if (pos && typeof pos.left === 'number' && typeof pos.top === 'number') {
                const w = 44, h = 44;
                const maxX = Math.max(0, window.innerWidth - w);
                const maxY = Math.max(0, window.innerHeight - h);
                const safeLeft = Math.min(Math.max(0, pos.left), maxX);
                const safeTop = Math.min(Math.max(0, pos.top), maxY);
                btn.style.left = safeLeft + 'px';
                btn.style.top = safeTop + 'px';
                btn.style.right = 'auto';
                btn.style.bottom = 'auto';
            }
        } catch (_) {}

        btn.addEventListener('click', (e) => {
            e.stopPropagation();
            openSettingsPanel({ suppress115Toast: true });
        });

        btn.addEventListener('mousedown', (e) => {
            if (e.button !== 0) return;
            e.preventDefault();
            const rect = btn.getBoundingClientRect();
            const offsetX = e.clientX - rect.left;
            const offsetY = e.clientY - rect.top;
            btn.classList.add('dragging');

            const onMove = (ev) => {
                const x = Math.min(Math.max(0, ev.clientX - offsetX), window.innerWidth - rect.width);
                const y = Math.min(Math.max(0, ev.clientY - offsetY), window.innerHeight - rect.height);
                btn.style.left = x + 'px';
                btn.style.top = y + 'px';
                btn.style.right = 'auto';
                btn.style.bottom = 'auto';
            };
            const onUp = () => {
                document.removeEventListener('mousemove', onMove);
                document.removeEventListener('mouseup', onUp);
                btn.classList.remove('dragging');
                try {
                    const left = parseFloat(btn.style.left || '0');
                    const top = parseFloat(btn.style.top || '0');
                    GM_setValue('floatingBtnPos', { left, top });
                } catch (_) {}
            };
            document.addEventListener('mousemove', onMove);
            document.addEventListener('mouseup', onUp);
        });

        document.body.appendChild(btn);
    }

    function openSettingsPanel(options = {}) {
        const suppress115Toast = !!options.suppress115Toast;
        const existing = document.getElementById('magnet-script-settings-overlay');
        if (existing) existing.remove();

        ensureModalStyles();

        const overlay = document.createElement('div');
        overlay.id = 'magnet-script-settings-overlay';
        overlay.className = 'modal-overlay';

        const panel = document.createElement('div');
        panel.className = 'modal-content';
        if (!document.getElementById('magnet-close-style')) {
            const style = document.createElement('style');
            style.id = 'magnet-close-style';
            style.textContent = `
                .modal-close-btn { 
                    appearance: none;
                    background: rgba(66,133,244, .15);
                    color: #4285f4;
                    border: none;
                    width: 28px; height: 28px;
                    padding: 0; margin: 0;
                    border-radius: 999px;
                    display: inline-flex; align-items: center; justify-content: center;
                    border: none;
                }
                .modal-close-btn:hover { background: rgba(66,133,244, .25); }
                .modal-close-btn:focus-visible { outline: none; box-shadow: 0 0 0 3px rgba(66,133,244,.35); }
                .modal-close-icon { width: 16px; height: 16px; display: inline-block; transition: transform .35s ease; }
                .modal-close-btn:hover .modal-close-icon { transform: rotate(180deg); }
            `;
            document.head.appendChild(style);
        }

        const header = document.createElement('div');
        header.className = 'modal-header modal-header-colored';
        const titleEl = document.createElement('h3');
        titleEl.className = 'modal-title';
        titleEl.textContent = '设置';
        const headerRight = document.createElement('div');
        headerRight.style.marginLeft = 'auto';
        const btnCloseX = document.createElement('button');
        btnCloseX.type = 'button';
        btnCloseX.title = '关闭';
        btnCloseX.className = 'modal-close-btn';
        btnCloseX.innerHTML = `
            <svg class="modal-close-icon" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true">
                <path d="M6 6l12 12M18 6L6 18" stroke="currentColor" stroke-width="2" stroke-linecap="round"/>
            </svg>
        `;
        btnCloseX.addEventListener('click', () => overlay.remove());
        headerRight.appendChild(btnCloseX);
        header.append(titleEl, headerRight);

        const section = (title) => {
            const s = document.createElement('div');
            const t = document.createElement('div');
            t.textContent = title;
            t.className = 'modal-section-title';
            s.appendChild(t);
            return s;
        };

        const row = (labelText, inputEl, helpText = '') => {
            const r = document.createElement('div');
            r.className = 'modal-form-group';
            const texts = document.createElement('div');
            texts.className = 'modal-form-texts';
            const l = document.createElement('label');
            l.textContent = labelText;
            l.className = 'modal-label';
            texts.appendChild(l);
            if (helpText) {
                const h = document.createElement('div');
                h.textContent = helpText;
                h.className = 'modal-help';
                texts.appendChild(h);
            }
            r.appendChild(texts);
            inputEl.classList?.add('modal-input', 'modal-control');
            r.appendChild(inputEl);
            return r;
        };

        const mkSwitch = (checked, onChange) => {
            const wrapper = document.createElement('label');
            wrapper.className = 'toggle';
            const input = document.createElement('input');
            input.type = 'checkbox';
            input.checked = !!checked;
            input.addEventListener('change', () => onChange(!!input.checked));
            const slider = document.createElement('span');
            slider.className = 'toggle-slider';
            wrapper.append(input, slider);
            Object.defineProperty(wrapper, 'disabled', {
                get() { return input.disabled; },
                set(v) { input.disabled = !!v; wrapper.classList.toggle('toggle-disabled', !!v); }
            });
            return wrapper;
        };

        const sec115 = section('115功能');
        const sec115TitleEl = sec115.querySelector?.('.modal-section-title');
        if (sec115TitleEl) {
            sec115TitleEl.classList.add('modal-section-header');
            sec115TitleEl.style.padding = '4px 8px';
            const vipBadge = document.createElement('span');
            vipBadge.className = 'vip-badge';
            vipBadge.innerHTML = '<span class="dot"></span><span class="text">VIP 信息加载中...</span>';
            sec115TitleEl.appendChild(vipBadge);

            const applyBadgeStatus = (status) => {
                vipBadge.classList.remove('vip', 'not-logged', 'non-vip');
                if (status) vipBadge.classList.add(status);
            };
            const setVipBadge = (txt, opts = {}) => {
                const { withLoginBtn = false, status = null } = opts;
                const t = vipBadge.querySelector('.text');
                if (t) t.textContent = txt;
                applyBadgeStatus(status);
                const oldBtn = vipBadge.querySelector('.vip-action');
                if (oldBtn) oldBtn.remove();
                if (withLoginBtn) {
                    const btn = document.createElement('button');
                    btn.type = 'button';
                    btn.className = 'vip-action';
                    btn.textContent = '去登录(不可用)';
                    btn.addEventListener('click', (e) => {
                        e.stopPropagation();
                        window.open('https://115.com/?mode=login', '_blank');
                    });
                    vipBadge.appendChild(btn);
                }
            };

            const CACHE_TTL = 10 * 60 * 1000;
            const readCache = (key) => { try { return GM_getValue(key, null); } catch (_) { return null; } };
            const writeCache = (key, data) => { try { GM_setValue(key, { t: Date.now(), data }); } catch (_) {} };
            const useCacheValid = (key) => {
                const c = readCache(key);
                return c && (Date.now() - (c.t || 0) < CACHE_TTL) ? c.data : null;
            };

            let latestVip = null;
            let vipFetched = false;
            let latestQuota = null;

            const updateBadge = () => {
                const okVip = latestVip && latestVip.state === true;
                const lvName = okVip ? (latestVip?.data?.user_limit?.level_name || '') : '';
                const quotaOk = latestQuota && typeof latestQuota.surplus !== 'undefined' && typeof latestQuota.count !== 'undefined';
                const isNonVip = lvName && /原石|原始|非会员/i.test(lvName);
                const statusClass = !lvName ? null : (isNonVip ? 'non-vip' : 'vip');

                if (!vipFetched) {
                    setVipBadge('VIP 信息加载中...');
                    return;
                }

                if (!okVip && !lvName) {
                    setVipBadge('未登录(不可用)或无权限', { withLoginBtn: true, status: 'not-logged' });
                    return;
                }

                if (quotaOk) {
                    const text = (lvName ? `等级:${lvName},` : '') + `离线额度:${latestQuota.surplus}/${latestQuota.count}`;
                    setVipBadge(text, { status: statusClass });
                } else if (lvName) {
                    setVipBadge(`等级:${lvName}`, { status: statusClass });
                } else {
                    setVipBadge('未登录(不可用)或无权限', { withLoginBtn: true, status: 'not-logged' });
                }
            };

            const fetchVip = (bypassCache = false) => {
                if (!bypassCache) {
                    const cached = useCacheValid('vip_info_cache_v2');
                    if (cached) { latestVip = cached; updateBadge(); }
                }
                GM_xmlhttpRequest({
                    url: 'https://webapi.115.com/user/vip_limit?feature=2',
                    method: 'GET',
                    anonymous: false,
                    headers: {
                        'Accept': 'application/json, text/plain, */*',
                        'Referer': 'https://115.com/web/lixian/'
                    },
                    onload: function (resp) {
                        try {
                            latestVip = JSON.parse(resp.responseText || 'null');
                            vipFetched = true;
                            writeCache('vip_info_cache_v2', latestVip);
                            updateBadge();
                        } catch (_) {
                            vipFetched = true;
                            updateBadge();
                        }
                    },
                    onerror: function () { vipFetched = true; updateBadge(); }
                });
            };

            const fetchQuota = (bypassCache = false) => {
                if (!bypassCache) {
                    const cached = useCacheValid('vip_quota_cache_v1');
                    if (cached) { latestQuota = cached; updateBadge(); }
                }
                GM_xmlhttpRequest({
                    url: 'https://115.com/web/lixian/?ct=lixian&ac=get_quota_package_info',
                    method: 'GET',
                    anonymous: false,
                    headers: {
                        'Accept': 'application/json, text/plain, */*',
                        'Referer': 'https://115.com/web/lixian/'
                    },
                    onload: function (resp) {
                        try {
                            const json = JSON.parse(resp.responseText || 'null');
                            const data = {
                                surplus: Number(json?.surplus ?? json?.package?.["1"]?.surplus ?? 0),
                                count: Number(json?.count ?? json?.package?.["1"]?.count ?? 0),
                                used: Number(json?.used ?? json?.package?.["1"]?.used ?? 0),
                            };
                            latestQuota = data;
                            writeCache('vip_quota_cache_v1', data);
                            updateBadge();
                        } catch (_) {
                            updateBadge();
                        }
                    },
                    onerror: function () { updateBadge(); }
                });
            };

            fetchVip(false);
            fetchQuota(false);

            const origCheck = check115AndUpdate;
            const bindRefresh = () => {
                if (typeof check115AndUpdate === 'function' && check115AndUpdate !== origCheck) return;
                if (typeof check115AndUpdate === 'function') {
                    const wrapped = async (notify = true) => {
                        const res = await origCheck(notify);
                        fetchVip(true);
                        fetchQuota(true);
                        return res;
                    };
                    window.check115AndUpdate = wrapped;
                } else {
                    setTimeout(bindRefresh, 50);
                }
            };
            bindRefresh();
        }

        const statusBtn = document.createElement('button');
        statusBtn.textContent = '检测115登录(不可用)状态';
        statusBtn.className = 'modal-btn modal-btn-secondary';
        const setStatus = (type) => {
            statusBtn.className = 'modal-btn';
            switch (type) {
                case 'checking':
                    statusBtn.classList.add('modal-btn-warning');
                    statusBtn.textContent = '检测中...';
                    break;
                case 'ok':
                    statusBtn.classList.add('modal-btn-success');
                    statusBtn.textContent = '账号已登录(不可用)';
                    break;
                case 'no':
                    statusBtn.classList.add('modal-btn-danger');
                    statusBtn.textContent = '当前浏览器未登录(不可用)';
                    break;
                default:
                    statusBtn.classList.add('modal-btn-warning');
                    statusBtn.textContent = '检测异常';
            }
        };
        async function check115AndUpdate(notify = true) {
            try {
                setStatus('checking');
                const ok = await check115Login(true);
                setStatus(ok ? 'ok' : 'no');
                if (notify) showNotification('115状态', ok ? '已登录(不可用)' : '未登录(不可用)');
            } catch (e) {
                console.error(e);
                setStatus('error');
                if (notify) showNotification('115状态', '检测异常');
            }
        }
        window.check115AndUpdate = check115AndUpdate;
        statusBtn.addEventListener('click', () => window.check115AndUpdate(true));
        const btnOpen115 = document.createElement('button');
        btnOpen115.textContent = '打开115网盘';
        btnOpen115.className = 'modal-btn modal-btn-primary';
        btnOpen115.addEventListener('click', () => window.open('https://115.com/?cid=0&offset=0&mode=wangpan', '_blank'));
        const row115 = document.createElement('div');
        row115.className = 'modal-row-center';
        row115.append(statusBtn, btnOpen115);
        sec115.appendChild(row115);
        setTimeout(() => window.check115AndUpdate(!suppress115Toast), 100);

        const secButtons = section('按钮显示');
        const btnsTitleEl = secButtons.querySelector?.('.modal-section-title');
        if (btnsTitleEl) {
            btnsTitleEl.classList.add('modal-section-header');
            btnsTitleEl.style.padding = '4px 8px';
            const btnsTip = document.createElement('span');
            btnsTip.className = 'modal-tip';
            btnsTip.style.margin = '0 0 0 10px';
            btnsTip.style.padding = '0 6px';
            btnsTip.style.fontSize = '11px';
            btnsTip.style.lineHeight = '16px';
            btnsTip.style.height = '18px';
            btnsTip.style.display = 'inline-flex';
            btnsTip.style.alignItems = 'center';
            btnsTip.style.boxSizing = 'border-box';
            btnsTip.textContent = '提示:可通过拖拽右侧三项调整顺序;开关控制子按钮显示/隐藏,变更将即时应用到页面。';
            btnsTitleEl.appendChild(btnsTip);
        }
        const swCopy = mkSwitch(CONFIG.enableCopyButton, (v) => { CONFIG.enableCopyButton = v; GM_setValue('enableCopyButton', v); showNotification('设置已保存', v ? '已启用“复制”按钮' : '已禁用“复制”按钮'); addActionButtons(); });
        const swOffline = mkSwitch(CONFIG.enableOfflineButton, (v) => { CONFIG.enableOfflineButton = v; GM_setValue('enableOfflineButton', v); showNotification('设置已保存', v ? '已启用“离线”按钮' : '已禁用“离线”按钮'); addActionButtons(); });
        const swOpen = mkSwitch(CONFIG.enableOpenButton, (v) => { CONFIG.enableOpenButton = v; GM_setValue('enableOpenButton', v); showNotification('设置已保存', v ? '已启用“打开”按钮' : '已禁用“打开”按钮'); addActionButtons(); });
        const btnGrid = document.createElement('div');
        btnGrid.className = 'modal-four-col';
        const makeBtnTile = (label, sw) => {
            const item = document.createElement('div');
            item.className = 'modal-tile';
            const name = document.createElement('span');
            name.className = 'modal-tile-label';
            name.textContent = label;
            const right = document.createElement('div');
            right.appendChild(sw);
            item.append(name, right);
            return item;
        };
        const tileCopy = makeBtnTile('复制按钮', swCopy); tileCopy.dataset.type = 'copy'; tileCopy.draggable = true;
        const tileOffline = makeBtnTile('离线按钮', swOffline); tileOffline.dataset.type = 'offline'; tileOffline.draggable = true;
        const tileOpen = makeBtnTile('打开按钮', swOpen); tileOpen.dataset.type = 'open'; tileOpen.draggable = true;

        const tilesMap = { copy: tileCopy, offline: tileOffline, open: tileOpen };
        const renderTilesByOrder = () => {
            btnGrid.innerHTML = '';
            const ord = getButtonsOrder();
            ord.forEach(t => { const el = tilesMap[t]; if (el) btnGrid.appendChild(el); });
        };
        renderTilesByOrder();

        let draggingEl = null;
        btnGrid.addEventListener('dragstart', (e) => {
            const tile = e.target.closest('.modal-tile');
            if (!tile) return;
            draggingEl = tile;
            e.dataTransfer.effectAllowed = 'move';
            try { e.dataTransfer.setData('text/plain', tile.dataset.type || ''); } catch (_) {}
            tile.style.opacity = '0.6';
        });
        btnGrid.addEventListener('dragend', () => {
            if (draggingEl) draggingEl.style.opacity = '';
            draggingEl = null;
            const newOrder = Array.from(btnGrid.querySelectorAll('.modal-tile')).map(el => el.dataset.type).filter(Boolean);
            if (newOrder.length) {
                setButtonsOrder(newOrder);
                applyButtonsOrderToExisting();
            }
        });
        btnGrid.addEventListener('dragover', (e) => {
            e.preventDefault();
            const tile = e.target.closest('.modal-tile');
            if (!tile || !draggingEl || tile === draggingEl) return;
            const rect = tile.getBoundingClientRect();
            const before = (e.clientY - rect.top) < rect.height / 2;
            btnGrid.insertBefore(draggingEl, before ? tile : tile.nextSibling);
        });

        secButtons.appendChild(btnGrid);
        applyButtonsOrderToExisting();

        const secFloat = section('悬浮设置按钮');
        const swFloat = mkSwitch(CONFIG.enableFloatingSettingsBtn, (v) => {
            CONFIG.enableFloatingSettingsBtn = v;
            GM_setValue('enableFloatingSettingsBtn', v);
            showNotification('设置已保存', v ? '已启用悬浮设置按钮' : '已关闭悬浮设置按钮');
            ensureFloatingSettingsButton();
        });
        secFloat.appendChild(row('启用左下角悬浮设置按钮', swFloat, '可拖动,点击打开“设置”'));

        const secAuto = document.createElement('div');
        const secAutoHeader = document.createElement('div');
        secAutoHeader.className = 'modal-section-title modal-section-header';
        const secAutoTitle = document.createElement('span');
        secAutoTitle.textContent = '自动异步获取磁力链';
        secAutoHeader.appendChild(secAutoTitle);
        secAutoHeader.style.padding = '4px 8px';
        const riskTip = document.createElement('span');
        riskTip.textContent = '提示:自动异步获取可能触发部分站点风控,按需开启。';
        riskTip.className = 'modal-tip';
        riskTip.style.margin = '0 0 0 10px';
        riskTip.style.padding = '0 6px';
        riskTip.style.fontSize = '11px';
        riskTip.style.lineHeight = '16px';
        riskTip.style.height = '18px';
        riskTip.style.display = 'inline-flex';
        riskTip.style.alignItems = 'center';
        riskTip.style.boxSizing = 'border-box';
        secAutoHeader.appendChild(riskTip);

        const masterSwitch = mkSwitch(CONFIG.autoFetchEnabled, (v) => {
            CONFIG.autoFetchEnabled = v; GM_setValue('autoFetchEnabled', v);
            showNotification('设置已保存', v ? '已开启自动异步获取' : '已关闭自动异步获取');
            [swBt4g, swSeedhub, swYhg, swCmg].forEach(sw => sw.disabled = !v);
            addActionButtons();
        });
        secAutoHeader.appendChild(masterSwitch);
        secAuto.appendChild(secAutoHeader);

        const swBt4g = mkSwitch(CONFIG.autoFetchSites.bt4g, (v) => {
            CONFIG.autoFetchSites.bt4g = v; GM_setValue('autoFetch_bt4g', v); addActionButtons();
        });
        const swSeedhub = mkSwitch(CONFIG.autoFetchSites.seedhub, (v) => {
            CONFIG.autoFetchSites.seedhub = v; GM_setValue('autoFetch_seedhub', v); addActionButtons();
        });
        const swYhg = mkSwitch(CONFIG.autoFetchSites.yuhuage, (v) => {
            CONFIG.autoFetchSites.yuhuage = v; GM_setValue('autoFetch_yuhuage', v); addActionButtons();
        });
        const swCmg = mkSwitch(CONFIG.autoFetchSites.cilimag, (v) => {
            CONFIG.autoFetchSites.cilimag = v; GM_setValue('autoFetch_cilimag', v); addActionButtons();
        });

        [swBt4g, swSeedhub, swYhg, swCmg].forEach(sw => sw.disabled = !CONFIG.autoFetchEnabled);

        const grid = document.createElement('div');
        grid.className = 'modal-four-col';

        const makeTile = (label, sw) => {
            const item = document.createElement('div');
            item.className = 'modal-tile';
            const name = document.createElement('span');
            name.className = 'modal-tile-label';
            const m = label.match(/^(.*)\((.*)\)$/);
            if (m) {
                let left = (m[1] || '').trim();
                let inside = (m[2] || '').trim();
                const hasCn = (s) => /[\u4e00-\u9fff]/.test(s);
                if (!hasCn(left) && hasCn(inside)) {
                    [left, inside] = [inside, left];
                }
                name.innerHTML = `${left}<br><span style="font-size:12px;color:#9CA3AF">(${inside})</span>`;
            } else {
                name.textContent = label;
            }
            const right = document.createElement('div');
            right.appendChild(sw);
            item.append(name, right);
            return item;
        };

        grid.append(
            makeTile('BT4G', swBt4g),
            makeTile('SeedHub', swSeedhub),
            makeTile('雨花阁(YuHuaGe)', swYhg),
            makeTile('ØMagnet(无极磁链)', swCmg)
        );
        secAuto.appendChild(grid);

        const secSites = section('网站规则');
        const sitesTitleEl = secSites.querySelector?.('.modal-section-title');
        if (sitesTitleEl) {
            sitesTitleEl.classList.add('modal-section-header');
            sitesTitleEl.style.padding = '4px 8px';
        }
        const sitesTip = document.createElement('span');
        sitesTip.className = 'modal-tip';
        sitesTip.style.margin = '0 0 0 10px';
        sitesTip.style.padding = '0 6px';
        sitesTip.style.fontSize = '11px';
        sitesTip.style.lineHeight = '16px';
        sitesTip.style.height = '18px';
        sitesTip.style.display = 'inline-flex';
        sitesTip.style.alignItems = 'center';
        sitesTip.style.boxSizing = 'border-box';
        sitesTip.textContent = '测速提示:频繁测速请求可能导致站点风控,建议合理使用测速,必要时放慢操作节奏。';
        const sitesHeader = secSites.querySelector?.('.modal-section-title') || secSites;
        sitesHeader.appendChild(sitesTip);

        const siteGrid = document.createElement('div');
        siteGrid.className = 'modal-four-col';

        if (!document.getElementById('magnet-site-links-icon-style')) {
            const style = document.createElement('style');
            style.id = 'magnet-site-links-icon-style';
            style.textContent = `
                .site-link-icon {\n\
                    background-image: url("data:image/svg+xml,%3Csvg width='14' height='14' viewBox='0 0 24 24' fill='none' stroke='currentColor' stroke-width='2' xmlns='http://www.w3.org/2000/svg'%3E%3Ccircle cx='12' cy='12' r='10'/%3E%3Cpath d='M2 12h20'/%3E%3Cpath d='M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z'/%3E%3C/svg%3E");\n\
                    background-repeat: no-repeat;\n\
                    background-size: contain;\n\
                    width: 14px;\n\
                    height: 14px;\n\
                    display: inline-block;\n\
                    vertical-align: -2px;\n\
                }\n\
                .modal-btn:hover .site-link-icon {\n\
                    background-image: url("data:image/svg+xml,%3Csvg width='14' height='14' viewBox='0 0 24 24' fill='none' stroke='%234caf50' stroke-width='2' xmlns='http://www.w3.org/2000/svg'%3E%3Ccircle cx='12' cy='12' r='10'/%3E%3Cpath d='M2 12h20'/%3E%3Cpath d='M12 2a15.3 15.3 0 0 1 4 10 15.3 15.3 0 0 1-4 10 15.3 15.3 0 0 1-4-10 15.3 15.3 0 0 1 4-10z'/%3E%3C/svg%3E");\n\
                }\n            `;
            document.head.appendChild(style);
        }

        const showLinksPopover = (triggerEl, links) => {
            document.querySelectorAll('.site-links-popover').forEach(p => p.remove());
            const pop = document.createElement('div');
            pop.className = 'site-links-popover';
            pop.style.cssText = `
                position: absolute;
                z-index: 100000;
                max-width: 420px;
                padding: 10px;
                background: #111;
                color: #ddd;
                border-radius: 6px;
                box-shadow: 0 8px 20px rgba(0,0,0,.5);
                line-height: 1.6;
            `;
            const list = document.createElement('div');
            const tests = [];
            const mkGroup = (title, arr, isPublish) => {
                if (!arr || !arr.length) return;
                const h = document.createElement('div');
                h.textContent = title;
                h.style.cssText = 'font-size:12px; color:#E5E7EB; background:#1f2937; text-align:center; padding:4px 8px; border-radius:4px; margin:8px 0 6px; border:1px solid #374151;';
                list.appendChild(h);
                arr.forEach((item, i) => {
                    const row = document.createElement('div');
                    row.style.cssText = 'display:flex; align-items:center; gap:8px; padding:2px 0;';
                    const idx = document.createElement('span');
                    idx.textContent = String(i + 1);
                    idx.style.cssText = 'flex:0 0 auto; width:18px; height:18px; display:inline-flex; align-items:center; justify-content:center; border-radius:50%; background:#334155; color:#E5E7EB; font-size:12px; border:1px solid #475569;';
                    const a = document.createElement('a');
                    a.href = item.url.startsWith('http') ? item.url : ('https://' + item.url.replace(/^\/+/, ''));
                    a.textContent = item.url + (item.note ? `(${item.note})` : '');
                    a.target = '_blank';
                    a.rel = 'noopener noreferrer';
                    a.style.cssText = `
                        flex: 1 1 auto; min-width:0; overflow:hidden; text-overflow:ellipsis; white-space:nowrap;
                        color:${isPublish ? '#F59E0B' : '#60A5FA'}; text-decoration:none;
                    `;
                    const badge = document.createElement('span');
                    badge.textContent = '测速中…';
                    badge.style.cssText = 'flex:0 0 auto; font-size:12px; color:#9CA3AF; border:1px solid #374151; padding:0 6px; border-radius:10px;';
                    row.appendChild(idx);
                    row.appendChild(a);
                    row.appendChild(badge);
                    list.appendChild(row);
                    tests.push(() => new Promise((resolve) => {
                        try {
                            const start = performance.now();
                            GM_xmlhttpRequest({
                                url: a.href,
                                method: 'GET',
                                headers: { 'Cache-Control': 'no-cache' },
                                timeout: 3000,
                                anonymous: true,
                                onload: () => {
                                    const ms = Math.max(1, Math.round(performance.now() - start));
                                    badge.textContent = ms + 'ms';
                                    if (ms < 500) {
                                        badge.style.color = '#10B981';
                                        badge.style.borderColor = '#065F46';
                                    } else if (ms < 1500) {
                                        badge.style.color = '#F59E0B';
                                        badge.style.borderColor = '#92400E';
                                    } else {
                                        badge.style.color = '#EF4444';
                                        badge.style.borderColor = '#7F1D1D';
                                    }
                                    resolve();
                                },
                                onerror: () => {
                                    badge.textContent = 'ERR';
                                    badge.style.color = '#EF4444';
                                    badge.style.borderColor = '#7F1D1D';
                                    resolve();
                                },
                                ontimeout: () => {
                                    badge.textContent = 'ERR';
                                    badge.style.color = '#EF4444';
                                    badge.style.borderColor = '#7F1D1D';
                                    resolve();
                                }
                            });
                        } catch (_) {
                            badge.textContent = 'ERR';
                            badge.style.color = '#EF4444';
                            badge.style.borderColor = '#7F1D1D';
                            resolve();
                        }
                    }));
                });
            };
            mkGroup('站点', links.sites || [], false);
            mkGroup('发布页', links.publish || [], true);
            if (!list.children.length) {
                list.textContent = '暂无收录地址';
                list.style.color = '#888';
            }
            pop.appendChild(list);
            (async () => {
                for (const job of tests) {
                    try { await job(); } catch (_) { /* ignore */ }
                }
            })();
            document.body.appendChild(pop);
            const rect = triggerEl.getBoundingClientRect();
            pop.style.top = `${rect.bottom + window.scrollY + 6}px`;
            pop.style.left = `${Math.min(rect.left + window.scrollX, window.scrollX + window.innerWidth - pop.offsetWidth - 12)}px`;
            const close = (e) => {
                if (!pop.contains(e.target) && e.target !== triggerEl) {
                    pop.remove();
                    document.removeEventListener('mousedown', close, true);
                    window.removeEventListener('scroll', close, true);
                    window.removeEventListener('resize', close, true);
                }
            };
            document.addEventListener('mousedown', close, true);
            window.addEventListener('scroll', close, true);
            window.addEventListener('resize', close, true);
        };

        const addSiteTile = (label, key, gmKey) => {
            const sw = mkSwitch(!!CONFIG.siteEnabled[key], (v) => {
                CONFIG.siteEnabled[key] = v;
                GM_setValue(gmKey, v);
                showNotification('站点规则', `${label}已${v ? '启用' : '禁用'}`);
                addActionButtons();
            });
            const item = document.createElement('div');
            item.className = 'modal-tile';
            const name = document.createElement('span');
            name.className = 'modal-tile-label';
            const m = label.match(/^(.*)\((.*)\)$/);
            if (m) {
                let left = (m[1] || '').trim();
                let inside = (m[2] || '').trim();
                const hasCn = (s) => /[\u4e00-\u9fff]/.test(s);
                if (!hasCn(left) && hasCn(inside)) {
                    [left, inside] = [inside, left];
                }
                name.innerHTML = `${left}<br><span style="font-size:12px;color:#9CA3AF">(${inside})</span>`;
            } else {
                name.textContent = label;
            }
            const right = document.createElement('div');
            right.style.display = 'flex';
            right.style.alignItems = 'center';
            right.style.gap = '6px';
            const linkBtn = document.createElement('button');
            linkBtn.innerHTML = '<span class="site-link-icon"></span>';
            linkBtn.className = 'modal-btn';
            linkBtn.style.cssText = 'padding:2px 6px;font-size:12px;background:transparent;color:#6B7280;border:none;border-radius:4px;';
            linkBtn.title = '地址';
            linkBtn.addEventListener('click', (e) => {
                e.stopPropagation();
                const links = SITES_LINKS[key] || {};
                showLinksPopover(linkBtn, links);
            });
            right.appendChild(linkBtn);
            right.appendChild(sw);
            item.append(name, right);
            siteGrid.appendChild(item);
        };

        addSiteTile('BT4G', 'bt4g', 'site_bt4g');
        addSiteTile('BTDigg', 'btdig', 'site_btdig');
        addSiteTile('BTSOW', 'btsow', 'site_btsow');
        addSiteTile('Nyaa', 'nyaa', 'site_nyaa');
        addSiteTile('動漫花園(DMHY)', 'dmhy', 'site_dmhy');
        addSiteTile('观影', 'gying_family', 'site_gying_family');
        addSiteTile('SeedHub', 'seedhub', 'site_seedhub');
        addSiteTile('梦幻天堂·龙网(LongWangBT)', 'longwangbt', 'site_longwangbt');
        addSiteTile('雨花阁(YuHuaGe)', 'yuhuage', 'site_yuhuage');
        addSiteTile('SOBT', 'sobt', 'site_sobt');
        addSiteTile('CLB(磁力宝)', 'clb', 'site_clb');
        addSiteTile('BT目录(BTMulu)', 'btmulu', 'site_btmulu');
        addSiteTile('ØMagnet(无极磁链)', 'cili_family', 'site_cili_family');
        addSiteTile('磁力帝', 'cilidi', 'site_cilidi');

        secSites.appendChild(siteGrid);


        const topRow = document.createElement('div');
        topRow.className = 'modal-two-col';
        topRow.append(sec115, secFloat);

        panel.append(header, topRow, secButtons, secSites, secAuto);
        overlay.appendChild(panel);
        document.body.appendChild(overlay);
    }

    function handleSeedhubSite() {
        processElements('.seeds a', (linkElement) => {
            const btnContainer = createButtonContainer({
                marginRight: '8px',
                customStyles: {
                    display: 'inline-block',
                    verticalAlign: 'middle'
                }
            });

            if (isAutoFetchEnabledFor('seedhub')) {
                const loadingBtn = createLoadingButton();
                btnContainer.appendChild(loadingBtn);
                processSeedhubMagnetLink(linkElement, btnContainer).then(success => {
                    if (!success) {
                        setupRetryButton(loadingBtn, () =>
                            processSeedhubMagnetLink(linkElement, btnContainer, 2, 6000)
                        );
                    }
                }).catch(error => {
                    console.error('SeedHub处理失败:', error);
                    setButtonError(loadingBtn, '处理失败');
                });
            } else {
                const combinedBtn = createCombinedButtons(async () => {
                    if (linkElement?.href?.startsWith('magnet:')) return linkElement.href;
                    return await fetchSeedhubMagnetFromDetail(linkElement.href);
                });
                btnContainer.appendChild(combinedBtn);
            }
            linkElement.parentNode.insertBefore(btnContainer, linkElement);
            return true;
        });
    }

    async function processSeedhubMagnetLink(linkElement, btnContainer, maxRetries = CONFIG.maxRetries, timeout = CONFIG.defaultTimeout) {
        if (!linkElement?.href) return false;

        return await retryOperation(async (attempt) => {
            const magnetLink = await Promise.race([
                fetchSeedhubMagnetFromDetail(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 handleYuhuageSite() {
        processElements('.search-item .item-title h3 > a[href^="/hash/"]', (titleLink) => {
            const btnContainer = createButtonContainer({ marginRight: '8px' });
            titleLink.parentNode.insertBefore(btnContainer, titleLink);

            if (isAutoFetchEnabledFor('yuhuage')) {
                const loadingBtn = createLoadingButton();
                btnContainer.appendChild(loadingBtn);
                processYuhuageMagnetLink(titleLink, btnContainer).then(success => {
                    if (!success) {
                        setupRetryButton(loadingBtn, () => 
                            processYuhuageMagnetLink(titleLink, btnContainer, 2, 6000)
                        );
                    }
                }).catch(error => {
                    console.error('Yuhuage处理失败:', error);
                    setButtonError(loadingBtn, '处理失败');
                });
            } else {
                const combinedBtn = createCombinedButtons(async () => {
                    return await fetchYuhuageMagnetFromDetail(titleLink.href);
                });
                btnContainer.appendChild(combinedBtn);
            }
            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 rgba(0,0,0,0.14);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' });
            linkElement.parentNode.insertBefore(btnContainer, linkElement);

            if (isAutoFetchEnabledFor('cilimag')) {
                const loadingBtn = createLoadingButton();
                btnContainer.appendChild(loadingBtn);
                processMagnetLink(linkElement, btnContainer).then(success => {
                    if (!success) {
                        setupRetryButton(loadingBtn, () => 
                            processMagnetLink(linkElement, btnContainer, 2, 6000)
                        );
                    }
                }).catch(error => {
                    console.error('CiliMag处理失败:', error);
                    setButtonError(loadingBtn, '处理失败');
                });
            } else {
                const combinedBtn = createCombinedButtons(async () => {
                    return await fetchMagnetFromDetailPage(linkElement.href);
                });
                btnContainer.appendChild(combinedBtn);
            }
            return true;
        }, 'ciliMagProcessed');
        
        processElements('div.input-group.magnet-box', (magnetBox) => {
            const magnetInput = magnetBox.querySelector('input[id="input-magnet"][value^="magnet:"]');
            const addonElement = magnetBox.querySelector('.input-group-addon');
            
            if (!magnetInput?.value.trim() || !addonElement) return false;
            
            if (addonElement.classList.contains('magnet-prefix')) {
                addonElement.style.padding = '2px 5px';
            }
            
            const btnContainer = createButtonContainer({
                marginLeft: '5px',
                customStyles: { display: 'inline-flex', alignItems: 'center' }
            });
            
            const combinedBtn = createCombinedButtons(magnetInput.value.trim());
            btnContainer.appendChild(combinedBtn);
            addonElement.appendChild(btnContainer);
            return true;
        }, 'magnetBoxProcessed');
    }

    async function fetchMagnetFromDetailPage(detailHref) {
        try {
            const html = await fetchWithRetry(detailHref, {
                headers: { 'User-Agent': navigator.userAgent }
            });
            
            const parser = new DOMParser();
            const doc = parser.parseFromString(html, 'text/html');
            
            const magnetInput = doc.querySelector('input[id="input-magnet"][value^="magnet:"]');
            if (magnetInput?.value) {
                return magnetInput.value.trim();
            }
            
            const magnetLink = doc.querySelector('a[href^="magnet:"]');
            if (magnetLink?.href) {
                return magnetLink.href.trim();
            }
            
            const magnetMatch = html.match(/magnet:\?xt=urn:btih:[a-f0-9]+[^"'>]+/i);
            if (magnetMatch?.[0]) {
                return magnetMatch[0].trim();
            }
            
            return null;
        } catch (error) {
            console.error('从详情页获取磁力链失败:', error);
            return null;
        }
    }

    function handleLongwangbtSite() {
        processElements('td.text_left a[href^="show.php?hash="]', (titleLink) => {
            const hashMatch = titleLink.href.match(/hash=([a-f0-9]{40})/i);
            if (!hashMatch) return false;

            const hash = hashMatch[1];
            const titleText = titleLink.textContent.trim();
            const magnetLink = `magnet:?xt=urn:btih:${hash}&dn=${encodeURIComponent(titleText)}`;

            const btnContainer = createButtonContainer({ marginRight: '8px' });
            const combinedBtn = createCombinedButtons(magnetLink);
            btnContainer.appendChild(combinedBtn);
            titleLink.parentNode.insertBefore(btnContainer, titleLink);
            return true;
        });
    }

    initializeScript();
})();

QingJ © 2025

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