全能搜索引擎切换助手

在搜索结果页面添加智能搜索引擎切换功能,支持多种布局和自定义设置

// ==UserScript==
// @name         全能搜索引擎切换助手
// @namespace    http://tampermonkey.net/
// @version      1.8
// @description  在搜索结果页面添加智能搜索引擎切换功能,支持多种布局和自定义设置
// @match        *://www.google.com/search*
// @match        *://www.google.co*/search*
// @match        *://www.google.com.*/search*
// @match        *://google.*/search*
// @match        https://www.google.com.hk/search*
// @match        www.google.com.hk/search*
// @match        *://www.bing.com/search*
// @match        *://cn.bing.com/search*
// @match        *://www.baidu.com/s*
// @match        *://www.sogou.com/web*
// @match        *://www.so.com/s*
// @match        *://duckduckgo.com/*
// @match        *://newtab/
// @match        *://www.google.com/
// @match        *://www.google.co*/
// @match        *://www.google.com.*/
// @match        about:blank
// @match        chrome://newtab/
// @author       Yuze
// @copyright    2025, Yuze (https://gf.qytechs.cn/users/Yuze Guitar)
// @license      MIT
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_addStyle
// @grant        GM_registerMenuCommand
// @run-at       document-end
// ==/UserScript==

(function() {
    'use strict';

    // 记录调试信息
    const DEBUG = true;
    function log(...args) {
        if (DEBUG) {
            console.log('[搜索引擎切换助手]', ...args);
        }
    }

    // 全局错误处理增强
    let errorRecoveryAttempts = 0;
    const MAX_RECOVERY_ATTEMPTS = 3;
    const ERROR_RECOVERY_DELAY = 1000; // 毫秒

    // 错误处理升级
    function handleError(error, context) {
        console.error(`[搜索引擎切换助手] 错误 (${context}):`, error);
        
        // 记录更详细的错误信息
        if (error && error.stack) {
            console.error(`[搜索引擎切换助手] 错误堆栈:`, error.stack);
        }
        
        // 如果是特定类型的错误,尝试恢复
        if (context === '创建搜索引擎切换器UI') {
            // 尝试创建一个最小化版本的UI
            try {
                setTimeout(() => {
                    createMinimalUI();
                }, 1000);
            } catch (e) {
                console.error('[搜索引擎切换助手] 无法创建备用UI:', e);
            }
        }
    }

    // 创建最小化版本的UI作为备选
    function createMinimalUI() {
        try {
            const minimalUI = document.createElement('div');
            minimalUI.id = 'search-engine-switcher-minimal';
            minimalUI.style.position = 'fixed';
            minimalUI.style.top = '10px';
            minimalUI.style.left = '10px';
            minimalUI.style.zIndex = '9999';
            minimalUI.style.background = '#fff';
            minimalUI.style.padding = '5px 10px';
            minimalUI.style.borderRadius = '4px';
            minimalUI.style.boxShadow = '0 2px 5px rgba(0,0,0,0.2)';
            minimalUI.style.fontSize = '14px';
            minimalUI.style.cursor = 'pointer';
            minimalUI.textContent = '搜索引擎切换';
            
            minimalUI.addEventListener('click', function() {
                const { currentEngine, query } = getCurrentEngineAndQuery();
                if (currentEngine && query) {
                    // 显示简单的引擎选择菜单
                    showSimpleEngineMenu(minimalUI, currentEngine, query);
                }
            });
            
            document.body.appendChild(minimalUI);
        } catch (error) {
            console.error('[搜索引擎切换助手] 创建最小化UI失败:', error);
        }
    }

    // 显示简单的引擎选择菜单
    function showSimpleEngineMenu(anchor, currentEngine, query) {
        try {
            // 移除现有菜单
            const existingMenu = document.getElementById('simple-engine-menu');
            if (existingMenu) {
                existingMenu.remove();
            }
            
            // 创建菜单
            const menu = document.createElement('div');
            menu.id = 'simple-engine-menu';
            menu.style.position = 'absolute';
            menu.style.top = '30px';
            menu.style.left = '0';
            menu.style.background = '#fff';
            menu.style.boxShadow = '0 2px 10px rgba(0,0,0,0.2)';
            menu.style.borderRadius = '4px';
            menu.style.padding = '5px 0';
            menu.style.zIndex = '10000';
            
            // 添加引擎选项
            for (const engine of defaultEngines) {
                if (engine.name === currentEngine.name) continue;
                
                const item = document.createElement('div');
                item.style.padding = '8px 15px';
                item.style.cursor = 'pointer';
                item.style.hoverBackground = '#f5f5f5';
                item.textContent = engine.name;
                
                item.addEventListener('click', function() {
                    const url = engine.url.replace('{query}', encodeURIComponent(query));
                    window.location.href = url;
                });
                
                item.addEventListener('mouseover', function() {
                    this.style.backgroundColor = '#f5f5f5';
                });
                
                item.addEventListener('mouseout', function() {
                    this.style.backgroundColor = '';
                });
                
                menu.appendChild(item);
            }
            
            // 添加到页面
            anchor.appendChild(menu);
            
            // 点击外部关闭菜单
            document.addEventListener('click', function closeMenu(e) {
                if (!menu.contains(e.target) && e.target !== anchor) {
                    menu.remove();
                    document.removeEventListener('click', closeMenu);
                }
            });
        } catch (error) {
            console.error('[搜索引擎切换助手] 显示简单菜单失败:', error);
        }
    }

    // 检测操作系统
    const isMac = navigator.platform.toUpperCase().indexOf('MAC') >= 0;
    const modifierKey = isMac ? '⌘' : 'Alt+';
    const modifierKeyCode = isMac ? 'metaKey' : 'altKey';

    // 定义搜索引擎
    const defaultEngines = [
        {
            name: 'Google',
            icon: 'https://www.google.com/favicon.ico',
            url: 'https://www.google.com/search?q={query}',
            matchPattern: 'google\\.[^/]+/search',
            queryParam: 'q',
            shortcut: 'G'
        },
        {
            name: 'Bing',
            icon: 'https://www.bing.com/favicon.ico',
            url: 'https://www.bing.com/search?q={query}',
            matchPattern: 'bing\\.com/search',
            queryParam: 'q',
            shortcut: 'B'
        },
        {
            name: '百度',
            icon: 'https://www.baidu.com/favicon.ico',
            url: 'https://www.baidu.com/s?wd={query}',
            matchPattern: 'baidu\\.com/s',
            queryParam: 'wd',
            shortcut: 'D'
        },
        {
            name: 'DuckDuckGo',
            // 内置图标数据,确保在任何环境下都能显示
            icon: '',
            url: 'https://duckduckgo.com/?q={query}',
            matchPattern: 'duckduckgo\\.com',
            queryParam: 'q',
            shortcut: 'K'
        }
    ];

    // 在defaultEngines数组后添加图标缓存系统
    const iconCache = {
        cache: {},
        
        // 获取图标,优先从缓存获取
        async getIcon(url) {
            try {
                // 如果是数据URL,直接返回
                if (!url || url.startsWith('data:')) {
                    return url;
                }
                
                // 检查缓存
                if (this.cache[url]) {
                    return this.cache[url];
                }
                
                // 尝试获取图标
                try {
                    // 简单地返回原始URL,不进行转换
                    // 在实际应用中,这里可以添加图标转换为dataURL的逻辑
                    this.cache[url] = url;
                    return url;
                } catch (error) {
                    console.warn(`无法获取图标: ${url}`, error);
                    return url;
                }
            } catch (e) {
                console.warn('图标缓存系统错误', e);
                return url;
            }
        },
        
        // 从字符串生成哈希值的辅助函数
        hashString(str) {
            let hash = 0;
            for (let i = 0; i < str.length; i++) {
                const char = str.charCodeAt(i);
                hash = ((hash << 5) - hash) + char;
                hash = hash & hash; // 转换为32位整数
            }
            return Math.abs(hash).toString(16);
        }
    };

    // 在defaultEngines数组后添加快捷搜索前缀配置
    const quickSearchPrefixes = [
        {
            prefix: 'gh ',
            name: 'GitHub',
            url: 'https://github.com/search?q={query}&type=repositories',
            icon: 'https://github.com/favicon.ico',
            description: '在GitHub上搜索仓库'
        },
        {
            prefix: 'so ',
            name: 'Stack Overflow',
            url: 'https://stackoverflow.com/search?q={query}',
            icon: 'https://stackoverflow.com/favicon.ico',
            description: '在Stack Overflow上搜索问题'
        },
        {
            prefix: 'npm ',
            name: 'NPM包',
            url: 'https://www.npmjs.com/search?q={query}',
            icon: 'https://static.npmjs.com/favicon.ico',
            description: '搜索NPM包'
        },
        {
            prefix: 'yt ',
            name: 'YouTube',
            url: 'https://youtube.com/results?search_query={query}',
            icon: 'https://www.youtube.com/favicon.ico',
            description: '在YouTube上搜索视频'
        },
        {
            prefix: 'bili ',
            name: '哔哩哔哩',
            url: 'https://www.bilibili.com/search?keyword={query}',
            icon: 'https://www.bilibili.com/favicon.ico',
            description: '在B站搜索视频'
        },
        {
            prefix: 'zh ',
            name: '知乎',
            url: 'https://www.zhihu.com/search?type=content&q={query}',
            icon: 'https://static.zhihu.com/heifetz/favicon.ico',
            description: '在知乎搜索'
        }
    ];

    // 添加CSS样式
    function addStyles() {
        try {
            GM_addStyle(`
                /* 主容器 - 增加拖拽支持 */
                #search-engine-switcher {
                    position: fixed;
                    top: 120px;
                    left: 10px;
                    z-index: 9999;
                    background: #fff;
                    border-radius: 8px;
                    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);
                    padding: 10px;
                    font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
                    font-size: 14px;
                    display: flex;
                    flex-direction: column;
                    gap: 12px;
                    transition: background 0.3s ease, box-shadow 0.3s ease;
                    min-width: 120px;
                    max-width: 180px;
                    cursor: move; /* 指示可拖动 */
                    touch-action: none; /* 触摸设备支持 */
                    user-select: none; /* 防止拖动时选中文本 */
                }

                /* 深色模式 */
                @media (prefers-color-scheme: dark) {
                    #search-engine-switcher {
                        background: #333;
                        color: #fff;
                        box-shadow: 0 2px 8px rgba(0, 0, 0, 0.5);
                    }
                    #search-engine-switcher .engine-btn {
                        color: #eee;
                    }
                    #search-engine-switcher .engine-btn:hover {
                        background: #444;
                    }
                    #search-engine-switcher .collapse-btn {
                        color: #ccc;
                    }
                    #search-engine-switcher.collapsed {
                        background: rgba(51, 51, 51, 0.9);
                    }
                }

                /* 折叠状态 - 优化折叠后的外观 */
                #search-engine-switcher.collapsed {
                    padding: 8px 10px;
                    min-width: unset;
                    max-width: unset;
                    width: auto;
                }

                #search-engine-switcher.collapsed .engine-container {
                    display: none;
                }

                #search-engine-switcher.collapsed .switcher-header {
                    margin: 0;
                    padding: 0;
                }

                #search-engine-switcher.collapsed .collapse-btn {
                    transform: rotate(180deg);
                    margin-left: 5px;
                }

                #search-engine-switcher.collapsed .switcher-title {
                    margin: 0;
                    font-size: 12px;
                    white-space: nowrap;
                }

                /* 头部样式 */
                .switcher-header {
                    display: flex;
                    justify-content: space-between;
                    align-items: center;
                    margin-bottom: 5px;
                }

                .switcher-title {
                    font-weight: bold;
                    font-size: 14px;
                    margin: 0;
                    user-select: none;
                }

                .settings-btn, .collapse-btn {
                    background: none;
                    border: none;
                    cursor: pointer;
                    padding: 0;
                    display: flex;
                    align-items: center;
                    justify-content: center;
                    width: 20px;
                    height: 20px;
                    opacity: 0.7;
                    transition: opacity 0.2s;
                }

                .settings-btn:hover, .collapse-btn:hover {
                    opacity: 1;
                }

                .header-actions {
                    display: flex;
                    gap: 8px;
                }

                /* 搜索引擎按钮容器 */
                .engine-container {
                    display: flex;
                    flex-direction: column;
                    gap: 8px;
                }

                /* 搜索引擎按钮 */
                .engine-btn {
                    display: flex;
                    align-items: center;
                    gap: 8px;
                    padding: 6px 8px;
                    border-radius: 6px;
                    border: none;
                    background: none;
                    cursor: pointer;
                    color: #333;
                    font-size: 14px;
                    text-align: left;
                    transition: background 0.2s;
                }

                .engine-btn:hover {
                    background: #f0f0f0;
                }

                .engine-icon {
                    width: 16px;
                    height: 16px;
                    object-fit: contain;
                }

                .shortcut {
                    margin-left: auto;
                    padding: 2px 4px;
                    border-radius: 3px;
                    background: #f0f0f0;
                    color: #666;
                    font-size: 10px;
                }

                @media (prefers-color-scheme: dark) {
                    .shortcut {
                        background: #444;
                        color: #ccc;
                    }
                }

                /* 设置面板样式 */
                .settings-overlay {
                    position: fixed;
                    top: 0;
                    left: 0;
                    right: 0;
                    bottom: 0;
                    background: rgba(0, 0, 0, 0.5);
                    display: flex;
                    align-items: center;
                    justify-content: center;
                    z-index: 10000;
                }

                .settings-container {
                    background: #fff;
                    border-radius: 8px;
                    padding: 20px;
                    width: 90%;
                    max-width: 400px;
                    max-height: 80vh;
                    overflow-y: auto;
                }

                @media (prefers-color-scheme: dark) {
                    .settings-container {
                        background: #333;
                        color: #fff;
                    }
                }

                .settings-header {
                    font-weight: bold;
                    font-size: 18px;
                    margin-bottom: 15px;
                }

                .settings-section {
                    margin-bottom: 20px;
                }

                .section-title {
                    font-weight: bold;
                    margin-bottom: 10px;
                }

                .checkbox-group {
                    display: flex;
                    flex-direction: column;
                    gap: 10px;
                }

                .settings-actions {
                    display: flex;
                    justify-content: flex-end;
                    gap: 10px;
                    margin-top: 20px;
                }

                .settings-button {
                    padding: 8px 16px;
                    border-radius: 4px;
                    border: none;
                    cursor: pointer;
                }

                .cancel-button {
                    background: #f0f0f0;
                    color: #333;
                }

                .save-button {
                    background: #4285f4;
                    color: white;
                }

                @media (prefers-color-scheme: dark) {
                    .cancel-button {
                        background: #555;
                        color: #eee;
                    }
                }

                /* Mac选项 */
                .mac-options {
                    display: flex;
                    gap: 15px;
                    margin-top: 10px;
                }

                .mac-option {
                    display: flex;
                    align-items: center;
                    gap: 5px;
                }

                /* 拖动手柄样式 */
                .drag-handle {
                    display: flex;
                    align-items: center;
                    justify-content: center;
                    width: 18px;
                    height: 18px;
                    margin-right: 5px;
                    opacity: 0.5;
                    cursor: move;
                    transition: opacity 0.2s;
                }

                .drag-handle:hover {
                    opacity: 0.8;
                }

                .drag-handle svg {
                    width: 14px;
                    height: 14px;
                }

                /* 快捷搜索提示样式 */
                #quick-search-hint {
                    font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
                    border: 1px solid #e1e4e8;
                    color: #24292e;
                }
                
                @media (prefers-color-scheme: dark) {
                    #quick-search-hint {
                        background-color: #333;
                        color: #fff;
                        border-color: #444;
                    }
                    #quick-search-hint div[style*="color: #666"] {
                        color: #aaa !important;
                    }
                }
            `);
            log('样式已添加');
        } catch (error) {
            handleError(error, '添加样式');
        }
    }

    // 获取启用的搜索引擎
    function getEnabledEngines() {
        try {
            const settings = loadSettings();
            const enabledEngineNames = settings.enabledEngines || defaultEngines.map(e => e.name);

            return defaultEngines.filter(engine =>
                enabledEngineNames.includes(engine.name)
            );
        } catch (error) {
            handleError(error, '获取启用的搜索引擎');
            return defaultEngines;
        }
    }

    // 加载设置
    function loadSettings() {
        try {
            const defaultSettings = {
                enabledEngines: defaultEngines.map(engine => engine.name),
                position: { top: 120, left: 10 },
                autoPosition: true,
                collapsed: false,
                keyboardShortcuts: true,
                macModifierType: 'command',
                quickSearchEnabled: true  // 添加快捷搜索开关
            };
            
            // 尝试从GM_getValue加载
            let savedSettings = null;
            try {
                savedSettings = GM_getValue('searchEngineSettings');
                if (savedSettings) {
                    log('从GM_getValue加载设置成功');
                }
            } catch (e) {
                log('从GM_getValue加载设置失败', e);
            }
            
            // 如果GM_getValue失败,尝试从localStorage加载
            if (!savedSettings) {
                try {
                    const localStorageSettings = localStorage.getItem('searchEngineSwitcherSettings');
                    if (localStorageSettings) {
                        savedSettings = localStorageSettings;
                        log('从localStorage加载设置成功');
                    }
                } catch (e) {
                    log('从localStorage加载设置失败', e);
                }
            }
            
            // 如果没有找到保存的设置,使用默认设置
            if (!savedSettings) {
                log('未找到保存的设置,使用默认设置');
                return defaultSettings;
            }
            
            // 解析设置
            try {
                let parsedSettings;
                if (typeof savedSettings === 'string') {
                    parsedSettings = JSON.parse(savedSettings);
                } else {
                    parsedSettings = savedSettings;
                }
                
                // 确保position是有效的
                if (!parsedSettings.position || typeof parsedSettings.position !== 'object') {
                    parsedSettings.position = defaultSettings.position;
                }
                
                // 合并默认设置和保存的设置
                const mergedSettings = {
                    ...defaultSettings,
                    ...parsedSettings
                };
                
                log('成功加载设置:', mergedSettings);
                return mergedSettings;
            } catch (e) {
                log('解析设置失败,使用默认设置', e);
                return defaultSettings;
            }
        } catch (error) {
            handleError(error, '加载设置');
            return {
                enabledEngines: defaultEngines.map(engine => engine.name),
                position: { top: 120, left: 10 },
                autoPosition: true,
                collapsed: false,
                keyboardShortcuts: true,
                macModifierType: 'command',
                quickSearchEnabled: true
            };
        }
    }

    // 保存设置
    function saveSettings(settings) {
        try {
            // 确保position是有效的
            if (!settings.position || typeof settings.position !== 'object') {
                settings.position = { top: 120, left: 10 };
            }
            
            // 确保top和left是数字
            settings.position.top = parseInt(settings.position.top) || 120;
            settings.position.left = parseInt(settings.position.left) || 10;
            
            // 记录保存的设置
            log('保存设置:', settings);
            
            // 尝试使用GM_setValue保存
            let gmSaveSuccess = false;
            try {
                GM_setValue('searchEngineSettings', settings);
                gmSaveSuccess = true;
                log('使用GM_setValue保存设置成功');
            } catch (e) {
                log('使用GM_setValue保存设置失败', e);
            }
            
            // 同时保存到localStorage作为备份
            try {
                localStorage.setItem('searchEngineSwitcherSettings', JSON.stringify(settings));
                log('使用localStorage备份设置成功');
            } catch (e) {
                log('使用localStorage备份设置失败', e);
            }
            
            return gmSaveSuccess;
        } catch (error) {
            handleError(error, '保存设置');
            return false;
        }
    }

    // 获取当前搜索引擎和查询词
    function getCurrentEngineAndQuery() {
        try {
            const url = window.location.href;

            let currentEngine = null;
            let query = '';

            // 遍历搜索引擎进行匹配
            for (const engine of defaultEngines) {
                // 创建正则表达式
                const regexPattern = new RegExp(engine.matchPattern, 'i');

                if (regexPattern.test(url)) {
                    log(`匹配到搜索引擎: ${engine.name}`);
                    currentEngine = engine;

                    // 提取查询词
                    const urlObj = new URL(url);
                    query = urlObj.searchParams.get(engine.queryParam) || '';

                    // 特殊处理DuckDuckGo
                    if (engine.name === 'DuckDuckGo' && !query) {
                        // 检查URL中是否包含q=参数(可能在hash中)
                        if (url.includes('q=')) {
                            const match = url.match(/[?&#]q=([^&#]*)/);
                            if (match && match[1]) {
                                query = decodeURIComponent(match[1]);
                            }
                        }
                    }

                    break;
                }
            }

            if (currentEngine) {
                log(`当前引擎: ${currentEngine.name}, 查询词: ${query}`);
            } else {
                log('未匹配到搜索引擎');
            }

            return { currentEngine, query };
        } catch (error) {
            handleError(error, '获取当前搜索引擎和查询词');
            return { currentEngine: null, query: '' };
        }
    }

    // 切换到指定搜索引擎
    function switchToEngine(engine, query) {
        try {
            if (!query) {
                log('没有查询词,取消切换');
                return;
            }

            // 将查询参数编码并替换到URL中
            const encodedQuery = encodeURIComponent(query);
            const targetUrl = engine.url.replace('{query}', encodedQuery);

            log(`切换到 ${engine.name}, URL: ${targetUrl}`);

            // 导航到新的URL
            window.location.href = targetUrl;
        } catch (error) {
            handleError(error, '切换搜索引擎');
        }
    }

    // 自动定位函数 - 新增
    function calculateOptimalPosition() {
        try {
            const { currentEngine } = getCurrentEngineAndQuery();
            if (!currentEngine) return { top: 120, left: 10 };

            let top = 120, left = 10;
            const windowWidth = window.innerWidth;
            const windowHeight = window.innerHeight;

            // 针对不同搜索引擎的特殊定位逻辑
            if (/google/.test(currentEngine.name.toLowerCase())) {
                // Google搜索页面,放在搜索框下方
                const searchBox = document.querySelector('form[role="search"]');
                if (searchBox) {
                    const rect = searchBox.getBoundingClientRect();
                    top = rect.bottom + 20;
                }
            } else if (/baidu/.test(currentEngine.name.toLowerCase())) {
                // 百度搜索页面,放在右侧
                left = windowWidth - 200;
                top = 150;
            } else if (/bing/.test(currentEngine.name.toLowerCase())) {
                // Bing搜索页面,优化位置
                const searchBox = document.querySelector('.b_searchbox');
                if (searchBox) {
                    const rect = searchBox.getBoundingClientRect();
                    top = rect.bottom + 15;
                }
            } else if (/duckduckgo/.test(currentEngine.name.toLowerCase())) {
                // DuckDuckGo搜索页面,放在左上角
                top = 80;
            }

            // 确保没有超出窗口边界
            top = Math.max(10, Math.min(windowHeight - 200, top));
            left = Math.max(10, Math.min(windowWidth - 200, left));

            return { top, left };
        } catch (error) {
            handleError(error, '计算最佳位置');
            return { top: 120, left: 10 };
        }
    }

    // 创建搜索引擎切换器UI
    function createSwitcherUI() {
        try {
            // 检查当前页面是否为搜索结果页
            const { currentEngine, query } = getCurrentEngineAndQuery();
            if (!currentEngine || !query) {
                log('未检测到搜索引擎或查询词,不创建UI');
                return;
            }

            // 检查是否已存在搜索引擎切换器,如果存在则移除
            let switcher = document.getElementById('search-engine-switcher');
            if (switcher) {
                log('搜索引擎切换器已存在,移除旧的');
                switcher.remove();
            }

            // 获取设置
            const settings = loadSettings();

            // 获取位置信息
            let position = settings.position;
            if (settings.autoPosition) {
                position = calculateOptimalPosition();
            }

            // 获取启用的搜索引擎
            const enabledEngines = getEnabledEngines();

            // 创建搜索引擎切换器容器
            switcher = document.createElement('div');
            switcher.id = 'search-engine-switcher';
            if (settings.collapsed) {
                switcher.classList.add('collapsed');
            }

            // 设置位置
            switcher.style.top = `${position.top}px`;
            switcher.style.left = `${position.left}px`;

            // 创建标题栏
            const header = document.createElement('div');
            header.className = 'switcher-header';

            // 添加拖动手柄
            const dragHandle = document.createElement('div');
            dragHandle.className = 'drag-handle';
            dragHandle.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M5 9h14M5 15h14"></path></svg>';
            dragHandle.title = '拖动';
            header.appendChild(dragHandle);

            // 创建标题
            const title = document.createElement('div');
            title.className = 'switcher-title';
            title.textContent = '搜索引擎切换';
            header.appendChild(title);

            // 创建按钮容器
            const headerActions = document.createElement('div');
            headerActions.className = 'header-actions';

            // 创建设置按钮
            const settingsButton = document.createElement('button');
            settingsButton.className = 'settings-btn';
            settingsButton.title = '设置';
            settingsButton.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="12" cy="12" r="3"></circle><path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"></path></svg>';
            settingsButton.addEventListener('click', function(e) {
                e.stopPropagation();
                showSettingsPanel();
            });
            headerActions.appendChild(settingsButton);

            // 创建折叠/展开按钮
            const collapseButton = document.createElement('button');
            collapseButton.className = 'collapse-btn';
            collapseButton.title = settings.collapsed ? '展开' : '折叠';
            collapseButton.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="18 15 12 9 6 15"></polyline></svg>';

            collapseButton.addEventListener('click', function(e) {
                e.stopPropagation();
                // 切换折叠状态
                switcher.classList.toggle('collapsed');
                // 更新设置
                const newSettings = {...settings, collapsed: switcher.classList.contains('collapsed')};
                saveSettings(newSettings);
                // 更新按钮提示
                this.title = newSettings.collapsed ? '展开' : '折叠';
            });
            headerActions.appendChild(collapseButton);

            header.appendChild(headerActions);
            switcher.appendChild(header);

            // 创建搜索引擎按钮容器
            const engineContainer = document.createElement('div');
            engineContainer.className = 'engine-container';

            // 创建每个搜索引擎按钮
            for (const engine of enabledEngines) {
                // 如果是当前搜索引擎,则跳过
                if (currentEngine && engine.name === currentEngine.name) {
                    continue;
                }

                // 创建按钮
                const engineBtn = document.createElement('button');
                engineBtn.className = 'engine-btn';
                engineBtn.title = `在${engine.name}中搜索 "${query}"`;
                engineBtn.addEventListener('click', function() {
                    switchToEngine(engine, query);
                });

                // 创建图标占位符
                const iconPlaceholder = document.createElement('div');
                iconPlaceholder.className = 'engine-icon-placeholder';
                iconPlaceholder.style.width = '16px';
                iconPlaceholder.style.height = '16px';
                iconPlaceholder.style.display = 'flex';
                iconPlaceholder.style.alignItems = 'center';
                iconPlaceholder.style.justifyContent = 'center';
                iconPlaceholder.style.backgroundColor = '#f0f0f0';
                iconPlaceholder.style.borderRadius = '3px';
                iconPlaceholder.style.fontSize = '10px';
                iconPlaceholder.style.fontWeight = 'bold';
                iconPlaceholder.textContent = engine.name.charAt(0);

                engineBtn.appendChild(iconPlaceholder);

                // 异步加载图标
                loadEngineIcon(engineBtn, iconPlaceholder, engine);

                // 添加搜索引擎名称
                engineBtn.appendChild(document.createTextNode(engine.name));

                // 如果启用快捷键,添加提示
                const settings = loadSettings();
                if (settings.keyboardShortcuts) {
                    const shortcutSpan = document.createElement('span');
                    shortcutSpan.className = 'shortcut';
                    shortcutSpan.textContent = `${modifierKey}${engine.shortcut}`;
                    engineBtn.appendChild(shortcutSpan);
                }

                engineContainer.appendChild(engineBtn);
            }

            // 如果没有其他搜索引擎按钮,添加一个提示
            if (engineContainer.children.length === 0) {
                const noEnginesMsg = document.createElement('div');
                noEnginesMsg.textContent = '没有其他可用的搜索引擎';
                noEnginesMsg.style.padding = '10px 0';
                noEnginesMsg.style.color = '#888';
                noEnginesMsg.style.fontSize = '12px';
                noEnginesMsg.style.textAlign = 'center';
                engineContainer.appendChild(noEnginesMsg);
            }

            switcher.appendChild(engineContainer);

            // 添加到页面
            document.body.appendChild(switcher);

            // 设置拖拽功能
            setupDraggable(switcher, dragHandle);

            // 添加点击事件,防止冒泡
            switcher.addEventListener('click', function(e) {
                e.stopPropagation();
            });

            log('搜索引擎切换器UI已创建');
        } catch (error) {
            handleError(error, '创建搜索引擎切换器UI');
        }
    }

    // 添加加载图标的辅助函数
    async function loadEngineIcon(engineBtn, iconPlaceholder, engine) {
        try {
            // 获取图标URL (使用简单方式,不进行转换)
            const iconUrl = engine.icon;
            
            // 创建图标
            const img = document.createElement('img');
            img.src = iconUrl;
            img.className = 'engine-icon';
            img.alt = engine.name;
            
            // 当图标加载完成时替换占位符
            img.onload = function() {
                if (iconPlaceholder.parentNode) {
                    iconPlaceholder.replaceWith(img);
                }
            };
            
            img.onerror = function() {
                // 图标加载失败时使用文本替代
                const textIcon = document.createElement('div');
                textIcon.className = 'engine-icon text-icon';
                textIcon.textContent = engine.name.charAt(0);
                if (iconPlaceholder.parentNode) {
                    iconPlaceholder.replaceWith(textIcon);
                }
            };
            
            // 添加到DOM,但保持隐藏状态直到加载完成
            if (iconPlaceholder.parentNode) {
                iconPlaceholder.parentNode.appendChild(img);
                img.style.display = 'none';
            }
        } catch (error) {
            log(`加载图标出错: ${error.message}`);
            // 出错时使用文本图标
            try {
                const textIcon = document.createElement('div');
                textIcon.className = 'engine-icon text-icon';
                textIcon.textContent = engine.name.charAt(0);
                if (iconPlaceholder.parentNode) {
                    iconPlaceholder.replaceWith(textIcon);
                }
            } catch (e) {
                // 忽略
            }
        }
    }

    // 设置拖拽功能 - 新增
    function setupDraggable(element, handle) {
        try {
            if (!element || !handle) return;

            let isDragging = false;
            let offset = { x: 0, y: 0 };
            let debounceTimer = null;

            handle.addEventListener('mousedown', startDrag);
            handle.addEventListener('touchstart', startDrag, { passive: false });

            function startDrag(e) {
                e.preventDefault();
                e.stopPropagation();

                // 获取初始位置
                const clientX = e.clientX || (e.touches && e.touches[0].clientX) || 0;
                const clientY = e.clientY || (e.touches && e.touches[0].clientY) || 0;
                const rect = element.getBoundingClientRect();

                offset = {
                    x: clientX - rect.left,
                    y: clientY - rect.top
                };

                isDragging = true;

                // 添加拖动和结束事件监听
                document.addEventListener('mousemove', drag);
                document.addEventListener('touchmove', drag, { passive: false });
                document.addEventListener('mouseup', stopDrag);
                document.addEventListener('touchend', stopDrag);

                // 添加正在拖动的样式
                element.style.transition = 'none';
                element.style.opacity = '0.8';
            }

            function drag(e) {
                if (!isDragging) return;
                e.preventDefault();

                // 获取当前位置
                const clientX = e.clientX || (e.touches && e.touches[0].clientX) || 0;
                const clientY = e.clientY || (e.touches && e.touches[0].clientY) || 0;

                // 计算新位置
                let newLeft = clientX - offset.x;
                let newTop = clientY - offset.y;

                // 边界检查
                const maxLeft = window.innerWidth - element.offsetWidth;
                const maxTop = window.innerHeight - element.offsetHeight;

                newLeft = Math.max(0, Math.min(maxLeft, newLeft));
                newTop = Math.max(0, Math.min(maxTop, newTop));

                // 应用新位置
                element.style.left = `${newLeft}px`;
                element.style.top = `${newTop}px`;
            }

            function stopDrag() {
                if (!isDragging) return;

                // 移除事件监听
                document.removeEventListener('mousemove', drag);
                document.removeEventListener('touchmove', drag);
                document.removeEventListener('mouseup', stopDrag);
                document.removeEventListener('touchend', stopDrag);

                // 恢复样式
                element.style.transition = '';
                element.style.opacity = '';

                // 获取最终位置并保存
                const rect = element.getBoundingClientRect();
                const finalPosition = {
                    top: rect.top,
                    left: rect.left
                };

                // 保存位置到设置
                clearTimeout(debounceTimer);
                debounceTimer = setTimeout(() => {
                    const settings = loadSettings();
                    settings.position = finalPosition;
                    settings.autoPosition = false; // 用户手动拖动后禁用自动定位
                    saveSettings(settings);
                    log('拖动位置已保存', finalPosition);
                }, 300);

                isDragging = false;
            }

            log('拖拽功能已设置');
        } catch (error) {
            handleError(error, '设置拖拽功能');
        }
    }

    // 显示设置面板
    function showSettingsPanel() {
        try {
            log('显示设置面板');

            // 获取当前设置
            const settings = loadSettings();

            // 创建设置覆盖层
            const settingsOverlay = document.createElement('div');
            settingsOverlay.className = 'settings-overlay';

            // 创建设置容器
            const settingsContainer = document.createElement('div');
            settingsContainer.className = 'settings-container';

            // 创建设置标题
            const settingsHeader = document.createElement('div');
            settingsHeader.className = 'settings-header';
            settingsHeader.textContent = '搜索引擎切换工具设置';
            settingsContainer.appendChild(settingsHeader);

            // 创建搜索引擎选择部分
            const enginesSection = document.createElement('div');
            enginesSection.className = 'settings-section';

            const enginesTitle = document.createElement('div');
            enginesTitle.className = 'section-title';
            enginesTitle.textContent = '选择要显示的搜索引擎:';
            enginesSection.appendChild(enginesTitle);

            const checkboxGroup = document.createElement('div');
            checkboxGroup.className = 'checkbox-group';

            // 添加每个搜索引擎的复选框
            defaultEngines.forEach(engine => {
                const label = document.createElement('label');

                const checkbox = document.createElement('input');
                checkbox.type = 'checkbox';
                checkbox.name = 'engine';
                checkbox.value = engine.name;
                checkbox.checked = settings.enabledEngines.includes(engine.name);
                label.appendChild(checkbox);

                label.appendChild(document.createTextNode(` ${engine.name}`));

                checkboxGroup.appendChild(label);
            });

            enginesSection.appendChild(checkboxGroup);
            settingsContainer.appendChild(enginesSection);

            // 创建快捷键部分
            const shortcutSection = document.createElement('div');
            shortcutSection.className = 'settings-section';

            const shortcutTitle = document.createElement('div');
            shortcutTitle.className = 'section-title';
            shortcutTitle.textContent = '快捷键设置:';
            shortcutSection.appendChild(shortcutTitle);

            const shortcutCheckbox = document.createElement('label');
            const shortcutInput = document.createElement('input');
            shortcutInput.type = 'checkbox';
            shortcutInput.name = 'keyboardShortcuts';
            shortcutInput.checked = settings.keyboardShortcuts;
            shortcutCheckbox.appendChild(shortcutInput);
            shortcutCheckbox.appendChild(document.createTextNode(' 启用键盘快捷键'));
            shortcutSection.appendChild(shortcutCheckbox);

            // Mac特定选项
            if (isMac) {
                const shortcutContent = document.createElement('div');
                shortcutContent.style.marginTop = '10px';
                shortcutContent.style.marginLeft = '20px';
                shortcutContent.style.display = settings.keyboardShortcuts ? 'block' : 'none';

                shortcutInput.addEventListener('change', function() {
                    shortcutContent.style.display = this.checked ? 'block' : 'none';
                });

                const shortcutDescription = document.createElement('div');
                shortcutDescription.style.marginTop = '10px';
                shortcutDescription.style.marginLeft = '20px';
                shortcutDescription.style.display = settings.keyboardShortcuts ? 'block' : 'none';

                // 根据平台和修饰键类型显示正确的快捷键描述
                const shortcutModifier = isMac ? 
                    (settings.macModifierType === 'command' ? '⌘ Command' : '⌥ Option') : 
                    'Alt';
                shortcutDescription.innerHTML = `按下 <b>${shortcutModifier}+搜索引擎首字母</b> 可快速切换搜索引擎<br><span style="color:#777;font-size:12px;">(例如: ${shortcutModifier}+B 切换到Bing)</span>`;

                shortcutContent.appendChild(shortcutDescription);

                const macOptions = document.createElement('div');
                macOptions.className = 'mac-options';

                // Command选项
                const commandOption = document.createElement('label');
                commandOption.className = 'mac-option';

                const commandRadio = document.createElement('input');
                commandRadio.type = 'radio';
                commandRadio.name = 'macModifier';
                commandRadio.value = 'command';
                commandRadio.checked = settings.macModifierType === 'command';

                commandOption.appendChild(commandRadio);
                commandOption.appendChild(document.createTextNode(' ⌘ (Command)'));

                // Option选项
                const optionOption = document.createElement('label');
                optionOption.className = 'mac-option';

                const optionRadio = document.createElement('input');
                optionRadio.type = 'radio';
                optionRadio.name = 'macModifier';
                optionRadio.value = 'option';
                optionRadio.checked = settings.macModifierType === 'option';

                optionOption.appendChild(optionRadio);
                optionOption.appendChild(document.createTextNode(' ⌥ (Option)'));

                macOptions.appendChild(commandOption);
                macOptions.appendChild(optionOption);

                shortcutContent.appendChild(macOptions);
                shortcutSection.appendChild(shortcutContent);
            } else {
                const shortcutDescription = document.createElement('div');
                shortcutDescription.style.marginTop = '10px';
                shortcutDescription.style.marginLeft = '20px';
                shortcutDescription.style.display = settings.keyboardShortcuts ? 'block' : 'none';
                shortcutDescription.textContent = '按下 Alt + 快捷键字母 可快速切换搜索引擎';

                shortcutInput.addEventListener('change', function() {
                    shortcutDescription.style.display = this.checked ? 'block' : 'none';
                });

                shortcutSection.appendChild(shortcutDescription);
            }

            settingsContainer.appendChild(shortcutSection);

            // 创建位置设置部分
            const positionSection = document.createElement('div');
            positionSection.className = 'settings-section';

            const positionTitle = document.createElement('div');
            positionTitle.className = 'section-title';
            positionTitle.textContent = '位置设置:';
            positionSection.appendChild(positionTitle);

            const positionCheckbox = document.createElement('label');
            const positionInput = document.createElement('input');
            positionInput.type = 'checkbox';
            positionInput.name = 'autoPosition';
            positionInput.checked = settings.autoPosition;
            positionCheckbox.appendChild(positionInput);
            positionCheckbox.appendChild(document.createTextNode(' 自动选择最佳位置'));

            const positionDescription = document.createElement('div');
            positionDescription.style.marginTop = '5px';
            positionDescription.style.marginLeft = '20px';
            positionDescription.style.fontSize = '12px';
            positionDescription.style.color = '#666';
            positionDescription.textContent = '禁用后可拖动切换器到任意位置';

            positionSection.appendChild(positionCheckbox);
            positionSection.appendChild(positionDescription);
            settingsContainer.appendChild(positionSection);

            // 创建折叠选项
            const collapseSection = document.createElement('div');
            collapseSection.className = 'settings-section';

            const collapseCheckbox = document.createElement('label');
            const collapseInput = document.createElement('input');
            collapseInput.type = 'checkbox';
            collapseInput.name = 'collapsed';
            collapseInput.checked = settings.collapsed;
            collapseCheckbox.appendChild(collapseInput);
            collapseCheckbox.appendChild(document.createTextNode(' 默认折叠'));

            collapseSection.appendChild(collapseCheckbox);
            settingsContainer.appendChild(collapseSection);

            // 添加快捷搜索设置部分
            const quickSearchSection = document.createElement('div');
            quickSearchSection.className = 'settings-section';
            
            const quickSearchTitle = document.createElement('div');
            quickSearchTitle.className = 'section-title';
            quickSearchTitle.textContent = '快捷搜索跳转:';
            quickSearchSection.appendChild(quickSearchTitle);
            
            const quickSearchCheckbox = document.createElement('label');
            const quickSearchInput = document.createElement('input');
            quickSearchInput.type = 'checkbox';
            quickSearchInput.name = 'quickSearchEnabled';
            quickSearchInput.checked = settings.quickSearchEnabled !== false; // 默认启用
            quickSearchCheckbox.appendChild(quickSearchInput);
            quickSearchCheckbox.appendChild(document.createTextNode(' 启用关键词快捷搜索'));
            quickSearchSection.appendChild(quickSearchCheckbox);
            
            const quickSearchDescription = document.createElement('div');
            quickSearchDescription.style.marginTop = '10px';
            quickSearchDescription.style.marginLeft = '20px';
            quickSearchDescription.style.fontSize = '12px';
            quickSearchDescription.style.color = '#666';
            quickSearchDescription.innerHTML = '例如: 输入 <b>gh react</b> 可快速跳转到GitHub搜索react<br>' +
                '支持的前缀: gh (GitHub), yt (YouTube), bili (哔哩哔哩), zh (知乎), so (Stack Overflow), npm (NPM包)';
            
            quickSearchSection.appendChild(quickSearchDescription);
            settingsContainer.appendChild(quickSearchSection);

            // 创建按钮
            const actions = document.createElement('div');
            actions.className = 'settings-actions';

            const cancelButton = document.createElement('button');
            cancelButton.className = 'settings-button cancel-button';
            cancelButton.textContent = '取消';
            cancelButton.addEventListener('click', function() {
                settingsOverlay.remove();
            });

            const saveButton = document.createElement('button');
            saveButton.className = 'settings-button save-button';
            saveButton.textContent = '保存';
            saveButton.addEventListener('click', function() {
                // 收集设置
                const newSettings = {
                    enabledEngines: Array.from(
                        settingsContainer.querySelectorAll('input[name="engine"]:checked')
                    ).map(checkbox => checkbox.value),
                    keyboardShortcuts: settingsContainer.querySelector('input[name="keyboardShortcuts"]').checked,
                    collapsed: settingsContainer.querySelector('input[name="collapsed"]').checked,
                    autoPosition: settingsContainer.querySelector('input[name="autoPosition"]').checked,
                    quickSearchEnabled: settingsContainer.querySelector('input[name="quickSearchEnabled"]').checked
                };

                // Mac特定选项
                if (isMac) {
                    const macModifier = settingsContainer.querySelector('input[name="macModifier"]:checked');
                    newSettings.macModifierType = macModifier ? macModifier.value : 'command';
                }

                // 保存设置
                const success = saveSettings(newSettings);
                if (success) {
                    settingsOverlay.remove();

                    // 刷新切换器界面
                    const switcher = document.getElementById('search-engine-switcher');
                    if (switcher) {
                        switcher.remove();
                    }
                    setTimeout(createSwitcherUI, 100);
                }
            });

            actions.appendChild(cancelButton);
            actions.appendChild(saveButton);

            settingsContainer.appendChild(actions);

            settingsOverlay.appendChild(settingsContainer);
            document.body.appendChild(settingsOverlay);

            // 点击外部区域关闭设置面板
            settingsOverlay.addEventListener('click', function(e) {
                if (e.target === settingsOverlay) {
                    settingsOverlay.remove();
                }
            });

            log('设置面板已显示');
        } catch (error) {
            handleError(error, '显示设置面板');
        }
    }

    // 设置键盘快捷键
    function setupKeyboardShortcuts() {
        try {
            const settings = loadSettings();
            if (!settings.keyboardShortcuts) {
                log('快捷键功能已禁用');
                return;
            }
            
            const { currentEngine, query } = getCurrentEngineAndQuery();
            if (!currentEngine || !query) {
                log('未检测到搜索引擎或查询词,不设置快捷键');
                return;
            }
            
            // 根据平台显示正确的快捷键提示
            const modifierKeyDisplay = isMac ? 
                (settings.macModifierType === 'command' ? '⌘+' : '⌥+') : 
                'Alt+';
            log(`设置键盘快捷键: ${modifierKeyDisplay}字母`);
            
            document.addEventListener('keydown', function(e) {
                // 检查修饰键是否按下 - 根据用户设置和平台正确检测
                let modifierPressed = false;
                
                if (isMac) {
                    if (settings.macModifierType === 'command') {
                        modifierPressed = e.metaKey;
                    } else if (settings.macModifierType === 'option') {
                        modifierPressed = e.altKey;
                    } else {
                        modifierPressed = e.metaKey; // 默认使用Command键
                    }
                } else {
                    modifierPressed = e.altKey; // 非Mac平台使用Alt键
                }
                
                // 如果没有按下修饰键,直接返回
                if (!modifierPressed) return;
                
                // 获取按下的键
                const key = e.key.toUpperCase();
                log(`检测到组合键: ${modifierKeyDisplay}${key}`);
                
                // 跳过输入框中的按键
                if (e.target.tagName === 'INPUT' || 
                    e.target.tagName === 'TEXTAREA' || 
                    e.target.isContentEditable) {
                    return;
                }
                
                // 查找匹配的搜索引擎
                const targetEngine = getEnabledEngines().find(engine => 
                    engine.shortcut && engine.shortcut.toUpperCase() === key && 
                    (!currentEngine || engine.name !== currentEngine.name)
                );
                
                if (targetEngine) {
                    // 阻止默认事件(如浏览器快捷键)
                    e.preventDefault();
                    e.stopPropagation();
                    
                    // 切换到目标搜索引擎
                    switchToEngine(targetEngine, query);
                    return false;
                }
            }, true); // 使用捕获阶段确保优先处理
            
            log('键盘快捷键设置完成');
        } catch (error) {
            handleError(error, '设置键盘快捷键');
        }
    }

    // 设置DOM变化观察器
    function setupMutationObserver() {
        try {
            const observer = new MutationObserver((mutations) => {
                for (const mutation of mutations) {
                    if (mutation.type === 'childList' && mutation.addedNodes.length > 0) {
                        // 检查是否需要重新创建UI
                        const switcherExists = document.querySelector('#search-engine-switcher');
                        if (!switcherExists) {
                            createSwitcherUI();
                        }
                        
                        // 检查新添加的搜索框
                        mutation.addedNodes.forEach(node => {
                            if (node.nodeType === Node.ELEMENT_NODE) {
                                const inputs = node.querySelectorAll('input[type="text"], input[type="search"], input:not([type])');
                                if (inputs.length > 0) {
                                    setupQuickSearch();
                                }
                            }
                        });
                    }
                }
            });

            // 观察整个文档的变化
            observer.observe(document.body, {
                childList: true,
                subtree: true
            });

            log('MutationObserver 已设置');
        } catch (error) {
            handleError(error, '设置 MutationObserver');
        }
    }

    // 添加菜单项
    function setupUserMenu() {
        try {
            GM_registerMenuCommand('搜索引擎切换工具设置', showSettingsPanel);
            
            // 添加快捷搜索切换选项
            const settings = loadSettings();
            const menuText = settings.quickSearchEnabled ? 
                '✓ 快捷搜索跳转 (已启用)' : 
                '○ 快捷搜索跳转 (已禁用)';
                
            GM_registerMenuCommand(menuText, function() {
                const settings = loadSettings();
                settings.quickSearchEnabled = !settings.quickSearchEnabled;
                saveSettings(settings);
                alert(`快捷搜索跳转已${settings.quickSearchEnabled ? '启用' : '禁用'}.\n刷新页面后生效.`);
                location.reload();
            });
            
            log('用户菜单已设置');
        } catch (error) {
            handleError(error, '设置用户菜单');
        }
    }

    // 添加URL监控功能实现快捷搜索
    function setupURLMonitor() {
        // 检查是否启用了快捷搜索
        const settings = loadSettings();
        if (!settings.quickSearchEnabled) {
            log('快捷搜索跳转功能已禁用');
            return;
        }
        
        log('启用快捷搜索跳转功能');
        
        // 上次检查的URL
        let lastCheckedURL = '';
        
        // 检查当前URL是否符合快捷搜索模式
        function checkCurrentURL() {
            // 获取当前URL
            const currentURL = window.location.href;
            
            // 如果URL没有变化,跳过
            if (currentURL === lastCheckedURL) {
                return;
            }
            
            // 更新lastCheckedURL
            lastCheckedURL = currentURL;
            
            // 定义搜索引擎URL模式和查询参数
            const searchEnginePatterns = [
                { pattern: /google\.[^/]+\/search/, paramName: 'q' },
                { pattern: /bing\.com\/search/, paramName: 'q' },
                { pattern: /baidu\.com\/s/, paramName: 'wd' },
                { pattern: /duckduckgo\.com/, paramName: 'q' },
                { pattern: /sogou\.com\/web/, paramName: 'query' },
                { pattern: /so\.com\/s/, paramName: 'q' }
            ];
            
            // 检查是否匹配任何搜索引擎模式
            for (const engine of searchEnginePatterns) {
                if (engine.pattern.test(currentURL)) {
                    // 尝试从URL中提取查询参数
                    const urlParams = new URLSearchParams(window.location.search);
                    const query = urlParams.get(engine.paramName);
                    
                    // 检查查询是否以特定前缀开头
                    if (query) {
                        log('检测到搜索查询:', query);
                        
                        // 检查前缀
                        for (const prefix of quickSearchPrefixes) {
                            if (query.startsWith(prefix.prefix)) {
                                // 提取搜索词
                                const searchTerm = query.substring(prefix.prefix.length).trim();
                                if (searchTerm.length > 0) {
                                    log(`检测到快捷搜索: ${prefix.name}, 搜索词: ${searchTerm}`);
                                    
                                    // 构建目标URL
                                    const targetURL = prefix.url.replace('{query}', encodeURIComponent(searchTerm));
                                    
                                    // 跳转到目标URL
                                    log(`正在跳转到: ${targetURL}`);
                                    window.location.href = targetURL;
                                    return;
                                }
                            }
                        }
                    }
                    
                    // 找到了搜索引擎但没有匹配前缀,不需要检查其他搜索引擎
                    break;
                }
            }
        }
        
        // 立即检查一次
        checkCurrentURL();
        
        // 设置定期检查
        setInterval(checkCurrentURL, 200);
        
        // 监听URL变化(支持单页应用)
        let lastUrl = location.href; 
        const observer = new MutationObserver(() => {
            if (location.href !== lastUrl) {
                lastUrl = location.href;
                checkCurrentURL();
            }
        });
        
        observer.observe(document, {subtree: true, childList: true});
        
        // 监听popstate事件(浏览器前进/后退)
        window.addEventListener('popstate', checkCurrentURL);
        
        log('URL监控已设置');
    }

    // 添加新标签页快捷搜索功能
    function setupNewTabQuickSearch() {
        // 检查是否是新标签页
        if (!/newtab|about:blank|chrome:\/\/newtab/.test(window.location.href) && 
            !(/google\.com\/?$/.test(window.location.href))) {
            return;
        }
        
        log('检测到新标签页,设置快捷搜索');
        
        // 创建搜索框
        const searchContainer = document.createElement('div');
        searchContainer.id = 'quick-search-container';
        searchContainer.style.position = 'fixed';
        searchContainer.style.top = '30%';
        searchContainer.style.left = '50%';
        searchContainer.style.transform = 'translate(-50%, -50%)';
        searchContainer.style.zIndex = '9999';
        searchContainer.style.width = '500px';
        searchContainer.style.padding = '20px';
        searchContainer.style.backgroundColor = '#fff';
        searchContainer.style.borderRadius = '8px';
        searchContainer.style.boxShadow = '0 4px 20px rgba(0,0,0,0.15)';
        searchContainer.style.fontFamily = '-apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif';
        
        // 添加标题
        const title = document.createElement('h2');
        title.textContent = '快捷搜索';
        title.style.margin = '0 0 15px 0';
        title.style.color = '#333';
        searchContainer.appendChild(title);
        
        // 添加说明
        const description = document.createElement('p');
        description.textContent = '输入前缀+搜索词,按回车直接跳转到对应网站搜索结果';
        description.style.margin = '0 0 15px 0';
        description.style.color = '#666';
        description.style.fontSize = '14px';
        searchContainer.appendChild(description);
        
        // 添加前缀列表
        const prefixList = document.createElement('div');
        prefixList.style.display = 'flex';
        prefixList.style.gap = '10px';
        prefixList.style.marginBottom = '15px';
        
        for (const prefix of quickSearchPrefixes) {
            const badge = document.createElement('div');
            badge.style.padding = '3px 8px';
            badge.style.backgroundColor = '#f0f0f0';
            badge.style.borderRadius = '4px';
            badge.style.fontSize = '13px';
            badge.style.cursor = 'pointer';
            badge.innerHTML = `<strong>${prefix.prefix}</strong> ${prefix.description}`;
            
            badge.addEventListener('click', () => {
                searchInput.value = prefix.prefix;
                searchInput.focus();
            });
            
            prefixList.appendChild(badge);
        }
        
        searchContainer.appendChild(prefixList);
        
        // 添加搜索框
        const searchInput = document.createElement('input');
        searchInput.type = 'text';
        searchInput.placeholder = '输入前缀+搜索词,如: gh cursor';
        searchInput.style.width = '100%';
        searchInput.style.padding = '10px 15px';
        searchInput.style.fontSize = '16px';
        searchInput.style.border = '1px solid #ddd';
        searchInput.style.borderRadius = '4px';
        searchInput.style.outline = 'none';
        searchInput.style.boxSizing = 'border-box';
        
        searchInput.addEventListener('focus', () => {
            searchInput.style.borderColor = '#4285f4';
            searchInput.style.boxShadow = '0 0 0 2px rgba(66, 133, 244, 0.2)';
        });
        
        searchInput.addEventListener('blur', () => {
            searchInput.style.borderColor = '#ddd';
            searchInput.style.boxShadow = 'none';
        });
        
        searchInput.addEventListener('keydown', (e) => {
            if (e.key === 'Enter') {
                const query = searchInput.value.trim();
                
                // 检查是否匹配任何前缀
                for (const prefixConfig of quickSearchPrefixes) {
                    if (query.startsWith(prefixConfig.prefix)) {
                        const searchTerm = query.substring(prefixConfig.prefix.length).trim();
                        if (searchTerm.length > 0) {
                            // 构建URL并跳转
                            const targetUrl = prefixConfig.url.replace('{query}', encodeURIComponent(searchTerm));
                            log(`新标签页快捷搜索跳转: ${targetUrl}`);
                            window.location.href = targetUrl;
                            return;
                        }
                    }
                }
                
                // 如果没有匹配任何前缀,使用默认搜索引擎
                window.location.href = `https://www.google.com/search?q=${encodeURIComponent(query)}`;
            }
        });
        
        searchContainer.appendChild(searchInput);
        
        // 添加到页面
        document.body.appendChild(searchContainer);
        
        // 自动聚焦搜索框
        setTimeout(() => {
            searchInput.focus();
        }, 300);
    }

    // 添加 setupQuickSearch 函数的实现
    function setupQuickSearch() {
        try {
            const settings = loadSettings();
            if (!settings.quickSearchEnabled) {
                log('快捷搜索功能已禁用');
                return;
            }

            log('设置快捷搜索功能');

            // 定义快捷搜索前缀和对应URL
            const quickSearchPatterns = {
                'gh ': 'https://github.com/search?q={query}',
                'so ': 'https://stackoverflow.com/search?q={query}',
                'yt ': 'https://www.youtube.com/results?search_query={query}',
                'bili ': 'https://search.bilibili.com/all?keyword={query}',
                'zh ': 'https://www.zhihu.com/search?type=content&q={query}',
                'npm ': 'https://www.npmjs.com/search?q={query}'
            };

            // 查找页面上的搜索输入框
            const searchInputs = document.querySelectorAll('input[type="text"], input[type="search"], input:not([type])');
            
            searchInputs.forEach(input => {
                // 避免重复添加事件监听
                if (input.dataset.quickSearchEnabled) return;
                
                input.dataset.quickSearchEnabled = 'true';
                
                input.addEventListener('keydown', function(e) {
                    if (e.key === 'Enter') {
                        const value = this.value.trim();
                        
                        // 检查是否匹配任何前缀
                        for (const [prefix, url] of Object.entries(quickSearchPatterns)) {
                            if (value.startsWith(prefix)) {
                                const query = value.substring(prefix.length).trim();
                                if (query) {
                                    e.preventDefault();
                                    e.stopPropagation();
                                    window.location.href = url.replace('{query}', encodeURIComponent(query));
                                    return;
                                }
                            }
                        }
                    }
                }, true);
            });

            log('快捷搜索功能已设置');
        } catch (error) {
            handleError(error, '设置快捷搜索');
        }
    }

    // 初始化
    function initialize() {
        try {
            log('开始初始化搜索引擎切换助手');
            
            // 添加全局错误处理
            window.addEventListener('error', function(event) {
                if (event.filename && event.filename.includes('search-engine-switcher')) {
                    console.error('[搜索引擎切换助手] 全局错误:', event.message);
                    return true; // 阻止默认处理
                }
            }, true);
            
            // 添加unhandledrejection处理
            window.addEventListener('unhandledrejection', function(event) {
                handleError(event.reason || new Error('未处理的Promise拒绝'), 'Promise错误');
                event.preventDefault();
            });
            
            addStyles();
            setupUserMenu();
            
            // 设置URL监控(新增)
            setupURLMonitor();
            
            // 根据页面类型选择功能
            if (/newtab|about:blank|chrome:\/\/newtab/.test(window.location.href) || 
                /google\.[^/]+\/?$/.test(window.location.href)) {
                // 如果是新标签页,设置新标签页快捷搜索
                setupNewTabQuickSearch();
            } else {
                // 如果是搜索结果页,创建搜索引擎切换器UI
                setTimeout(() => {
                    try {
                        createSwitcherUI();
                        
                        // 设置快捷搜索功能
                        setupQuickSearch();
                        
                        // 然后设置键盘快捷键
                        setTimeout(() => {
                            try {
                                setupKeyboardShortcuts();
                                
                                // 最后设置MutationObserver
                                setTimeout(() => {
                                    try {
                                        setupMutationObserver();
                                    } catch (e) {
                                        handleError(e, '设置MutationObserver');
                                    }
                                }, 200);
                            } catch (e) {
                                handleError(e, '设置键盘快捷键');
                            }
                        }, 150);
                    } catch (e) {
                        handleError(e, '创建UI');
                    }
                }, 100);
            }
            
            log('初始化完成');
        } catch (error) {
            handleError(error, '初始化');
        }
    }

    // 立即执行初始化
    initialize();

    // 确保DOM加载后执行
    if (document.readyState !== 'complete') {
        document.addEventListener('DOMContentLoaded', function() {
            setTimeout(createSwitcherUI, 300);
        });
    }

    // 页面完全加载后再次执行
    window.addEventListener('load', function() {
        setTimeout(createSwitcherUI, 500);
    });
})();

QingJ © 2025

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