Github搜索净化

净化Github搜索页,屏蔽cirosantilli等人的敏感仓库。

目前为 2024-01-03 提交的版本。查看 最新版本

// ==UserScript==
// @name         Github搜索净化
// @name:zh-CN   Github搜索净化
// @name:en      Github Search Purification
// @namespace    https://github.com/danicastarr
// @version      1.2.0
// @description  净化Github搜索页,屏蔽cirosantilli等人的敏感仓库。
// @description:zh-CN  净化Github搜索页,屏蔽敏感仓库。
// @description:en Clean up Github search page, block sensitive repositories by cirosantilli and others.
// @icon       
// @license      GPLv3
// @author       DanicaStar ch3rry
// @match        *://github.com/search*
// @grant        GM_registerMenuCommand
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_addStyle
// @grant        unsafeWindow
// @run-at       document-end
// ==/UserScript==
(function (){
    "use strict" // 严格模式
    let ban = ['cirosantilli', 'wumaoland', 'codin-stuffs', 'cheezcharmer', 'gege-circle', 'zhaohmng-outlook-com', 'zaohmeing', 'Daravai1234', 'candice531033938', 'jk-ice-cream', 'jk-ice-cream-250', 'sky8964', 'pxvr-official', 'zpc1314521', 'jjzhang166', 'panbinibn', 'programthink', 'hello-world-1989', 'b0LBwZ7r5HOeh6CBMuQIhVu3-s-random-fork', 'thethetpvmy', 'wwwswitch520cc', 'shotoanqrob', 'sitempeanhkkwg', 'fukeluo', '1206256980', 'curees', 'yuoppo', 'Createree', 'vghl', 'wholedata', 'dunjian', 'mksshare', 'abshare', 'tpxdat', 'jhdyg', 'changfengqj', 'Dujltqzv', 'xmq1024', 'golade', 'kdjfhd', 'dkjhy', 'junsolg', 'dkjiiu', 'faithhow', 'yamtioy', 'zugzuc', 'lusvont', 'kenyatas', 'koeheu', 'juttama', 'duspub', 'wuqdid', 'visxud', 'suyfic', 'qokkod', 'roepuo', 'purfob', 'gitprocode', 'ynwynw', 'hanguodianying', 'hgyw', '69sm', 'urlapp', 'Augensternhx', 'urlweb', 'fuliso', 'nishjd', '36dshipin', 'hapump', 'zhguoxmw', 'KoreanMovies', 'hanjutv', 'mamadepengyou', 'mamatouyunmuxuan', 'erzideqizi', 'wodeqizidejiejie', 'xiaoyizidemeng', 'qingyuzongheng', 'jiangnanerxi', 'hanguobiaomei', 'djhgy', 'XXOOBY', 'baoyu1024', 'kk234kkkk', '15923-ORIX', 'wutaed', 'webzhibo', 'apptuijian', 'follow666', 'yu90892', 'aconteet', 'getmal', 'itxinfei', 'mingtiana', 'midoushipin', 'paofushipin', 'yinghanshipin', 'GTVapp', 'huangyouquan', 'devlookme', 'audwq', 'jhdgy', 'di6gandh', 'shuangyuzhibo', 'lvchazhibo', 'xiaolanshipin', 'bofangqi', 'yingtaoshipin', 'xiangfeizhibo', 'lvchaApp', 'luoshenzhibo', 'yaojizhibo', 'mudanzhibo', 'aiaizhibo', 'gaochaoqwe', 'jiolde', 'lsdhw', 'kanavdaohang', 'harnh', 'kuadaner', 'wapquan', 'laoyeer', 'reteres', 'haoersn', 'zhengjianzhong0107', 'huaaweiCode', 'jianjian00001', 'm2ak-dev', 'yyzwz', 'froginwe11', 'luanmenglei', 'xijinping0', 'cyqqq', 'qldaisd', 'lTbgykio', 'yao270161651', 'jt0008jt0008', '15625103741', 'sky1234566778', 'chfucao', 'chifuyidaocao', 'updrmeltm', 'alice548', 'yazm000', 'cpnorg', 'tffygbu', 'Liberty-China', '1989CCP', 'liulihaocai', 'RevolutionaryCommitteeCPC', 'LeiyanLu', 'webdao', 'GC4WP', 'tu01', 'ziliao1', 'zzs70', 'ff2017', 'guitu2017', 'tu2017', 'wm001', 'wnel2017', 'dunhlino', 'nelaliox', 'jianjian3219', 'giteecode', '666bears', 'wang-buer', 'id681ilyg316', 'uhjid', 'usdui', 'uhskl', 'uyjks', 'uhskldf', 'itgsi5', 'uifskv', 'uhgask', 'igfkld', 'udsjd', 'ufodk', 'uigsjt', 'ighfrs', 'haivs', 'idrkkld', 'yuisju', 'uldydj', 'uyuek', 'tydfj', 'uuedif', 'ykwsw3', 'uigsi7', 'tyiis', 'ykeik', 'ukvdj', 'uyikl', 'ufzekg', 'yiksure', 'rhksgz', 'rthls', 'rhjaw', 'rehlxs', 'thzsgt', 'tdidst', 'eglct', 'tjkdyu', 'tjlks', 'tjjds', 'rllfs', 'rhkstd', 'yjscdr', 'servisee', 'ufsjzf', 'bvnbvnfgd', 'duliyingshi', 'calendi', 'mayeobey', 'QQMusic-Jay-Chou', 'boylovecomic', 'bt9527', 'FarmerChina', 'Waymon102092', 'baofx', 'biehd', 'moonpas', 'lyqilo', 'liliqh', 'hourv', 'xinfue', 'jijidianying', 'YuyanCai', 'jtdh', 'isdkxr', 'yhildyu', 'ykldyld', 'igsigk', 'uidekj', 'iufskw', 'udsjhf', 'tjkdx', 'rtkist', 'tjlsyh', 'euhf', 'rjzsht', 'rhkdzu', 'ehkkld', 'xzgfsw', 'iofgd', 'yufdk', 'ujkdub', 'iofgdsk', 'dyghikg', 'ugdskf', 'ifwaih', 'oigsiu', 'yjksku', 'yfdkkrf', 'thjsqd', 'yjsyhf', 'ydjsu6', 'igseyf', 'ujudy8', 'tykde', 'ykmdi8', 'yklzrf', 'uijdkd', 'yjkshc', 'tkajc', 'ykdzs', 'jklsx', 'ejldux', 'ifxspo', 'ogsvtf', 'ifdeu', 'yudfdi', 'ofssj', 'igegkx', 'ugfkd', 'ugdsk', 'udskts', 'yjlkdss', 'fkdryl', 'rtuyjsr', 'tus56f', 'yjdsd', 'yuet6h', 'ugtw', 'tlkxt', 'yesrs', 'ykkds', 'yjksu', 'yhyshs', 'xdzfby', 'yujzdh', 'znfl', 'kjiud', 'shijuezhishi', 'hy1980boy', 'ww0304', 'ZXCASD854', 'zfpdh', 'batiyadh', 'yinsedh', 'yyfxz', 'bllpooe', 'joodfer', 'qdmang', 'chaenet', 'mzsyv', 'kzhaoes', 'clnnews', 'kendnes', 'hongnews', 'luokez', 'li721-LY', 'itunsr', 'cctnews', 'htmle', 'xmmj2', 'younownews', '445435213', 'seseClub', 'enewse', 'wsnewse', 'qsnews', 'soasmoughroy', 'adminewhat', 'wsermusic', 'molingfer', 'zhihues', '95movies', '99fuli', 'qnewse', 'tareres', 'hukioip', 'Hochoclate713', 'ervnme', 'greenleaf8888', '93-days', 'doubanm', 'xhydh', 'fvckslvt', 'MDCM-FB', 'b08240', 'm3u8-ekvod', 'huan768468', 'SweeOBC', 'ningmengsuan7788', 'supperqb', 'idskjs', 'ifsird', 'gklksr', 'ifsjxr', 'ifskxt', 'ghjklsd', 'udsskd', 'tgsjk', 'ihgsk', 'ujsjk', 'ijhdf', 'fghhgks', 'udfae4', 'jujwdj', 'ydsdk', 'uyfgsj', 'ykkxrd', 'branono', 'hytcd', 'kjiuo', 'SaolApp', 'lourv', 'uisdlk', 'hutuhai', 'dengminna', 'whmnoe4j', 'txy9704', 'ufsjl', 'udsks', 'uifsjk', 'ygsaj', 'udsts', 'yurdek', 'ghklsr', 'ifsnx', 'ufskd', 'yujst6', 'ifsurjn', 'saoyagma', 'yusyrdk', 'uijhgr', 'geeeeeeeek', 'gfjklk', 'uiskv', 'ccccsp', 'rrrsp', 'udjxs', 'qiezisp', 'egklkd', 't6korf', 'line915577', 'haijv', 'huaxinzhibo', 'haijiaofabuye', 'haijiaoshequ', 'HaijiaoCommunity', 'haijiao-app', 'fulibaike', 'lurmarp', 'entvasa', 'gotwib', 'hghkiiy121', 'gubcem', 'uijssu', 'yjhuk', 'yklsd', 'haijiaoWeb', 'winston779', 'tyukkst', 'ujsnmc', 'ygssk', 'igdkdy', 'qiezishiping', 'kjuhd', 'xiaogongzhuAPP', 'babyzhibo', 'yaojingzhibo', 'balizhibo', 'jiuaizhibo', 'liuyuezhibo', '69live', 'asidw', 'kuaimaoVIP', 'siguaha', 'mizhizhibo', 'lihzd', 'caomeizhibo', '36DAPP', 'luolisheApp', '69zhibo', 'jiejiezhibo', 'k8japan', 'buyaoshan', 'dk111222', 'fanbaovpn', 'HGcrowntiyu', '196tiyu', 'parryno', 'boyiscode', 'moonews', 'kim1528'];
    let isKeepDiv = GM_getValue("isKeepDiv", false);
    let isPrecise = GM_getValue("isPrecise", false);
    let detectMode = GM_getValue("detectMode", "mutationobserver");
    let detectDelay = GM_getValue("detectDelay", 100);

    // 关闭菜单函数
    function closeSettings() {
        let github_area = document.body.childNodes[1];
        let settingsWindow = document.getElementsByClassName("settings")[0];
        settingsWindow.style.opacity = 0;
        setInterval(()=>{settingsWindow.remove()}, 300);
        github_area.style.filter = '';
    }

    // 注册(不可用)菜单——脚本设置
    GM_registerMenuCommand(
        "⚙️脚本设置",
        function () {
            let settingMenu = document.createElement("div");
            settingMenu.className = "settings";
            settingMenu.innerHTML = '<h2><a href="https://gf.qytechs.cn/zh-CN/scripts/473912-github%E6%90%9C%E7%B4%A2%E5%87%80%E5%8C%96">Github搜索净化</a></h2><span class="userLoadNum">已加载屏蔽用户数量:加载中...</span><hr><div class="settings-block"><span>是否保留屏蔽项目Div的框:</span><label class="settings-switch"><input type="checkbox" id="isKeepDiv"><span class="slider round"></span></label></div><div class="settings-block"><span>是否精确匹配:</span><label class="settings-switch"><input type="checkbox" id="isPrecise"><span class="slider round"></span></label></div><div class="settings-block"><span>检测模式:</span><input type="radio" name="detectMode" class="settings-radio">MutationObserver(推荐)<input type="radio" name="detectMode" class="settings-radio">Loop<input type="radio" name="detectMode" class="settings-radio">eventListener</div><div class="settings-block"><span>每次检测循环间隔的时间 (毫秒) :</span><input type="number" class="settings-input"></div><button id="help" onclick="showHelp()">帮助</button><button id="save">保存</button><button id="cancel">取消</button><div class="help-div" hidden><hr><p>1. 是否保留屏蔽项目Div的框:若选择不保留,会直接在搜索结果中删除被屏蔽仓库;若选择保留,被屏蔽的仓库会显示一个包含信息"⛔该仓库被脚本屏蔽"的框。</p><p>2. 是否精确匹配:若不精确匹配,则任何innerText包含屏蔽词的仓库都会被屏蔽(.include),可能会有误杀情况;若精确匹配,则只会屏蔽屏蔽词内仓库所有者的仓库。</p><p>3. 检测模式:共有三种模式——MutationObserver、Loop和eventListener:</p><table border="1"><tr><td>检测方法</td><td>检测方式</td><td>性能开销</td><td>检测不及时情况</td></tr><tr><td>MutationObserver(推荐)</td><td>检测页面元素变动</td><td>较小</td><td>未发现</td></tr><tr><td>Loop</td><td>循环检测</td><td>较大</td><td>不可能发生</td></tr><tr><td>eventListener</td><td>检测pushState事件</td><td>很小</td><td>特定情境下发生</td></tr></table><p>4. 每次检测循环间隔的时间,单位为毫秒(1000秒 = 1秒),可根据自身设备性能调整,默认为100毫秒。</p></div>';
            document.body.appendChild(settingMenu);
            // 更新设置页面
            document.getElementsByClassName("userLoadNum")[0].innerText = "已加载屏蔽用户数量:" + ban.length;
            isKeepDiv ? document.getElementById("isKeepDiv").checked = true : document.getElementById("isKeepDiv").checked = false;
            isPrecise ? document.getElementById("isPrecise").checked = true : document.getElementById("isPrecise").checked = false;
            switch(detectMode){
                case "mutationobserver": document.getElementsByClassName("settings-radio")[0].checked = true;break;
                case "loop": document.getElementsByClassName("settings-radio")[1].checked = true;break;
                case "eventListen": document.getElementsByClassName("settings-radio")[2].checked = true;break;
            }
                let mode = document.getElementsByClassName("settings-radio");
            document.getElementsByClassName("settings-input")[0].value = detectDelay;

            // 背景模糊
            document.body.style.transition = '0.2s';
            let github_area = document.body.childNodes[1];
            github_area.style.filter = 'blur(10px)';

            // 添加按钮事件——帮助
            document.getElementById("help").onclick = function() {
                let settingsWindow = document.getElementsByClassName("settings")[0];
                let helpDiv = document.getElementsByClassName('help-div')[0];
                if (helpDiv.hidden) {
                    settingsWindow.style.height = '740px';
                    helpDiv.hidden = false;
                }
                else {
                    settingsWindow.style.height = '380px';
                    helpDiv.hidden = true;
                }
            };

            // 添加按钮事件——保存
            document.getElementById("save").onclick = function() {
                document.getElementById("isKeepDiv").checked == true ? GM_setValue("isKeepDiv", true) : GM_setValue("isKeepDiv", false);
                document.getElementById("isPrecise").checked == true ? GM_setValue("isPrecise", true) : GM_setValue("isPrecise", false);
                if (document.getElementsByClassName("settings-radio")[0].checked == true) {GM_setValue("detectMode", "mutationobserver");}
                else if (document.getElementsByClassName("settings-radio")[1].checked == true) {GM_setValue("detectMode", "loop");}
                else if (document.getElementsByClassName("settings-radio")[2].checked == true) {GM_setValue("detectMode", "eventListen");}
                let newdelayTime = parseInt(document.getElementsByClassName("settings-input")[0].value);
                if (newdelayTime > 0 && newdelayTime < 10000) {GM_setValue("detectDelay", newdelayTime)}
                else {alert("输入的delayTime有误,保存失败");}
                closeSettings();
                location.reload();
            }

            // 添加按钮事件——取消
            document.getElementById("cancel").onclick = function() {
                closeSettings();
            };
      }
    );

    // 注册(不可用)菜单——重置设置
    GM_registerMenuCommand(
    "🔄️重置设置",
    function () {
        if (confirm("是否重置脚本设置?") == true) {
            GM_setValue("isKeepDiv", false);
            GM_setValue("isPrecise", false);
            GM_setValue("detectMode", "mutationobserver");
            GM_setValue("detectDelay", 100);
            location.reload();
            alert("脚本已重置");
        }
    });
    
    
    // 屏蔽执行
    function clean() {
        if (document.querySelector("div[data-testid='results-list']") !== null) {
            let search_list = document.querySelector("div[data-testid='results-list']").childNodes;
            for (let i = 0; i < search_list.length; i++) {
                if (isBan(search_list[i], isPrecise)) {
                    if (isKeepDiv) {
                        search_list[i].firstChild.remove();
                        search_list[i].append("⛔该仓库被脚本屏蔽");
                    }
                    else {
                        search_list[i].remove();
                    }
                }
            }
        }
    };
    
    
    // 判断是否屏蔽
    function isBan(target ,isPrecise) {
        if (isPrecise) {
            if (target.getElementsByTagName("a").length !== 0) {
                let repositoryName = target.getElementsByTagName("a")[0].innerText;
                let userName = repositoryName.split("/")[0];
                for (let j = 0; j < ban.length; j++) {
                    if (userName == ban[j]) {
                        return true;
                    }
                }
                return false;
            }
            else {return false;}
        }
        else {
            for (let j = 0; j < ban.length; j++) {
                if (target.innerText.includes(ban[j])) {
                    return true;
                }
            }
            return false;
        }
    };
    //**********************************************
    //*                               *
    //*       MutationObserver检测代码       *
    //*                               *
    //**********************************************
    function cleanByMutationObserver() {
        console.log("Running:MutationObserver");
        const targetNode = document.body;
        // 观察器的配置(需要观察什么变动)
        const config = { childList: true, subtree: true };
        // 创建一个观察器实例并传入回调函数
        const observer = new MutationObserver(clean);
        // 以上述配置开始观察目标节点
        observer.observe(targetNode, config);
    }

    //**********************************************
    //*                               *
    //*         Loop检测代码              *
    //*                               *
    //**********************************************
    function cleanByLoop() {
        console.log("Running:Loop");
        setInterval(function(){
            clean();
        },detectDelay);
    };
    
    //**********************************************
    //*                               *
    //*        eventListener检测代码        *
    //*                               *
    //**********************************************
    // github 路由更新时
    function pageChange(url) {
        // 保证时机
        setTimeout(() => {
        clean()
      }, 1000)
    }
    
    
    // 重写 history event
    let _wr = function(type) {
        let orig = history[type];
        return function() {
            let rv = orig.apply(this, arguments);
            let e = new Event(type);
            e.arguments = arguments;
            window.dispatchEvent(e);
            return rv;
        }
    }
    
    function cleanByEventListener() {
        console.log("Running:EventListener");
        setTimeout(() => {clean()}, 1000);
        clean();
        // 重写方法
        history.pushState = _wr('pushState');
        //监听
        window.addEventListener('pushState', function(e) {
            pageChange(location.href);
        })
    }
    
    //**********************************************
    //*                               *
    //*              入口               *
    //*                               *
    //**********************************************

    // 添加脚本设置界面CSS
    function GM_addStyle(cssStr){
    var n = document.createElement('style');
    n.type = "text/css";
    n.innerHTML = cssStr;
    document.getElementsByTagName('head')[0].appendChild(n);
    }

    GM_addStyle(`
        div.settings {\n    transition: 0.2s; \n    position: fixed;\n    font-family:system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;\n    text-align: center;\n    font-size: large;\n    height: 380px;\n    width: 650px;\n    left: 50%;\n    top: 50%;\n    padding: 0px 20px;\n    transform:translate(-50%,-50%);\n    background-color: aliceblue;\n    border: solid lightgray 1px;\n    border-radius: 6px;\n}\n\ndiv.settings h2 {\n  margin: 20px 0px 0px 0px;\n}\n\ndiv.settings span#load {\n  font-size: small;\n}\n\ndiv.settings .settings-block {\n    padding: 10px 0px;\n}\n\ndiv.settings hr {\n    background-color: lightgray;\n    height: 1px;\n    border: none;\n}\n\ndiv.settings a:link {\n    color: black;\n    text-decoration: none;\n}\n\ndiv.settings label.settings-switch {\n    position: relative;\n    display: inline-block;\n    width: 38px;\n    height: 20px;\n}\n\ndiv.settings .slider {\n    position: absolute;\n    cursor: pointer;\n    top: 0;\n    left: 0;\n    right: 0;\n    bottom: 0;\n   background-color: lightgray;\n    -webkit-transition: .2s;\n    transition: .2s;\n    border: solid lightgray 1px;\n  }\n  \n  div.settings .slider:before {\n    position: absolute;\n    content: "";\n    height: 16px;\n    width: 16px;\n    left: 1.1px;\n    bottom: 1.3px;\n    background-color: white;\n    -webkit-transition: .2s;\n    transition: .2s;\n  }\n  \n  div.settings input:checked + .slider {\n    background-color: #2196F3;\n  }\n  \n  div.settings input:focus + .slider {\n    box-shadow: 0 0 1px #2196F3;\n  }\n  \n  div.settings input:checked + .slider:before {\n    -webkit-transform: translateX(17.2px);\n    -ms-transform: translateX(17.2x);\n    transform: translateX(17.2px);\n  }\n  \n  div.settings .slider.round {\n    border-radius: 40px;\n  }\n  \n  div.settings .slider.round:before {\n    border-radius: 50%;\n  }\n\ndiv.settings input.settings-switch {\n    opacity: 0;\n    width: 50px;\n    height: 0;\n}\n\ndiv.settings .slider:hover {\n    border: solid #4096ff 1px;\n}\n\ndiv.settings input.settings-radio {\n    transition: 0.2s;\n    zoom: 1.2;\n}\n\ndiv.settings button{\n    transition: 0.2s;\n    height: 30px;\n    width: 80px;\n    margin: 5px;\n    background-color: white;\n    border: solid lightgray 1px;\n    border-radius: 5px;\n}\n\ndiv.settings button:hover{\n    transition: 0.2s;\n    color: #4096ff;\n    border: solid #4096ff 1px;\n}\n\ndiv.settings input.settings-input{\n    transition: 0.2s;\n    height: 25px;\n    width: 70px;\n    margin: 5px;\n    background-color: white;\n    border: solid lightgray 1px;\n    border-radius: 5px;\n    padding: 0px 10px;\n}\n\ndiv.settings input.settings-input:hover{\n    transition: 0.2s;\n    color: #4096ff;\n    border: solid #4096ff 1px;\n    border-radius: 5px;\n}\n\ndiv.settings input.settings-input:focus{\n    outline:none;\n    border: solid #4096ff 1px;\n}\n\ninput[type="number"]::-webkit-inner-spin-button,\ninput[type="number"]::-webkit-outer-spin-button {\n  height: auto;\n  -webkit-appearance: none;\n}\n\ndiv.settings div.help-div {\n  text-align: left;\n  font-size: medium;\n}\n\ndiv.settings div.help-div p {\n  margin: 10px 0px;\n}
    `);

    console.log("====================\n脚本:" + GM_info.script.name + " 开始执行\n" + "作者:" + GM_info.script.author + " 版本:" + GM_info.script.version + "\n脚本地址:https://gf.qytechs.cn/zh-CN/scripts/473912-github搜索净化" + "\n====================\n" +"【脚本配置】\n" + "isKeepDiv: " + isKeepDiv + "\nisPrecise: " + isPrecise + "\ndetectMode: " + detectMode + "\ndetectDelay: " + detectDelay + "\n====================");

    switch (detectMode) {
        case "mutationobserver": cleanByMutationObserver();break;
        case "loop": cleanByLoop();break;
        case "eventListen": cleanByEventListener();break;
    }

})()

QingJ © 2025

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