搜索引擎净化

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

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==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,
});