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