搜索引擎净化

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

当前为 2024-12-07 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

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

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         搜索引擎净化
// @namespace    https://github.com/QingJ01/Search_clear
// @version      1.0.0
// @description  一个轻量级的搜索引擎优化脚本。自动净化百度、谷歌、必应等搜索结果页面,支持移除广告、优化重定向链接、清理URL、隐藏弹窗等功能,让搜索体验更简洁高效
// @author       QingJ
// @icon         
// @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
// @run-at       document-start
// @supportURL   https://github.com/QingJ01/Search_clear/issues
// @homepageURL  https://github.com/QingJ01/Search_clear
// ==/UserScript==

(function() {
    'use strict';

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

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

    /**
     * 检查当前域名是否匹配
     * @param {string} host - 要检查的域名
     * @returns {boolean}
     */
    function isMatchHost(host) {
        return location.host.includes(host);
    }

    /**
     * 搜索引擎广告过滤
     */
    function removeAds() {
        // 百度广告过滤
        if(isMatchHost(HOSTS.BAIDU)) {
            // 移除顶部和右侧广告
            removeElements([
                '.ec_wise_ad',
                '#content_right',
                // 移除带有广告标记的内容
                '.c-container >>> .c-container:has(.f13>span:starts-with("广告"))',
                '#content_right>div:has(a:contains("广告"))',
                '#content_left>div:has(span.tuiguang:contains("广告"))',
                '#content_left>div:has(span.brand:contains("广告"))',
                '#content_left>div:has(a:contains("广告"))',
                // 移除多余元素
                '#content_right>br',
                '#content_right>div:not([id])',
                '#content_left>div:has(.na-like-container)',
                // 移除劫持和推荐
                '.res_top_banner',
                '#content_left div[class*="_rs"]'
            ].join(','));

            // 移除手机版广告
            if(location.host.includes('m.baidu.com')) {
                removeElements([
                    '#page-bd div:not(.result)',
                    '#page-bd div:not([class])',
                    '.na-like-container'
                ].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="]',
                '.b_algo:has(.rms_img[src*="/th?id=OADD2."][src$="21.2"])',
                '.b_algo:has(.rms_img[src*="=AdsPlus"])',
                'DIV#bnp_container',
                'li.b_ad',
                '.ad_sc'
            ].join(','));

            // 处理特殊广告标记
            const resList = [...document.querySelectorAll("ol>li")].filter(one => one.querySelector('p'));
            resList.filter(one => 
                window.getComputedStyle(one.querySelector('p'), '::before')
                     .getPropertyValue('content')
                     .includes('url')
            ).forEach(ad => ad.remove());
        }

        // 搜狗广告过滤
        if(isMatchHost(HOSTS.SOGOU)) {
            removeElements([
                '#so_kw-ad',
                '#m-spread-left',
                '#m-spread-bottom',
                '#righttop_box li:has(span:contains("广告"))'
            ].join(','));
        }

        // 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"]').forEach(link => {
                try {
                    const match = link.href.match(/url=(.+)&/);
                    if(match && match[1]) {
                        link.href = decodeURIComponent(match[1]);
                    }
                } catch(e) {
                    console.error('百度重定向处理失败:', e);
                }
            });
        }

        // 处理谷歌重定向
        if(isMatchHost(HOSTS.GOOGLE)) {
            document.querySelectorAll('a[onmousedown]').forEach(link => {
                link.removeAttribute('onmousedown');
                link.setAttribute('target', '_blank');
                link.removeAttribute('data-jsarwt');
            });
        }

        // 处理知乎重定向
        if(isMatchHost('zhihu.com')) {
            document.querySelectorAll('a[href*="//link.zhihu.com/?target"]').forEach(link => {
                try {
                    link.href = decodeURIComponent(
                        link.href.replace(/https?:\/\/link\.zhihu\.com\/\?target=/, '')
                    );
                } catch(e) {
                    console.error('知乎重定向处理失败:', e);
                }
            });
        }

        // 处理百度学术重定向
        if(isMatchHost('xueshu.baidu.com')) {
            document.querySelectorAll('a[href*="sc_vurl=http"]').forEach(link => {
                try {
                    const xurl = new URLSearchParams(link.href).get('sc_vurl');
                    if(xurl) link.href = decodeURIComponent(xurl);
                } catch(e) {
                    console.error('百度学术重定向处理失败:', e);
                }
            });
        }

        // 处理360搜索重定向
        if(isMatchHost(HOSTS.SO)) {
            document.querySelectorAll('a[href*="so.com/link"]').forEach(link => {
                try {
                    const url = new URL(link.href);
                    const targetUrl = url.searchParams.get('url');
                    if(targetUrl) {
                        link.href = decodeURIComponent(targetUrl);
                    }
                } catch(e) {
                    console.error('360搜索重定向处理失败:', e);
                }
            });
        }

        // 处理神马搜索重定向
        if(isMatchHost(HOSTS.SM)) {
            document.querySelectorAll('a[href*="//yz.m.sm.cn/adclick"]').forEach(link => {
                try {
                    const url = new URL(link.href);
                    const targetUrl = url.searchParams.get('url');
                    if(targetUrl) {
                        link.href = decodeURIComponent(targetUrl);
                    }
                } catch(e) {
                    console.error('神马搜索重定向处理失败:', e);
                }
            });
        }
    }

    /**
     * 隐藏必应APP弹窗
     */
    function hideBingPopup() {
        if(isMatchHost(HOSTS.BING)) {
            // 隐藏弹窗容器
            GM_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();
        // 监听URL变化
        window.addEventListener('locationchange', function() {
            sturl();
        });
    }

    /**
     * URL参数清理主函数 
     */
    function sturl() {
        try {
            // 当前URL
            var url = window.location.href;
            // 新URL
            var nurl = window.location.href;
            
            // 需要移除的查询参数
            const qs = [
                // 百度参数
                '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',
                
                // 谷歌参数
                'tbas','ved','uact','ei','ie','oq','sclient','cshid','dpr','iflsig',
                'aqs','gs_lcp','source','sourceid','sxsrf','pccc','sa','biw','bih',
                'hl','newwindow',
                
                // 必应参数
                'tsc','sp','FORM','form','pq','sc','qs','sk','cvid','lq','ghsh',
                'ghacc','ghpl','ghc'
            ];

            // 需要在特定值时移除的参数
            const qseq = [['start', '0']];

            // 移除普通参数
            nurl = rmqs(nurl, qs);
            // 移除特定值参数
            nurl = rmqseq(nurl, qseq);

            // URL未改变则不处理
            if(url === nurl) {
                return false;
            }

            // 更新地址栏URL
            window.history.replaceState(null, null, nurl);

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

    /**
     * 移除普通查询参数
     */
    function rmqs(url, qs) {
        url = new URL(url);
        qs.forEach(function(param) {
            url.searchParams.delete(param);
        });
        return url.toString();
    }

    /**
     * 移除特定值的查询参数
     */
    function rmqseq(url, qseq) {
        url = new URL(url);
        qseq.forEach(function(item) {
            if(url.searchParams.get(item[0]) === item[1]) {
                url.searchParams.delete(item[0]);
            }
        });
        return url.toString();
    }

    // 添加 locationchange 事件监听支持
    history.pushState = (f => function pushState() {
        var ret = f.apply(this, arguments);
        window.dispatchEvent(new Event('pushstate'));
        window.dispatchEvent(new Event('locationchange'));
        return ret;
    })(history.pushState);

    history.replaceState = (f => function replaceState() {
        var ret = f.apply(this, arguments);
        window.dispatchEvent(new Event('replacestate')); 
        window.dispatchEvent(new Event('locationchange'));
        return ret;
    })(history.replaceState);

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

    /**
     * 初始化函数
     */
    function init() {
        // 执行功能
        removeAds();
        redirectOptimize();
        hideBingPopup();
        shortenUrl();
        
        // 监听页面变化
        const observer = new MutationObserver(() => {
            removeAds();
            redirectOptimize();
        });
        
        observer.observe(document.body, {
            childList: true,
            subtree: true
        });
    }

    // 在页面加载完成后执行初始化
    if(document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', init);
    } else {
        init();
    }

})();