搜索引擎净化

一个轻量级的搜索引擎优化脚本。自动净化百度、谷歌、必应等搜索结果页面,支持移除广告、优化重定向链接、清理URL、隐藏弹窗等功能,让搜索体验更简洁高效

Zainstaluj skrypt?
Skrypt zaproponowany przez autora

Może Ci się również spodobać. 123云盘解锁

Zainstaluj skrypt
// ==UserScript==
// @name         搜索引擎净化
// @namespace    https://github.com/QingJ01/Search_clear
// @version      1.1.0
// @icon         
// @description  一个轻量级的搜索引擎优化脚本。自动净化百度、谷歌、必应等搜索结果页面,支持移除广告、优化重定向链接、清理URL、隐藏弹窗等功能,让搜索体验更简洁高效
// @author       QingJ
// @license      Apache Licence 2.0
// @match        *://*.google.com/*
// @match        *://*.google.com.hk/*
// @match        *://*.google.co.jp/*
// @match        *://*.baidu.com/*
// @match        *://*.bing.com/*
// @match        *://*.cn.bing.com/*
// @match        *://*.s.cn.bing.net/*
// @match        *://*.sogou.com/*
// @match        *://*.m.sogou.com/*
// @match        *://*.wap.sogou.com/*
// @match        *://*.so.com/*
// @match        *://*.m.so.com/*
// @match        *://*.sm.cn/*
// @match        *://*.m.sm.cn/*
// @match        *://*.yz.m.sm.cn/*
// @match        *://*.so.toutiao.com/*
// @grant        GM_addStyle
// @grant        unsafeWindow
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_openInTab
// @grant        GM_xmlhttpRequest
// @connect      api.staticj.top
// @run-at       document-start
// @require      https://code.jquery.com/jquery-3.6.0.min.js
// ==/UserScript==

(function (cat) {
    "use strict";

    const userConfig = {
        css: " {display: none !important;width: 0 !important;height: 0 !important;} ",
        timeout: 10000,
        tryCount: 5,
        tryTimeout: 500,
    };

    const commonFunctionObject = {
        GMgetValue(key, defaultValue) {
            return GM_getValue(key, defaultValue);
        },
        GMsetValue(key, value) {
            GM_setValue(key, value);
        },
        GMopenInTab(url) {
            GM_openInTab(url, { active: true });
        },
        randomNumber() {
            return Math.floor(Math.random() * 100000000);
        },
        webToast(config) {
            const message = config.message || '';
            const background = config.background || '#333';
            const toast = document.createElement('div');
            toast.style.cssText = `
                position: fixed;
                top: 50%;
                left: 50%;
                transform: translate(-50%, -50%);
                padding: 10px 20px;
                background: ${background};
                color: #fff;
                border-radius: 4px;
                z-index: 999999;
            `;
            toast.textContent = message;
            document.body.appendChild(toast);
            setTimeout(() => toast.remove(), 2000);
        },
        request(method, url, data) {
            return new Promise((resolve, reject) => {
                GM_xmlhttpRequest({
                    method: method,
                    url: url,
                    data: data,
                    onload: function (response) {
                        resolve(response);
                    },
                    onerror: function (error) {
                        reject(error);
                    }
                });
            });
        }
    };

    /**
     * 常量定义
     */
    const HOSTS = {
        BAIDU: 'baidu.com',
        GOOGLE: 'google.com',
        BING: 'bing.com',
        SOGOU: 'sogou.com',
        SO: 'so.com',
        SM: 'sm.cn'
    };

    /**
     * 工具函数:移除元素
     */
    function removeElements(selector) {
        try {
            document.querySelectorAll(selector).forEach(el => el.remove());
        } catch (e) {
            console.error('移除元素失败:', selector, e);
        }
    }

    /**
     * 检查当前域名是否匹配
     */
    function isMatchHost(host) {
        return location.host.includes(host);
    }

    /**
     * 添加样式
     */
    function addStyle(css, pass = 0) {
        let el;
        if (pass >= userConfig.tryCount) return;
        if (typeof cat.GM_addStyle == "function") {
            el = cat.GM_addStyle(css);
        } else {
            el = document.createElement("style");
            el.textContent = css;
            document.documentElement.appendChild(el);
        }
        if (typeof el == "object") {
            if (!el || !document.documentElement.contains(el)) {
                setTimeout(() => {
                    addStyle(css, pass + 1);
                }, userConfig.tryTimeout);
            }
        }
    }

    /**
     * 搜索引擎广告过滤
     */
    function removeAds() {
        // 百度广告过滤
        if (isMatchHost(HOSTS.BAIDU)) {
            // 移除顶部和右侧广告
            removeElements('.ec_wise_ad');
            removeElements('#content_right');

            // 移除带有广告标记的内容 - 更精确的选择器
            document.querySelectorAll('#content_left > div').forEach(container => {
                // 检查是否包含广告标记
                if (container.querySelector('.f13 > span')?.textContent?.includes('广告') ||
                    container.querySelector('a[data-landurl]') ||
                    container.querySelector('span.tuiguang')?.textContent?.includes('广告') ||
                    container.querySelector('span.brand')?.textContent?.includes('广告') ||
                    container.querySelector('span[data-tuiguang]')) {
                    container.remove();
                }
            });

            // 移除多余元素 - 更精确的选择器
            removeElements('#content_right > br');
            removeElements('#content_right > div:not([id])');
            removeElements('#content_left > div.result-op');
            removeElements('#content_left > div[class*="recommend"]');

            // 移除劫持和推荐
            removeElements('.res_top_banner');
            removeElements('#content_left div[class*="_rs"]');

            // 移除手机版广告
            if (location.host.includes('m.baidu.com')) {
                // 基础广告元素
                removeElements([
                    '.ec_wise_ad',
                    '.ec-result-inner',
                    '.c-result.result-op',
                    '.download-tip',
                    '.float-ball',
                    '.ball-wrapper',
                    '.na-like-container'
                ].join(','));

                // 移除带有广告标记的内容
                document.querySelectorAll('.c-container').forEach(container => {
                    if (container.querySelector('.c-icons-outer')?.textContent?.includes('广告') ||
                        container.hasAttribute('data-tuiguang') ||
                        container.querySelector('[data-tuiguang]')) {
                        container.remove();
                    }
                });

                // 移除底部推广
                removeElements([
                    '.c-recommends',
                    '.c-flex-recommend',
                    '.c-recommend-tip',
                    '[data-module="recommend"]'
                ].join(','));
            }
        }

        // 谷歌广告过滤
        if (isMatchHost(HOSTS.GOOGLE)) {
            removeElements([
                '.commercial-unit',
                '#tads',
                '#bottomads',
                'div[aria-label="广告"]',
                'div[aria-label="Ads"]'
            ].join(','));
        }

        // 必应广告过滤
        if (isMatchHost(HOSTS.BING)) {
            removeElements([
                'li.b_ad',
                '.pa_sb',
                '.adsMvC',
                'a[h$=",Ads"]',
                'a[href*="/aclick?ld="]',
                'DIV#bnp_container',
                '.ad_sc'
            ].join(','));

            // 移除特定图片的广告
            document.querySelectorAll('.b_algo').forEach(algo => {
                const img = algo.querySelector('.rms_img');
                if (img && (img.src.includes('/th?id=OADD2.') ||
                    img.src.includes('=AdsPlus'))) {
                    algo.remove();
                }
            });
        }

        // 搜狗广告过滤
        if (isMatchHost(HOSTS.SOGOU)) {
            removeElements([
                '#so_kw-ad',
                '#m-spread-left',
                '#m-spread-bottom'
            ].join(','));

            // 移除带有广告标记的内容
            document.querySelectorAll('#righttop_box li').forEach(li => {
                if (li.querySelector('span')?.textContent.includes('广告')) {
                    li.remove();
                }
            });
        }

        // 360搜索广告过滤
        if (isMatchHost(HOSTS.SO)) {
            removeElements([
                '.res-mediav',
                '.e_result',
                '.c-title-tag',
                'DIV.res-mediav-right',
                'DIV.inner_left',
                '#so-activity-entry',
                'DIV.tg-wrap'
            ].join(','));
        }

        // 神马搜索广告过滤
        if (isMatchHost(HOSTS.SM)) {
            removeElements([
                '.ad-wrapper',
                '.ec_wise_ad',
                '.qb-download-banner-non-share',
                'DIV[data-text-ad]',
                '.ad-block'
            ].join(','));

            // 移除手机版广告
            if (location.host.includes('m.sm.cn')) {
                removeElements([
                    'DIV.ad-alert-info',
                    '.se-recommend-word-list-container',
                    '#se-recommend-word-list-container',
                    '[class*="ball-wrapper"]',
                    'DIV#page-copyright.se-page-copyright[style*="margin-bottom: 50px"]',
                    'DIV[style*="position: fixed; bottom: 0px"]',
                    '[ad_dot_url*="http"]',
                    '.dl-banner-without-logo',
                    '.ad_result',
                    '.biz_sponsor'
                ].join(','));
            }
        }
    }

    /**
     * 搜索引擎重定向优化
     */
    function redirectOptimize() {
        // 处理百度重定向
        if (isMatchHost(HOSTS.BAIDU)) {
            document.querySelectorAll('a[href*="baidu.com/link"]:not([ac-redirect-processed])').forEach(link => {
                try {
                    // 标记已处理,避免重复
                    link.setAttribute('ac-redirect-processed', '1');

                    // 保存原始链接
                    const originalHref = link.href;

                    // 移除原有的点击事件
                    link.removeAttribute('onclick');
                    link.removeAttribute('onmousedown');

                    // 使用GM_xmlhttpRequest处理重定向
                    link.addEventListener('click', function (e) {
                        e.preventDefault();
                        e.stopPropagation();

                        GM_xmlhttpRequest({
                            method: "GET",
                            url: originalHref,
                            headers: {
                                "Accept": "*/*",
                                "Referer": originalHref
                            },
                            timeout: 5000,
                            onload: function (response) {
                                let directUrl = response.finalUrl;
                                if (!directUrl) {
                                    // 尝试从响应文本中提取URL
                                    const matches = /URL='([^']+)'/.exec(response.responseText);
                                    directUrl = matches ? matches[1] : null;
                                }
                                if (directUrl && !directUrl.includes('baidu.com/link')) {
                                    window.open(directUrl, '_blank');
                                } else {
                                    window.open(originalHref, '_blank');
                                }
                            },
                            onerror: function () {
                                window.open(originalHref, '_blank');
                            }
                        });
                    }, { once: true }); // 确保事件只触发一次

                } catch (e) {
                    console.error('百度重定向处理失败:', e);
                }
            });
        }

        // 处理谷歌重定向
        if (isMatchHost(HOSTS.GOOGLE)) {
            document.querySelectorAll('a[onmousedown]:not([ac-redirect-processed]), a[data-jsarwt]:not([ac-redirect-processed])').forEach(link => {
                try {
                    // 标记已处理
                    link.setAttribute('ac-redirect-processed', '1');

                    // 移除Google的重定向属性
                    link.removeAttribute('onmousedown');
                    link.removeAttribute('data-jsarwt');
                    link.removeAttribute('ping');

                    // 新标签页打开
                    link.setAttribute('target', '_blank');

                } catch (e) {
                    console.error('谷歌重定向处理失败:', e);
                }
            });
        }

        // 处理必应重定向
        if (isMatchHost(HOSTS.BING)) {
            document.querySelectorAll('a[href*="/click?"]:not([ac-redirect-processed]), a[href*="go.microsoft.com"]:not([ac-redirect-processed])').forEach(link => {
                try {
                    // 标记已处理
                    link.setAttribute('ac-redirect-processed', '1');

                    // 尝试从href中提取目标URL
                    const url = new URL(link.href);
                    let targetUrl = url.searchParams.get('u') || url.searchParams.get('r');

                    if (targetUrl) {
                        // 设置解码后的URL
                        link.href = decodeURIComponent(targetUrl);
                        link.setAttribute('target', '_blank');

                        // 移除原有的点击事件
                        link.removeAttribute('onclick');
                        link.removeAttribute('onmousedown');
                    }
                } catch (e) {
                    console.error('必应重定向处理失败:', e);
                }
            });
        }
    }

    /**
     * 隐藏必应APP弹窗
     */
    function hideBingPopup() {
        if (isMatchHost(HOSTS.BING)) {
            addStyle('div#bnp_container {display: none !important;}');

            const observer = new MutationObserver(() => {
                const closeBtn = document.querySelector('div#sacs_close');
                if (closeBtn) {
                    closeBtn.click();
                }
            });

            observer.observe(document.body, {
                childList: true,
                subtree: true
            });
        }
    }

    /**
     * URL参数清理
     */
    function shortenUrl() {
        sturl();
        window.addEventListener('locationchange', sturl);
    }

    function sturl() {
        try {
            let url = new URL(window.location.href);
            let changed = false;

            // 需要移除的查询参数
            const params = [
                // 百度参数 - 完整列表
                'rsp', 'prefixsug', 'fr', 'bsst', 'f', 'inputT', 'usm', 'rsv_page',
                'rqlang', 'rsv_t', 'oq', 'rsv_pq', 'rsv_spt', 'ie', 'rsv_enter',
                'rsv_sug1', 'rsv_sug7', 'rsv_sug2', 'rsv_sug3', 'rsv_iqid',
                'rsv_bp', 'rsv_btype', 'rsv_idx', 'rsv_dl', 'issp', 'cshid',
                'tn', 'rsv_sug4', 'rtt', 'bsst', 'rsv_sid', 'rsv_tn', '_ss',
                'rsv_jmp', 'rsv_bl', 'rsv_sug5', 'rsv_sug6', 'rsv_sug8', 'rsv_sug9', 'bar',

                // 谷歌参数 - 完整列表
                'tbas', 'ved', 'uact', 'ei', 'ie', 'oq', 'sclient', 'cshid', 'dpr',
                'iflsig', 'aqs', 'gs_lcp', 'source', 'sourceid', 'sxsrf', 'pccc',
                'sa', 'biw', 'bih', 'hl', 'newwindow', 'stick', 'gws_rd', 'client',
                'gs_rn', 'tbo', 'dcr', 'safe', 'ssui', 'psi',

                // 必应参数 - 完整列表
                'tsc', 'sp', 'FORM', 'form', 'pq', 'sc', 'qs', 'sk', 'cvid', 'lq',
                'ghsh', 'ghacc', 'ghpl', 'ghc', 'ubireng', 'FPIG', 'PC', 'setmkt',
                'setlang', 'mkt', 'qpvt', 'ensearch', 'first'
            ];

            // 移除参数
            params.forEach(param => {
                if (url.searchParams.has(param)) {
                    url.searchParams.delete(param);
                    changed = true;
                }
            });

            // 特殊参数处理
            const specialParams = [
                ['start', '0'],
                ['page', '1'],
                ['offset', '0'],
                ['first', '1']
            ];

            if (url.searchParams.get('start') === '0') {
                url.searchParams.delete('start');
                changed = true;
            }

            // 如果有改变则更新URL
            if (changed) {
                window.history.replaceState(null, null, url.toString());
            }

        } catch (e) {
            console.error('URL参数清理失败:', e);
        }
    }

    // 添加 locationchange 事件支持
    const originalPushState = history.pushState;
    const originalReplaceState = history.replaceState;

    history.pushState = function () {
        originalPushState.apply(this, arguments);
        window.dispatchEvent(new Event('locationchange'));
    };

    history.replaceState = function () {
        originalReplaceState.apply(this, arguments);
        window.dispatchEvent(new Event('locationchange'));
    };

    window.addEventListener('popstate', () => {
        window.dispatchEvent(new Event('locationchange'));
    });

    // SearchEnginesNavigation 类定义
    class SearchEnginesNavigation {
        constructor() {
            this.navigationDataCache = "navigation_data_cache";
            this.customNavigationkey = "custom-navigation-key-8898";
            this.searchEnginesData = [
                { "host": "www.baidu.com", "element": "#content_right", "elementInput": "#kw" },
                { "host": "www.so.com", "element": "#side", "elementInput": "#keyword" },
                { "host": "www.sogou.com", "element": "#right", "elementInput": "#upquery" },
                { "host": "cn.bing.com", "element": "#b_context", "elementInput": "#sb_form_q" },
                { "host": "www.bing.com", "element": "#b_context", "elementInput": "#sb_form_q" },
                { "host": "www4.bing.com", "element": "#b_context", "elementInput": "#sb_form_q" },
                { "host": "so.toutiao.com", "element": ".s-side-list", "elementInput": "input[type='search']" },
                { "host": "www.google.com", "element": "#rhs", "elementInput": "input[name='q']" }
            ];
        }

        getNavigationData(element, elementInput) {
            const defaultNavigationData = [
                {
                    "name": "搜索引擎", "list": [
                        { "name": "百度", "url": "https://www.baidu.com/s?wd=@@" },
                        { "name": "必应", "url": "https://cn.bing.com/search?q=@@" },
                        { "name": "Google", "url": "https://www.google.com/search?q=@@" },
                        { "name": "360搜索", "url": "https://www.so.com/s?ie=utf-8&fr=none&src=360sou_newhome&nlpv=basest&q=@@" },
                        { "name": "搜狗", "url": "https://www.sogou.com/web?query=@@" },
                        { "name": "头条搜索", "url": "https://so.toutiao.com/search?dvpf=pc&source=input&keyword=@@" }
                    ]
                },
                {
                    "name": "资源搜索", "list": [
                        { "name": "财经搜索", "url": "https://www.shaduizi.com/s/search?q=@@&currentPage=1" },
                        { "name": "百度百科", "url": "https://baike.baidu.com/item/@@" },
                        { "name": "知乎搜索", "url": "https://www.zhihu.com/search?type=content&q=@@" },
                        { "name": "B站搜索", "url": "https://search.bilibili.com/all?keyword=@@&from_source=webtop_search&spm_id_from=333.851" },
                        { "name": "抖音搜索", "url": "https://www.douyin.com/search/@@?aid=0a9fc74b-01e8-4fb0-9509-307c5c07fda1&publish_time=0&sort_type=0&source=normal_search&type=general" },
                        { "name": "搜狗|公众号", "url": "https://weixin.sogou.com/weixin?type=2&query=@@" },
                        { "name": "搜狗|知乎", "url": "https://www.sogou.com/sogou?pid=sogou-wsse-ff111e4a5406ed40&insite=zhihu.com&ie=utf8&p=73351201&query=@@&ie=utf8&p=73351201&query=@@" },
                        { "name": "豆瓣搜索", "url": "https://www.douban.com/search?q=@@" },
                        { "name": "电影搜索", "url": "https://www.cupfox.com/search?key=@@" },
                        { "name": "维基百科", "url": "https://en.wikipedia.org/w/index.php?search=@@" },
                        { "name": "法律法规", "url": "https://www.pkulaw.com/law/chl?Keywords=@@" },
                        { "name": "icon搜索", "url": "https://www.iconfont.cn/search/index?searchType=icon&q=@@" },
                        { "name": "github", "url": "https://github.com/search?q=@@" },
                        { "name": "csdn", "url": "https://so.csdn.net/so/search?q=@@&t=&u=" },
                        { "name": "stackoverflow", "url": "https://stackoverflow.com/" }
                    ]
                }
            ];

            let cacheNavigationData = commonFunctionObject.GMgetValue(this.navigationDataCache, null);
            if (!cacheNavigationData) {
                cacheNavigationData = defaultNavigationData;
            }

            let finalNavigationData = null;
            try {
                let customNavigationData = commonFunctionObject.GMgetValue(this.customNavigationkey, null);
                finalNavigationData = customNavigationData ? cacheNavigationData.concat(customNavigationData) : cacheNavigationData;
            } catch (e) {
                finalNavigationData = cacheNavigationData;
            }

            this.createHtml(element, elementInput, finalNavigationData);

            // 更新缓存数据
            commonFunctionObject.request("get", "http://api.staticj.top/script/api/get/navigation_json_url?t=" + new Date().getTime(), null)
                .then(resultData => {
                    let dataJson = JSON.parse(resultData.data);
                    if (dataJson?.url) {
                        commonFunctionObject.request("get", dataJson.url, null)
                            .then(resultData2 => {
                                let serverNavigationData = resultData2.data;
                                if (!cacheNavigationData ||
                                    (cacheNavigationData && serverNavigationData.length != JSON.stringify(cacheNavigationData).length)) {
                                    commonFunctionObject.GMsetValue(this.navigationDataCache, JSON.parse(serverNavigationData));
                                }
                            })
                            .catch(() => { });
                    }
                })
                .catch(() => { });
        }

        createCss(elementNum) {
            const cssContent = `
        /* 导航容器样式 */
        #QingjByeBug {
            background: #fff;
            border-radius: 8px;
            box-shadow: 0 2px 8px rgba(0,0,0,0.1);
            padding: 15px;
            margin-bottom: 20px;
        }

        /* 分类样式 */
        .ddfdfd${elementNum}dffssqa {
            margin-top: 13px;
        }

        .ddfdfd${elementNum}dffssqa:first-child {
            margin-top: 0;
        }

        /* 标题样式 */
        .ddfdfd${elementNum}dffssqa>.title {
            font-size: 15px;
            color: #333;
            margin-bottom: 10px;
            padding-bottom: 8px;
            border-bottom: 1px solid #eee;
        }

        .ddfdfd${elementNum}dffssqa>.title b {
            position: relative;
            padding-left: 10px;
        }

        .ddfdfd${elementNum}dffssqa>.title b:before {
            content: '';
            position: absolute;
            left: 0;
            top: 50%;
            transform: translateY(-50%);
            width: 4px;
            height: 16px;
            background: #4e6ef2;
            border-radius: 2px;
        }

        /* 链接列表样式 */
        .ddfdfd${elementNum}dffssqa>.content-list {
            display: flex;
            flex-wrap: wrap;
            gap: 8px;
        }

        /* 链接样式 */
        .ddfdfd${elementNum}dffssqa>.content-list>a {
            flex: 0 0 calc(31% - 4px);
            text-decoration: none;
            color: #333;
            background: #f5f5f5;
            border: 1px solid transparent;
            border-radius: 4px;
            padding: 6px 2px;
            text-align: center;
            font-size: 13px;
            line-height: 1.5;
            overflow: hidden;
            white-space: nowrap;
            text-overflow: ellipsis;
            transition: all 0.3s ease;
        }

        .ddfdfd${elementNum}dffssqa>.content-list>a:hover {
            background: #4e6ef2;
            color: #fff;
            transform: translateY(-1px);
            box-shadow: 0 2px 8px rgba(78,110,242,0.3);
        }

        /* 底部信息样式 */
        #QingjByeBug > div:last-child {
            margin-top: 15px;
            padding-top: 10px;
            border-top: 1px solid #eee;
            color: #999;
            font-size: 12px;
        }

        #QingjByeBug > div:last-child a {
            color: #666;
            text-decoration: none;
            transition: color 0.3s ease;
        }

        #QingjByeBug > div:last-child a:hover {
            color: #4e6ef2;
        }

        /* 自定义按钮样式 */
        a[name="customNavigation"] {
            display: inline-flex;
            align-items: center;
            gap: 4px;
        }

        a[name="customNavigation"]:before {
            content: '🔧';
            font-size: 14px;
        }
    `;

            if ($("#plugin_css_style_dddsoo").length == 0) {
                $("body").prepend(`<style id='plugin_css_style_dddsoo'>${cssContent}</style>`);
            }
        }

        createHtml(element, elementInput, navigationData) {
            $("#QingjByeBug").remove(); // 保持原有ID

            const elementNum = commonFunctionObject.randomNumber();
            let isComplete = true;

            // 添加这一行,确保CSS样式被添加
            this.createCss(elementNum);

            const elementInterval = setInterval(() => {
                if (isComplete) {
                    const $element = $(element);
                    const $box = $("#QingjByeBug");
                    isComplete = false;

                    if ($element.length != 0 && $box.length == 0) {
                        let html = "<div id='QingjByeBug'>";

                        // 遍历导航分类
                        navigationData.forEach(category => {
                            html += `
                                <div class='ddfdfd${elementNum}dffssqa'>
                                    <div class='title'><b>${category.name}</b></div>
                                    <div class='content-list'>
                            `;

                            // 遍历分类下的链接
                            category.list.forEach(item => {
                                html += `<a target='_blank' name='navigation' data-url='${item.url}' href='javascript:void(0);'>${item.name}</a>`;
                            });

                            html += "</div></div>";
                        });

                        html += `
                            <div style='margin-bottom:10px;margin-top:5px;font-size:12px;'>
                                <a target='_blank' href='https://greasyfork.org/zh-CN/scripts/520018'>*该数据由 搜索引擎净化 提供</a>
                                &nbsp;&nbsp;
                                <a href="javascript:void(0);" name="customNavigation">自定义网址</a>
                            </div>
                        </div>`;

                        // 插入导航面板
                        $element.prepend(html);

                        // 绑定链接点击事件
                        $("#QingjByeBug a[name='navigation']").on("click", function (e) {
                            e.preventDefault();
                            const url = $(this).data("url").replace("@@", $(elementInput).val());
                            commonFunctionObject.GMopenInTab(url);
                        });

                        // 绑定自定义导航事件
                        $("#QingjByeBug a[name='customNavigation']").on("click", function (e) {
                            e.preventDefault();
                            self.showSetingDialog();
                        });
                    }
                    isComplete = true;
                }
            }, 100);
        }

        showSetingDialog() {
            const customNavigation = commonFunctionObject.GMgetValue(this.customNavigationkey, null);
            const customNavigationData = customNavigation ? JSON.stringify(customNavigation, null, 4) : "";

            const content = `
        <div class="custom-navigation-dialog">
            <div class="notice-section">
                <h3>注意事项:</h3>
                <ol>
                    <li>请严格按照格式添加,否则不生效</li>
                    <li>数据为json格式,请确保json格式正确,必要时请到 <a target="_blank" href="https://www.json.cn/">json.cn</a> 校验</li>
                    <li>点击下面"示例"按钮,查看具体格式情况</li>
                    <li>链接中的搜索关键词请用"@@"代替,脚本会自动替换成当前搜索词</li>
                    <li>清空 -> 保存,则取消自定义的导航网址</li>
                </ol>
            </div>
            <div class="textarea-section">
                <textarea 
                    placeholder="请严格按照格式填写,否则不生效"
                    class="navigation-textarea"
                >${customNavigationData}</textarea>
            </div>
            <div class="button-section">
                <button class="navigation-example">示例</button>
                <button class="navigation-clear">清空</button>
                <button class="navigation-save">保存自定义导航</button>
            </div>
        </div>
    `;

            // 创建弹窗样式
            const style = document.createElement('style');
            style.textContent = `
        .custom-navigation-dialog {
            font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif;
            color: #333;
            padding: 20px;
        }

        .custom-navigation-dialog .notice-section {
            background: #f8f9fa;
            border-radius: 8px;
            padding: 15px 20px;
            margin-bottom: 20px;
        }

        .custom-navigation-dialog .notice-section h3 {
            color: #e74c3c;
            margin: 0 0 10px 0;
            font-size: 16px;
            font-weight: 600;
        }

        .custom-navigation-dialog .notice-section ol {
            margin: 0;
            padding-left: 20px;
        }

        .custom-navigation-dialog .notice-section li {
            line-height: 1.6;
            margin-bottom: 5px;
            color: #666;
            font-size: 14px;
        }

        .custom-navigation-dialog .notice-section a {
            color: #2196f3;
            text-decoration: none;
            transition: color 0.3s;
        }

        .custom-navigation-dialog .notice-section a:hover {
            color: #1976d2;
            text-decoration: underline;
        }

        .custom-navigation-dialog .textarea-section {
            margin: 20px 0;
        }

        .custom-navigation-dialog .navigation-textarea {
            width: 100%;
            height: 200px;
            padding: 12px;
            border: 1px solid #ddd;
            border-radius: 8px;
            font-family: Consolas, Monaco, 'Courier New', monospace;
            font-size: 14px;
            line-height: 1.5;
            resize: none;
            background-color: #fff;
            transition: border-color 0.3s, box-shadow 0.3s;
        }

        .custom-navigation-dialog .navigation-textarea:focus {
            outline: none;
            border-color: #4e6ef2;
            box-shadow: 0 0 0 3px rgba(78,110,242,0.1);
        }

        .custom-navigation-dialog .button-section {
            text-align: center;
            margin-top: 20px;
        }

        .custom-navigation-dialog button {
            padding: 8px 20px;
            margin: 0 8px;
            border: none;
            border-radius: 6px;
            font-size: 14px;
            font-weight: 500;
            cursor: pointer;
            transition: all 0.3s ease;
        }

        .custom-navigation-dialog button:hover {
            transform: translateY(-1px);
        }

        .custom-navigation-dialog .navigation-example {
            background: #f5f5f5;
            color: #333;
        }

        .custom-navigation-dialog .navigation-example:hover {
            background: #e0e0e0;
        }

        .custom-navigation-dialog .navigation-clear {
            background: #ff4d4f;
            color: white;
        }

        .custom-navigation-dialog .navigation-clear:hover {
            background: #ff7875;
        }

        .custom-navigation-dialog .navigation-save {
            background: #4e6ef2;
            color: white;
        }

        .custom-navigation-dialog .navigation-save:hover {
            background: #6c87f5;
        }

        .custom-navigation-dialog-overlay {
            position: fixed;
            top: 0;
            left: 0;
            right: 0;
            bottom: 0;
            background: rgba(0, 0, 0, 0.5);
            z-index: 999998;
            animation: fadeIn 0.3s ease;
        }

        .custom-navigation-dialog-container {
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: white;
            border-radius: 12px;
            box-shadow: 0 4px 24px rgba(0, 0, 0, 0.15);
            z-index: 999999;
            width: 90%;
            max-width: 600px;
            max-height: 90vh;
            overflow: auto;
            animation: slideIn 0.3s ease;
        }

        .custom-navigation-dialog-close {
            position: absolute;
            right: 16px;
            top: 16px;
            width: 28px;
            height: 28px;
            border-radius: 50%;
            border: none;
            background: #f5f5f5;
            color: #666;
            font-size: 20px;
            line-height: 28px;
            text-align: center;
            cursor: pointer;
            transition: all 0.3s ease;
            padding: 0;
            display: flex;
            align-items: center;
            justify-content: center;
        }

        .custom-navigation-dialog-close:hover {
            background: #ff4d4f;
            color: white;
        }

        @keyframes fadeIn {
            from { opacity: 0; }
            to { opacity: 1; }
        }

        @keyframes slideIn {
            from {
                opacity: 0;
                transform: translate(-50%, -48%);
            }
            to {
                opacity: 1;
                transform: translate(-50%, -50%);
            }
        }
    `;

            // 创建遮罩层
            const overlay = document.createElement('div');
            overlay.className = 'custom-navigation-dialog-overlay';

            // 创建弹窗容器
            const dialogContainer = document.createElement('div');
            dialogContainer.className = 'custom-navigation-dialog-container';
            dialogContainer.innerHTML = content;

            // 创建关闭按钮
            const closeBtn = document.createElement('button');
            closeBtn.className = 'custom-navigation-dialog-close';
            closeBtn.innerHTML = '×';
            dialogContainer.appendChild(closeBtn);

            // 添加样式和元素到页面
            document.head.appendChild(style);
            document.body.appendChild(overlay);
            document.body.appendChild(dialogContainer);

            // 绑定事件
            const textarea = dialogContainer.querySelector('.navigation-textarea');
            const exampleJson = [{
                "name": "我的导航",
                "list": [
                    { "name": "百度", "url": "https://www.baidu.com/s?wd=@@" },
                    { "name": "必应", "url": "https://cn.bing.com/search?q=@@" }
                ]
            }];

            // 示例按钮事件
            dialogContainer.querySelector('.navigation-example').onclick = () => {
                textarea.value = JSON.stringify(exampleJson, null, 4);
            };

            // 清空按钮事件
            dialogContainer.querySelector('.navigation-clear').onclick = () => {
                textarea.value = '';
            };

            // 保存按钮事件
            dialogContainer.querySelector('.navigation-save').onclick = () => {
                const content = textarea.value;
                if (!content) {
                    commonFunctionObject.GMsetValue(this.customNavigationkey, null);
                    commonFunctionObject.webToast({ "message": "保存成功:数据为空", "background": "#4e6ef2" });
                    closeDialog();
                    return;
                }

                try {
                    const contentJson = JSON.parse(content);
                    if (this.validateNavigationFormat(contentJson)) {
                        commonFunctionObject.GMsetValue(this.customNavigationkey, contentJson);
                        commonFunctionObject.webToast({ "message": "保存成功", "background": "#4e6ef2" });
                        closeDialog();
                        // 刷新导航显示
                        this.show();
                    } else {
                        commonFunctionObject.webToast({ "message": "格式错误,请更正", "background": "#ff4d4f" });
                    }
                } catch (e) {
                    commonFunctionObject.webToast({ "message": "格式错误,请更正", "background": "#ff4d4f" });
                }
            };

            // 关闭弹窗函数
            const closeDialog = () => {
                overlay.remove();
                dialogContainer.remove();
                style.remove();
            };

            // 关闭按钮事件
            closeBtn.onclick = closeDialog;
            // 点击遮罩层关闭
            overlay.onclick = closeDialog;
            // 阻止弹窗点击事件冒泡到遮罩
            dialogContainer.onclick = (e) => e.stopPropagation();
        }

        validateNavigationFormat(data) {
            if (!Array.isArray(data)) return false;

            return data.every(category => {
                if (typeof category !== 'object' || !category.name || !Array.isArray(category.list)) {
                    return false;
                }
                return category.list.every(item => {
                    return typeof item === 'object' &&
                        typeof item.name === 'string' &&
                        typeof item.url === 'string';
                });
            });
        }

        show() {
            const host = window.location.host;
            const href = window.location.href;

            if ((host === "www.baidu.com") ||
                (host === "www.so.com" && href.includes("www.so.com/s")) ||
                (host === "www.sogou.com" && (href.includes("www.sogou.com/web") || href.includes("www.sogou.com/sogou"))) ||
                (host === "cn.bing.com" && href.includes("cn.bing.com/search")) ||
                (host === "www.bing.com" && href.includes("www.bing.com/search")) ||
                (host === "www4.bing.com" && href.includes("www4.bing.com/search")) ||
                (host === "so.toutiao.com" && href.includes("so.toutiao.com/search")) ||
                (host === "www.google.com" && href.includes("www.google.com/search"))) {

                const currentSearchEnginesData = this.searchEnginesData.find(item => host === item.host);
                if (currentSearchEnginesData) {
                    this.getNavigationData(currentSearchEnginesData.element, currentSearchEnginesData.elementInput);
                }
            }
        }

        start() {
            this.show();
        }
    }

    /**
     * 初始化函数
     */
    function init() {
        removeAds();
        redirectOptimize();
        hideBingPopup();
        shortenUrl();
        new SearchEnginesNavigation().start();
    }

    // 使用 MutationObserver 监听DOM变化
    const observer = new MutationObserver(() => {
        removeAds();
        redirectOptimize();
    });

    observer.observe(document, {
        childList: true,
        subtree: true,
        attributes: true,
        attributeFilter: ["id", "class"]
    });

    // 初始化执行
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', init);
    } else {
        init();
    }

})({
    GM_info: typeof GM_info == "object" ? GM_info : {},
    unsafeWindow: typeof unsafeWindow == "object" ? unsafeWindow : window,
    GM_addStyle: typeof GM_addStyle == "function" ? GM_addStyle : undefined,
});