跳转到Emby播放

👆👆👆👆👆👆👆在 ✅JavBus✅Javdb✅Sehuatang ✅supjav 高亮emby存在的视频,并提供标注一键跳转功能

// ==UserScript==
// @name         跳转到Emby播放
// @namespace    https://github.com/cgkings
// @version      0.0.8
// @description  👆👆👆👆👆👆👆在 ✅JavBus✅Javdb✅Sehuatang ✅supjav 高亮emby存在的视频,并提供标注一键跳转功能
// @author       cgkings
// @match        *://www.javbus.com/*
// @match        *://javdb*.com/v/*
// @match        *://javdb*.com/search?q=*
// @match        *://www.javdb.com/*
// @match        *://javdb.com/*
// @match        *://supjav.com/*
// @match        *://*.sehuatang.*/*
// @match        *://*.sehuatang.net/*
// @match        https://.*/thread-*
// @match        https://.*/forum.php?mod=viewthread&tid=*
// @grant        GM_xmlhttpRequest
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// @grant        GM_addStyle
// @run-at       document-start
// @license      MIT
// @supportURL   https://github.com/cgkings/cg_tampermonkey_script/issues
// @homepageURL  https://github.com/cgkings/cg_tampermonkey_script
// @icon         
// @license MIT
// ==/UserScript==

(function () {
    'use strict';

    // 全局配置对象
    const Config = {
        get embyAPI() { return GM_getValue('embyAPI', '') },
        get embyBaseUrl() { return GM_getValue('embyBaseUrl', '') },
        get highlightColor() { return GM_getValue('highlightColor', '#52b54b') },
        get maxConcurrentRequests() { return GM_getValue('maxConcurrentRequests', 50) },
        // 添加徽章相关配置
        get badgeColor() { return GM_getValue('badgeColor', '#000') },
        get badgeTextColor() { return GM_getValue('badgeTextColor', '#fff') },
        get badgeSize() { return GM_getValue('badgeSize', 'medium') }, // small, medium, large

        set embyAPI(val) { GM_setValue('embyAPI', val) },
        set embyBaseUrl(val) { GM_setValue('embyBaseUrl', val) },
        set highlightColor(val) { GM_setValue('highlightColor', val) },
        set maxConcurrentRequests(val) { GM_setValue('maxConcurrentRequests', val) },
        // 添加徽章相关配置的setter
        set badgeColor(val) { GM_setValue('badgeColor', val) },
        set badgeTextColor(val) { GM_setValue('badgeTextColor', val) },
        set badgeSize(val) { GM_setValue('badgeSize', val) },

        isValid() { return !!this.embyAPI && !!this.embyBaseUrl }
    };

    // 获取徽章尺寸样式
    function getBadgeSizeStyle() {
        switch (Config.badgeSize) {
            case 'small':
                return { fontSize: '10px', padding: '1px 4px' };
            case 'large':
                return { fontSize: '14px', padding: '3px 7px' };
            case 'medium':
            default:
                return { fontSize: '12px', padding: '2px 5px' };
        }
    }

    // 初始化DOM样式
    const badgeSize = getBadgeSizeStyle();
    GM_addStyle(`
        .emby-jump-settings-panel {position:fixed; top:50%; left:50%; transform:translate(-50%,-50%); background:#fff; border-radius:8px; box-shadow:0 0 20px rgba(0,0,0,0.3); padding:20px; z-index:10000; width:400px; max-width:90%; display:none}
        .emby-jump-settings-header {display:flex; justify-content:space-between; align-items:center; margin-bottom:15px; padding-bottom:10px; border-bottom:1px solid #eee}
        .emby-jump-settings-close {cursor:pointer; font-size:18px; color:#999}
        .emby-jump-settings-field {margin-bottom:15px}
        .emby-jump-settings-field label {display:block; margin-bottom:5px; font-weight:bold}
        .emby-jump-settings-field input, .emby-jump-settings-field select {width:100%; padding:8px; border:1px solid #ddd; border-radius:4px; box-sizing:border-box}
        .emby-jump-settings-buttons {display:flex; justify-content:flex-end; gap:10px; margin-top:15px}
        .emby-jump-settings-buttons button {padding:8px 15px; border:none; border-radius:4px; cursor:pointer}
        .emby-jump-settings-save {background-color:#52b54b; color:white}
        .emby-jump-settings-cancel {background-color:#f0f0f0; color:#333}
        .emby-jump-status-indicator {position:fixed; bottom:20px; right:20px; background:rgba(0,0,0,0.7); color:white; padding:8px 12px; border-radius:4px; font-size:14px; z-index:9999; transition:opacity 0.3s; box-shadow:0 2px 8px rgba(0,0,0,0.2); max-width:300px; display:flex; align-items:center; opacity:0}
        .emby-jump-status-indicator .progress {display:inline-block; margin-left:10px; width:100px; height:6px; background:rgba(255,255,255,0.3); border-radius:3px}
        .emby-jump-status-indicator .progress-bar {height:100%; background:#52b54b; border-radius:3px; transition:width 0.3s}
        .emby-jump-status-indicator.success {background-color:rgba(82,181,75,0.9)}
        .emby-jump-status-indicator.error {background-color:rgba(220,53,69,0.9)}
        .emby-jump-status-indicator .close-btn {margin-left:10px; cursor:pointer; font-size:16px; font-weight:bold}

        /* 徽章样式 - 保留彩虹边框,使用自定义颜色和大小 */
        .emby-badge {position:absolute; top:5px; right:5px; color:${Config.badgeTextColor}; padding:${badgeSize.padding}; font-size:${badgeSize.fontSize}; font-weight:bold; z-index:10; border:2px solid transparent; border-radius:4px; background-origin:border-box; background-clip:padding-box,border-box; background-image:linear-gradient(${Config.badgeColor} 0 0), linear-gradient(50deg,#ff0000,#ff7f00,#ffff00,#00ff00,#0000ff,#4b0082,#8b00ff);}
        .emby-badge:hover {color:#000; background-clip:padding-box,border-box; background-image:linear-gradient(#fff 0 0),linear-gradient(50deg,#ff0000,#ff7f00,#ffff00,#00ff00,#0000ff,#4b0082,#8b00ff);}
        .emby-highlight {outline:4px solid ${Config.highlightColor} !important; position:relative;}

        /* 传统链接样式 */
        .emby-jump-link {background:${Config.highlightColor}; border-radius:3px; padding:3px 6px; margin-top:5px; margin-bottom:3px}
        .emby-jump-link a {color:white; text-decoration:none; display:block;}
    `);

    // 单例状态指示器
    const Status = (() => {
        let el, bar, timeout;
        const debounce = (fn, ms) => {
            let timer;
            return (...args) => {
                clearTimeout(timer);
                timer = setTimeout(() => fn(...args), ms);
            };
        };

        // 创建UI
        const createUI = () => {
            if (el) return;
            el = document.createElement('div');
            el.className = 'emby-jump-status-indicator';
            el.innerHTML = `<span class="status-text">准备中...</span><div class="progress"><div class="progress-bar"></div></div><span class="close-btn">&times;</span>`;
            document.body.appendChild(el);
            bar = el.querySelector('.progress-bar');
            el.querySelector('.close-btn').addEventListener('click', hide);
        };

        // 显示消息
        const show = (msg, type = '') => {
            createUI();
            if (timeout) clearTimeout(timeout);
            el.className = 'emby-jump-status-indicator ' + type;
            el.querySelector('.status-text').textContent = msg;
            el.style.opacity = '1';
        };

        // 隐藏
        const hide = () => {
            if (!el) return;
            el.style.opacity = '0';
            timeout = setTimeout(() => {
                if (el && el.parentNode) el.parentNode.removeChild(el);
                el = bar = null;
            }, 300);
        };

        // 更新进度
        const updateProgress = (current, total) => {
            const percent = Math.min(Math.round((current / total) * 100), 100);
            if (bar) bar.style.width = `${percent}%`;
            show(`查询中: ${current}/${total} (${percent}%)`);
        };

        return {
            show,
            success: (msg, autoHide) => { show(msg, 'success'); if (autoHide) setTimeout(hide, 3000); },
            error: (msg, autoHide) => { show(msg, 'error'); if (autoHide) setTimeout(hide, 5000); },
            updateProgress,
            updateProgressDebounced: debounce(updateProgress, 100),
            hide
        };
    })();

    // 设置面板
    const SettingsUI = {
        show() {
            let panel = document.getElementById('emby-jump-settings-panel');
            if (panel) { panel.style.display = 'block'; return; }

            panel = document.createElement('div');
            panel.id = 'emby-jump-settings-panel';
            panel.className = 'emby-jump-settings-panel';
            panel.innerHTML = `
                <div class="emby-jump-settings-header">
                    <h3 style="margin:0">Emby 设置</h3>
                    <span class="emby-jump-settings-close">&times;</span>
                </div>
                <div class="emby-jump-settings-field">
                    <label for="emby-url">Emby 服务器地址</label>
                    <input type="text" id="emby-url" placeholder="例如: http://192.168.1.100:8096/" value="${Config.embyBaseUrl}">
                    <small style="color:#666">请确保包含http://或https://前缀和最后的斜杠 /</small>
                </div>
                <div class="emby-jump-settings-field">
                    <label for="emby-api">Emby API密钥</label>
                    <input type="text" id="emby-api" placeholder="在Emby设置中获取API密钥" value="${Config.embyAPI}">
                </div>
                <div class="emby-jump-settings-field">
                    <label for="highlight-color">高亮颜色</label>
                    <input type="color" id="highlight-color" value="${Config.highlightColor}">
                </div>
                <div class="emby-jump-settings-field">
                    <label for="max-requests">最大并发请求数</label>
                    <input type="number" id="max-requests" min="1" max="100" value="${Config.maxConcurrentRequests}">
                    <small style="color:#666">因为是本地请求,可以设置较大值</small>
                </div>
                <!-- 添加徽章设置项 -->
                <div class="emby-jump-settings-field">
                    <label for="badge-size">徽章大小</label>
                    <select id="badge-size">
                        <option value="small" ${Config.badgeSize === 'small' ? 'selected' : ''}>小</option>
                        <option value="medium" ${Config.badgeSize === 'medium' ? 'selected' : ''}>中</option>
                        <option value="large" ${Config.badgeSize === 'large' ? 'selected' : ''}>大</option>
                    </select>
                </div>
                <div class="emby-jump-settings-field">
                    <label for="badge-color">徽章背景颜色</label>
                    <input type="color" id="badge-color" value="${Config.badgeColor}">
                    <small style="color:#666">背景颜色将与彩虹边框一起显示</small>
                </div>
                <div class="emby-jump-settings-field">
                    <label for="badge-text-color">徽章文字颜色</label>
                    <input type="color" id="badge-text-color" value="${Config.badgeTextColor}">
                </div>
                <div class="emby-jump-settings-buttons">
                    <button class="emby-jump-settings-cancel">取消</button>
                    <button class="emby-jump-settings-save">保存</button>
                </div>
            `;

            document.body.appendChild(panel);

            // 绑定事件
            const closePanel = () => panel.style.display = 'none';
            panel.querySelector('.emby-jump-settings-close').addEventListener('click', closePanel);
            panel.querySelector('.emby-jump-settings-cancel').addEventListener('click', closePanel);
            panel.querySelector('.emby-jump-settings-save').addEventListener('click', () => {
                const url = document.getElementById('emby-url').value;
                if (!url.match(/^https?:\/\/.+\/$/)) {
                    alert('请输入有效的Emby服务器地址,包含http://或https://前缀和最后的斜杠 /');
                    return;
                }

                Config.embyBaseUrl = url;
                Config.embyAPI = document.getElementById('emby-api').value;
                Config.highlightColor = document.getElementById('highlight-color').value;
                Config.maxConcurrentRequests = parseInt(document.getElementById('max-requests').value, 10);
                Config.badgeSize = document.getElementById('badge-size').value;
                Config.badgeColor = document.getElementById('badge-color').value;
                Config.badgeTextColor = document.getElementById('badge-text-color').value;

                closePanel();
                alert('设置已保存!请刷新页面以应用更改。');
            });

            panel.style.display = 'block';
        }
    };

    // Emby API和请求控制
    class EmbyAPI {
        constructor() {
            this.active = 0;
            this.waiting = [];
            this.total = 0;
            this.completed = 0;
        }

        // 查询单个番号
        async fetchData(code) {
            if (!code) return { Items: [] };

            try {
                const url = `${Config.embyBaseUrl}emby/Users/${Config.embyAPI}/Items?api_key=${Config.embyAPI}&Recursive=true&IncludeItemTypes=Movie&SearchTerm=${encodeURIComponent(code.trim())}&Fields=Name,Id,ServerId`;
                const response = await this.request(url);
                const data = JSON.parse(response.responseText);
                return data;
            } catch (error) {
                console.error(`查询数据出错 ${code}:`, error);
                return { Items: [] };
            }
        }

        // 批量查询
        async batchQuery(codes) {
            if (!codes || codes.length === 0) return [];

            this.total = codes.length;
            this.completed = 0;
            const results = new Array(this.total);

            return new Promise(resolve => {
                const checkComplete = () => {
                    if (this.completed >= this.total && this.active === 0) {
                        const found = results.filter(r => r?.Items?.length > 0).length;
                        Status.success(`查询完成: 找到${found}个匹配项`, false);
                        resolve(results);
                    }
                };

                const processRequest = (index) => {
                    const code = codes[index];
                    this.active++;
                    Status.updateProgressDebounced(this.completed, this.total);

                    this.fetchData(code).then(result => {
                        results[index] = result;
                        this.active--;
                        this.completed++;

                        if (this.waiting.length > 0) {
                            processRequest(this.waiting.shift());
                        }

                        checkComplete();
                    }).catch(() => {
                        results[index] = null;
                        this.active--;
                        this.completed++;

                        if (this.waiting.length > 0) {
                            processRequest(this.waiting.shift());
                        }

                        checkComplete();
                    });
                };

                for (let i = 0; i < this.total; i++) {
                    if (this.active < Config.maxConcurrentRequests) {
                        processRequest(i);
                    } else {
                        this.waiting.push(i);
                    }
                }
            });
        }

        // 通用请求方法
        request(url) {
            return new Promise((resolve, reject) => {
                GM_xmlhttpRequest({
                    method: "GET",
                    url,
                    headers: { accept: "application/json" },
                    timeout: 10000,
                    onload: res => res.status >= 200 && res.status < 300 ? resolve(res) : reject(new Error(`HTTP错误: ${res.status}`)),
                    onerror: () => reject(new Error("请求错误")),
                    ontimeout: () => reject(new Error("请求超时"))
                });
            });
        }

        // 创建Emby链接元素
        createLink(data) {
            if (!data?.Items?.length) return null;

            const item = data.Items[0];
            const embyUrl = `${Config.embyBaseUrl}web/index.html#!/item?id=${item.Id}&serverId=${item.ServerId}`;

            const el = document.createElement('div');
            el.className = 'emby-jump-link';
            el.innerHTML = `<a href="${embyUrl}" target="_blank"><b>跳转到emby👉</b></a>`;

            return el;
        }

        // 创建Emby徽章元素
        createBadge(data) {
            if (!data?.Items?.length) return null;

            const item = data.Items[0];
            const embyUrl = `${Config.embyBaseUrl}web/index.html#!/item?id=${item.Id}&serverId=${item.ServerId}`;

            const el = document.createElement('a');
            el.className = 'emby-badge';
            el.href = embyUrl;
            el.target = '_blank';
            el.textContent = 'Emby';

            return el;
        }
    }

    // 站点处理器基类
    const BaseProcessor = {
        init(api) {
            this.api = api;
            this.processed = new WeakSet();
            return this;
        },

        // 处理列表项 - 徽章样式
        async processItemsWithBadge(items) {
            if (!items?.length) return;

            Status.show(`正在收集番号: 共${items.length}个项目`);

            // 收集番号
            const toProcess = [];
            const codes = [];

            for (const item of items) {
                if (this.processed.has(item)) continue;
                this.processed.add(item);

                const code = this.extractCode(item);
                if (!code) continue;

                // 查找第一个包含图片的元素
                const imgContainer = this.findImgContainer(item);
                if (!imgContainer) continue;

                toProcess.push({ item, code, imgContainer });
                codes.push(code);
            }

            if (codes.length > 0) {
                const results = await this.api.batchQuery(codes);

                // 处理结果 - 批量操作
                const operations = [];

                for (let i = 0; i < results.length; i++) {
                    if (i < toProcess.length && results[i]?.Items?.length > 0) {
                        const { item, imgContainer } = toProcess[i];
                        const badge = this.api.createBadge(results[i]);

                        if (badge) {
                            // 准备添加徽章
                            operations.push(() => {
                                // 确保图片容器是相对定位
                                if (window.getComputedStyle(imgContainer).position === 'static') {
                                    imgContainer.style.position = 'relative';
                                }

                                // 添加高亮样式 - 使用outline避免影响布局
                                item.classList.add('emby-highlight');

                                // 添加徽章
                                imgContainer.appendChild(badge);
                            });
                        }
                    }
                }

                // 批量DOM操作 - 减少重排
                if (operations.length > 0) {
                    requestAnimationFrame(() => {
                        operations.forEach(op => op());
                    });
                }
            }
        },

        // 处理列表项 - 传统链接样式
        async processItemsWithLink(items) {
            if (!items?.length) return;

            Status.show(`正在收集番号: 共${items.length}个项目`);

            // 收集番号
            const toProcess = [];
            const codes = [];

            for (const item of items) {
                if (this.processed.has(item)) continue;
                this.processed.add(item);

                const code = this.extractCode(item);
                const element = this.getElement(item);

                if (code && element) {
                    toProcess.push({ element, code });
                    codes.push(code);
                }
            }

            if (codes.length > 0) {
                const results = await this.api.batchQuery(codes);
                const processedElements = [];

                // 处理结果
                for (let i = 0; i < results.length; i++) {
                    if (i < toProcess.length && results[i]?.Items?.length > 0) {
                        const { element } = toProcess[i];
                        const link = this.api.createLink(results[i]);

                        if (link) {
                            const target = element.parentNode || element;

                            // 高亮条目样式
                            let current = element;
                            const containerClasses = ['item', 'masonry-brick', 'grid-item', 'movie-list', 'post'];
                            while (current && current !== document.body) {
                                for (const className of containerClasses) {
                                    if (current.classList?.contains(className)) {
                                        current.style.cssText += `border:3px solid ${Config.highlightColor};background-color:${Config.highlightColor}22`;
                                        break;
                                    }
                                }
                                current = current.parentElement;
                            }

                            processedElements.push({
                                target,
                                link,
                                position: element.nextSibling
                            });
                        }
                    }
                }

                // 批量插入DOM
                requestAnimationFrame(() => {
                    processedElements.forEach(({ target, link, position }) => {
                        target.insertBefore(link, position);
                    });
                });
            }
        },

        // 主处理函数
        async process() {
            const items = document.querySelectorAll(this.listSelector);

            if (items.length > 0) {
                // 始终使用徽章样式
                await this.processItemsWithBadge(items);
            }

            await this.processDetailPage();
            this.setupObserver();
        },

        // 查找图片容器
        findImgContainer(item) {
            // 不同网站的图片容器选择器
            const imgSelectors = [
                '.img', // supjav
                'a.movie-box', // javbus
                '.cover', // javdb
                'img'
            ];

            for (const selector of imgSelectors) {
                const imgContainer = item.querySelector(selector);
                if (imgContainer) return imgContainer;
            }

            // 找不到合适的容器,尝试找第一个包含图片的元素
            return item.querySelector('a') || item;
        },

        // 观察器设置
        setupObserver() {
            let pending = [];
            let timer = null;

            const processMutations = () => {
                const newElements = [];

                for (const mutation of pending) {
                    if (mutation.type === 'childList') {
                        for (const node of mutation.addedNodes) {
                            if (node.nodeType !== 1) continue;

                            if (node.matches?.(this.listSelector)) {
                                newElements.push(node);
                            }

                            if (node.querySelectorAll) {
                                node.querySelectorAll(this.listSelector).forEach(el => {
                                    newElements.push(el);
                                });
                            }
                        }
                    }
                }

                if (newElements.length > 0) {
                    // 始终使用徽章样式
                    this.processItemsWithBadge(newElements);
                }

                pending = [];
                timer = null;
            };

            new MutationObserver(mutations => {
                pending.push(...mutations);
                if (!timer) timer = setTimeout(processMutations, 300);
            }).observe(document.body, { childList: true, subtree: true });
        }
    };

    // 站点处理器
    const Processors = {
        javbus: Object.assign(Object.create(BaseProcessor), {
            listSelector: '.item.masonry-brick, #waterfall .item',
            extractCode: item => item.querySelector('.item date')?.textContent?.trim(),
            getElement: item => item.querySelector('.item date'),

            async processDetailPage() {
                // 防止重复处理
                if (document.querySelector('.emby-jump-link, .emby-badge')) return;

                const infoElement = document.querySelector('.col-md-3.info p');
                if (!infoElement) return;

                const spans = infoElement.querySelectorAll('span');
                if (spans.length > 1) {
                    const code = spans[1].textContent?.trim();
                    if (code) {
                        Status.show('查询中...');
                        const data = await this.api.fetchData(code);
                        if (data.Items?.length > 0) {
                            const link = this.api.createLink(data);
                            if (link) {
                                spans[1].parentNode.insertBefore(link, spans[1].nextSibling);
                                Status.success('找到匹配项', false);
                            }
                        } else {
                            Status.error('未找到匹配项', false);
                        }
                    }
                }
            }
        }),

        javdb: Object.assign(Object.create(BaseProcessor), {
            listSelector: '.movie-list .item, .grid-item',
            extractCode: item => item.querySelector('.video-title strong')?.textContent?.trim(),
            getElement: item => item.querySelector('.video-title strong'),

            async processDetailPage() {
                // 防止重复处理
                if (document.querySelector('.emby-jump-link, .emby-badge')) return;

                const detailElement = document.querySelector('body > section > div > div.video-detail > h2 > strong') ||
                    document.querySelector('.video-detail h2 strong');

                if (!detailElement) return;

                const code = detailElement.textContent.trim().split(' ')[0];
                if (code) {
                    Status.show('查询中...');
                    const data = await this.api.fetchData(code);
                    if (data.Items?.length > 0) {
                        const link = this.api.createLink(data);
                        if (link) {
                            detailElement.parentNode.insertBefore(link, detailElement.nextSibling);
                            Status.success('找到匹配项', false);
                        }
                    } else {
                        Status.error('未找到匹配项', false);
                    }
                }
            }
        }),

        supjav: Object.assign(Object.create(BaseProcessor), {
            listSelector: '.post',
            extractCode: function (item) {
                // 从标题中提取番号
                const title = item.querySelector('h3 a')?.textContent?.trim();
                if (!title) return null;

                // 匹配番号格式
                const match = title.match(/([a-zA-Z0-9]+-\d+)/i);
                return match ? match[1] : null;
            },
            getElement: function (item) {
                return item.querySelector('h3 a');
            },

            async processDetailPage() {
                // 防止重复处理 - 限制在主内容区域
                if (document.querySelector('.video-wrap .emby-jump-link, .video-wrap .emby-badge')) return;

                // 在详情页面查找标题 - 只处理主标题,不处理相关推荐
                const titleElement = document.querySelector('.video-wrap .archive-title h1');
                if (!titleElement) return;

                const title = titleElement.textContent.trim();
                const match = title.match(/([a-zA-Z0-9]+-\d+)/i);

                if (!match) return;

                const code = match[1];
                if (code) {
                    Status.show('查询中...');
                    const data = await this.api.fetchData(code);
                    if (data.Items?.length > 0) {
                        const link = this.api.createLink(data);
                        if (link) {
                            titleElement.parentNode.insertBefore(link, titleElement.nextSibling);
                            Status.success('找到匹配项', false);
                        }
                    } else {
                        Status.error('未找到匹配项', false);
                    }
                }
            }
        }),

        sehuatang: Object.assign(Object.create(BaseProcessor), {
            listSelector: '',

            async process() {
                // 防止重复处理
                if (document.querySelector('.emby-jump-link, .emby-badge')) return;

                const title = document.title.trim();
                const codes = this.extractCodes(title);

                if (codes.length > 0) {
                    Status.show(`找到${codes.length}个可能的番号,开始查询...`);

                    const results = await this.api.batchQuery(codes);
                    let foundAny = false;

                    for (const data of results) {
                        if (data?.Items?.length > 0) {
                            const container = document.querySelector('#thread_subject') ||
                                document.querySelector('h1.ts') ||
                                document.querySelector('h1');

                            if (container) {
                                const link = this.api.createLink(data);
                                if (link) {
                                    container.parentNode.insertBefore(link, container.nextSibling);
                                    foundAny = true;
                                }
                            }
                        }
                    }

                    if (foundAny) Status.success('找到匹配项', false);
                    else Status.error('未找到匹配项', false);
                }
            },

            extractCodes(title) {
                if (!title) return [];

                const patterns = [
                    /([a-zA-Z]{2,15})[-\s]?(\d{2,15})/i,
                    /FC2[-\s]?PPV[-\s]?(\d{6,7})/i
                ];

                const results = [];
                for (const pattern of patterns) {
                    const match = title.match(pattern);
                    if (match) {
                        if (match[2]) results.push(`${match[1]}-${match[2]}`);
                        else if (match[1]) results.push(match[0]);
                    }
                }

                return results;
            }
        })
    };

    // 站点检测
    function detectSite() {
        if (location.hostname.includes('javbus') || document.querySelector('footer')?.textContent?.includes('JavBus')) {
            return 'javbus';
        }

        if (location.hostname.includes('javdb') || document.querySelector('#footer')?.textContent?.includes('javdb')) {
            return 'javdb';
        }

        if (location.hostname.includes('supjav') || document.title.includes('SupJav')) {
            return 'supjav';
        }

        if (location.hostname.includes('sehuatang') || document.querySelector('#flk')?.textContent?.includes('色花堂')) {
            return 'sehuatang';
        }

        return null;
    }

    // 主函数
    async function main() {
        console.log('Emby跳转脚本启动 (极简版)');

        // 注册菜单命令
        GM_registerMenuCommand("Emby 设置", () => SettingsUI.show());

        // 检查API配置
        if (!Config.isValid()) {
            Status.error('配置无效', true);
            setTimeout(() => {
                alert('请先设置您的Emby服务器地址和API密钥');
                SettingsUI.show();
            }, 500);
            return;
        }

        Status.show('正在初始化...');

        // 检测当前站点
        const site = detectSite();
        if (!site) {
            Status.error('未识别到支持的站点', false);
            return;
        }

        Status.show(`检测到站点: ${site},开始处理...`);

        // 创建并执行站点处理器
        const processor = Processors[site].init(new EmbyAPI());
        if (processor) await processor.process();
    }

    // 页面加载完成后启动
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', main);
    } else {
        main();
    }
})();