// ==UserScript==
// @name (改)B站成分检测器(键政贵物DLC)
// @version 1.35.9
// @author hmjz100,xulaupuz,trychen
// @namespace github.com/hmjz100
// @license GPLv3
// @description B站评论区自动标注成分,支持动态、发言历史、收藏、稿件和关注识别,默认标注包括原神、崩坏3、崩坏星穹铁道、明日方舟、碧蓝航线、Vtuber、Asoul、王者荣耀、三国杀、我的世界、迷你世界、初生科技。
// @match *://*.bilibili.com/*
// @icon 
// @connect bilibili.com
// @connect bing.com
// @grant GM_xmlhttpRequest
// @grant GM_registerMenuCommand
// @grant GM_getResourceText
// @grant GM_getValue
// @grant GM_setValue
// @require https://unpkg.com/jquery@3.6.3/dist/jquery.min.js
// @require https://unpkg.com/sweetalert2@11/dist/sweetalert2.min.js
// @resource Swal https://unpkg.com/sweetalert2@11/dist/sweetalert2.min.css
// @resource SwalDark https://unpkg.com/@sweetalert2/theme-dark@5.0.15/dark.min.css
// ==/UserScript==
$(function () {
// 准备好sweetalert
let toast = Swal.mixin({
toast: true,
position: 'top-end',
showConfirmButton: false,
showCloseButton: true,
timer: 2000,
timerProgressBar: true,
didOpen: (toast) => {
toast.addEventListener('mouseenter', Swal.stopTimer);
}
});
const message = {
success: (text) => {
toast.fire({title: text, icon: 'success'});
},
error: (text) => {
toast.fire({title: text, icon: 'error'});
},
warning: (text) => {
toast.fire({title: text, icon: 'warning'});
},
info: (text) => {
toast.fire({title: text, icon: 'info'});
},
question: (text) => {
toast.fire({title: text, icon: 'question'});
}
};
// 增加CSS界面样式
let base = {
addStyle(id, tag, css) {
tag = tag || 'style';
let doc = document, styleDom = doc.getElementById(id);
if (styleDom) styleDom.remove();
let style = doc.createElement(tag);
style.rel = 'stylesheet';
style.id = id;
tag === 'style' ? style.innerHTML = css : style.href = css;
doc.getElementsByTagName('head')[0].appendChild(style);
},
addSwalStyle() {
let color = "#574AB8";
let swalcss = `
.swal2-loader{display:none;align-items:center;justify-content:center;width:2.2em;height:2.2em;margin:0 1.875em;-webkit-animation:swal2-rotate-loading 1.5s linear 0s infinite normal;animation:swal2-rotate-loading 1.5s linear 0s infinite normal;border-width:.25em;border-style:solid;border-radius:100%;border-color:${color} transparent ${color} transparent }
.swal2-styled.swal2-confirm{border:0;border-radius:.25em;background:initial;background-color:${color};color:#fff;font-size:1em}
.swal2-styled.swal2-confirm:focus{box-shadow:0 0 0 3px ${color}80 }
.swal2-timer-progress-bar-container{position:absolute;right:0;bottom:0;left:0;grid-column:auto;overflow:hidden;border-bottom-right-radius:5px;border-bottom-left-radius:5px}
.swal2-timer-progress-bar{width:100%;height:.25em;background:${color}33 }
.swal2-progress-steps .swal2-progress-step{z-index:20;flex-shrink:0;width:2em;height:2em;border-radius:2em;background:${color};color:#fff;line-height:2em;text-align:center}
.swal2-progress-steps .swal2-progress-step.swal2-active-progress-step{background:${color} }
.swal2-progress-steps .swal2-progress-step-line{z-index:10;flex-shrink:0;width:2.5em;height:.4em;margin:0 -1px;background:${color}}
.swal2-popup {padding:0 0 1.25em;flex-direction:column}
.swal2-close {position:absolute;top:0;right:0}
div:where(.swal2-container) .swal2-html-container{margin: 1em 1.3em 0.3em;}
.composition-badge {
display: inline-flex;
justify-content: center;
align-items: center;
width: fit-content;
background: #7367f026;
border-radius: 10px;
margin: 0 5px;
font-family: PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif;
cursor: pointer;
}
.composition-name {
line-height: 13px;
font-size: 13px;
color: ${color} !important;
padding: 2px 8px;
}
.composition-icon {
width: 25px;
height: 25px;
border-radius: 50%;
border: 2px solid ${color}80;
margin: -6px;
margin-right: 5px;
}
.composition-badge-control {
display: inline-flex;
justify-content: center;
align-items: center;
width: fit-content;
background: #00000008 !important;
border-radius: 10px;
margin: 0 5px;
font-family: PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif;
}
.composition-name-control {
line-height: 13px;
font-size: 12px;
color: #00000050 !important;
padding: 2px 8px;
}
`;
// 先监听颜色方案变化 SweetAlert2-Default
window.matchMedia('(prefers-color-scheme: dark)').addListener((e) => {
if (e.matches) {
// 切换到暗色主题
this.addStyle('swal-pub-style', 'style', GM_getResourceText('SwalDark'));
} else {
// 切换到浅色主题
this.addStyle('swal-pub-style', 'style', GM_getResourceText('Swal'));
}
this.addStyle('SweetAlert2-User', 'style', swalcss);
});
// 再修改主题 SweetAlert2-Default
if (window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches) {
// 切换到暗色主题
this.addStyle('swal-pub-style', 'style', GM_getResourceText('SwalDark'));
} else {
// 切换到浅色主题
this.addStyle('swal-pub-style', 'style', GM_getResourceText('Swal'));
}
this.addStyle('SweetAlert2-User', 'style', swalcss);
},
};
base.addSwalStyle();
// 在这里配置要检查的成分,以及直接 拉黑(使用指定UID评论的人直接添加标签)。
// 比如你要直接给指定UID添加一个标签,就这样写:blacklist: [401742377]
const checkers = [
{
displayName: "抽奖",
displayIcon: "",
keywords: ["互动抽奖","转发本条动态"],
},
{
displayName: "原神",
displayIcon: "https://i2.hdslb.com/bfs/face/d2a95376140fb1e5efbcbed70ef62891a3e5284f.jpg",
keywords: ["互动抽奖 #原神","#原神","#米哈游#","#miHoYo#","原神","芙宁娜","白术","赛诺","神里绫人","神里绫华","夏洛蒂","珊瑚宫","心海","北斗","五郎","九条裟罗","雷神","班尼特","夜阑","夜兰","那维莱特","行秋","枫原万叶","万叶","钟离","纳西妲","香菱","锅巴","八重神子","久岐忍","菲谢尔","艾尔海森","胡桃","林尼","达达利亚","提纳里","宵宫","莫娜","甘雨","罗莎莉亚","刻晴","北斗","九条裟罗","温蒂","温迪","阿贝多","云堇","芭芭拉","可莉","迪卢克","烟绯","早柚","重云","米卡","雷泽","多莉","凝光","丽莎","坎蒂丝","安柏","辛焱"],
followings: [401742377] // 原神官方号的 UID
},
{
displayName: "崩坏3",
displayIcon: "https://i0.hdslb.com/bfs/face/f861b2ff49d2bb996ec5fd05ba7a1eeb320dbf7b.jpg",
keywords: ["互动抽奖 #崩坏3","#崩坏3","崩坏3"],
followings: [27534330] // 崩坏3官方号的 UID
},
{
displayName: "崩坏星穹铁道",
displayIcon: "https://i2.hdslb.com/bfs/face/57b6e8c16b909a49bfc8d8394d946f908cabe728.jpg",
keywords: ["互动抽奖 #崩坏星穹铁道","#崩坏星穹铁道","崩坏星穹铁道","星铁","崩铁"],
followings: [1340190821] // 崩坏星穹铁道官方号的 UID
},
{
displayName: "明日方舟",
displayIcon: "https://i2.hdslb.com/bfs/face/d4005a0f9b898d8bb049caf9c6355f8e8f772a8f.jpg",
keywords: ["明日方舟","#明日方舟"],
followings: [
161775300, // 明日方舟官方号的 UID
]
},
{
displayName: "碧蓝航线",
displayIcon: "https://i1.hdslb.com/bfs/face/1fd5b43d5f619e6df8c8adcf13c962a3e80ee971.jpg",
keywords: ["碧蓝航线","#碧蓝航线","#舰船新增#","#碧蓝航线5周年生日快乐"],
followings: [
233114659, // 碧蓝航线官方号的 UID
]
},
{
displayName: "VTuber",
displayIcon: "https://i2.hdslb.com/bfs/face/d399d6f5cf7943a996ae96999ba3e6ae2a2988de.jpg",
keywords: ["雪蓮","塔菲","七海","草莓猫","嘉然"],
followings: [
1437582453, // 東雪蓮Official
1265680561, // 永雏塔菲
434334701, // 七海Nana7mi
1210816252, // 草莓猫Taffy
672328094, // 嘉然今天吃什么
672342685, // 乃琳Queen
351609538, // 珈乐Carol
672346917, // 向晚大魔王
672353429, // 贝拉kira
]
},
{
displayName: "Asoul",
displayIcon: "https://i2.hdslb.com/bfs/face/43b21998da8e7e210340333f46d4e2ae7ec046eb.jpg",
keywords: ["@A-SOUL_Official", "#A_SOUL#"],
followings: [
703007996, // Asoul
547510303, // Asoul二创计画
672328094, // 嘉然今天吃什么
672342685, // 乃琳Queen
351609538, // 珈乐Carol
672346917, // 向晚大魔王
672353429, // 贝拉kira
]
},
{
displayName: "王者荣耀",
displayIcon: "https://i2.hdslb.com/bfs/face/effbafff589a27f02148d15bca7e97031a31d772.jpg",
keywords: ["互动抽奖 #王者荣耀","#王者荣耀","王者荣耀"],
followings: [
57863910, // 王者荣耀
392836434, // 哔哩哔哩王者荣耀赛事
]
},
{
displayName: "三国杀",
displayIcon: "https://i0.hdslb.com/bfs/face/fe2e1a6e3dc702a6c91378e096ef37ca71bf4629.jpg",
keywords: ["互动抽奖 #三国杀","#三国杀","三国杀","#2023三国杀"],
followings: [1254932367] // 三国杀十周年官方号的 UID
},
{
displayName: "Minecraft",
displayIcon: "https://i2.hdslb.com/bfs/face/c5578966c447a70edf831bbf7e522b7be6090fea.jpg",
keywords: ["MC","我的世界","minecraft","#我的世界","我的世界拜年祭","MCBBS","我的世界中文论坛","MC玩家"],
followings: [
43310262, // 我的世界官方号的 UID
39914211, // 我的世界中文论坛(MCBBS)官方号的 UID
]
},
{
displayName: "迷你世界",
displayIcon: "https://i0.hdslb.com/bfs/face/a7591e5e0278aafb76cc083b11ca5dd46f049420.jpg",
keywords: ["mnsj","迷你世界","miniworld","#迷你世界","迷你世界拜年祭"],
followings: [
470935187, // 迷你世界官方号的 UID
]
},{
displayName: "兔友",
displayIcon: "https://pic2.zhimg.com/80/v2-8e08912298d7c7c1bf1671a448f15369_1440w.webp",
keywords: ["种花", "昂撒","殖人", "润人", "偷国", "七哥", "心医", "公知","躺匪","新冠后遗症","美狗","1450","入关","唐飞","电子宠物","不爱国","撅醒","大棋","华为","倪师","中医","文化输出","丑国","恨国","犹太","西方","渗透","白左"],
followings: [346563107, 439478093, 19248926, 289189019, 1482025194, 77701536, 323397658,648113003,397672,1458767615,510362725,2233213],
}, {
displayName: "网左",
displayIcon: "https://i0.hdslb.com/bfs/emote/0922c375da40e6b69002bd89b858572f424dcfca.png@112w_112h.webp",
keywords: ["革命", "同志", "阶级","觉醒","资本","剥削","斗争","无产","劳动","左壬","苏联","红军","布尔什维克","斯大林","列宁","中苏","封建","共产","选集","全集","马克思"],
followings: [23191782, 521249172,43219807]
}, {
displayName: "毛左",
displayIcon: "https://i0.hdslb.com/bfs/emote/2caafee2e5db4db72104650d87810cc2c123fc86.png@48w_48h.webp",
keywords: ["教员","毛主席","一声扑向","艾公","修正","邓","前三十年","前30年","艾跃进","艾老师"],
followings: [605727461,2022537742,3493133453101082]
}, {
displayName: "黄鹅",
displayIcon: "https://i1.hdslb.com/bfs/archive/87cadbe23ec449f2114cb998d04ce7160f7dfdec.jpg@672w_378h_1c_!web-search-common-cover.avif",
keywords: ["大帝", "乌贼", "鱿鱼", "大佐", "乌拉", "战斗民族","喀秋莎"],
followings: [153890218, 664086886, 622986240, 501247999,433443590]
}, {
displayName: "中医粉",
displayIcon: "https://i0.hdslb.com/bfs/face/c0629ce0c4cc3d442101704b33756b2ef738f7aa.jpg@96w_96h_1c_1s_!web-avatar.avif",
keywords: ["国医", "国药", "吃中药", "针灸", "中医黑", "汉方","经络","倪海厦","倪师","易经","弟子规","国学","汤剂","曾仕强","南怀瑾","易学","养生","中医","太极","中醫","寒湿"],
followings: []
}, {
displayName: "恨猫闹钟",
displayIcon: "https://img0.baidu.com/it/u=1427754838,2773895594&fm=253&fmt=auto&app=138&f=JPEG?w=394&h=450",
keywords: ["哈基十","猫孝子","爱猫","改猫","灭猫","猫奴","消灭流浪猫","人道毁灭","捕杀流浪猫","处理流浪猫","猫肉","狗粉","杰克辣条","流浪猫虐杀","动保"],
followings: []
},{
displayName: "键政神童",
displayIcon: "https://i2.hdslb.com/bfs/face/880c533cfd7434acf77b91552e5799d11d25ddcc.jpg@240w_240h_1c_1s_!web-avatar-search-user.webp",
keywords: ["班级","中考","高考","四六级","不放假","防沉迷","上学","班里","班委","放寒假","写作业","学校的","我们老师","开学"],
followings: []
},{
displayName: "初生科技",
displayIcon: "https://i2.hdslb.com/bfs/face/eb4c7bbea813eed3a92ee194809d85715e6a7659.jpg",
keywords: ["易语言","编程猫","scratch","破解","黑客","ramos","winpe","bsod","memz","MEMZ","WindowsCE","下崽器","aero","setup","DWM","CmzPrep","RAM","虚拟机","VMWare","希沃白板","捡垃圾","root","解BL"],
followings: [
//- 组1/关键词:system -//
493998035, // SYSTEM-RAMOS-ZDY 唯一真神,他的脸皮比医院设备还要厚
702028797, // JERRY-SYSTEM
631731585, // system-bootmgr-L
501355555, // MS-SYSTEM
1865727084, // SYSTEM-WinPE-CHD
//- 组2/关键词:bsod -//
451475014, // STR-BSOD
1511907771, // MEMZ-BSOD
1975308950, // BSOD-MEMZ
397847418, // 蓝屏钙BSOD
1776025003, // 蓝瓶钙BSoD
//- 组3/关键词:memz -//
21927744, // 360MEMZ
1353783215, // MEMZ-Chrome
412777837, // 注册表MEMZ
457692234, // 奇怪的MEMZ
298993710, // 注册表编辑器MEMZ
//- 组4/关键词:Aero setup -//
435972058, // WindowsAero毛玻璃
1452376557, // 没有Aero就没有灵魂
1911529131, // Aero8m
1321946754, // 没有Aero的Windows7
//- 组5/关键词:setup -//
589370259, // setup-windows安装
2050076822, // Windows-Setup
//- 组6/关键词:Start -//
524501321, // Start-hs888
2030178992, // Start-BME
//- 自定义组/依照个人判断 -//
1157923020, // 仗义的老班长
401094700, // 旮沓曼_gt428
356882513, // 被重组吃掉的虚拟桌面
1151325757, // SYSTEM-OPS-LJY
1304244190, // System-NBNB
504179884, // MYB_CKLS
1776456802, // 奇怪的MEMZ的小号
1534842751, // 爱WinPE的MEMZ
2112060594, // WINPE-SYSTEM
1439352366, // SYSTEM-WINPE-EXE
678414222, // Windows-regedit
1863175083, // 半不了世的空城
1736202379, // 胡桃玩VM
1322183332, // WindowsCEMEMZ新账号
414666753, // 桌面窗口管理器_DWM
1380415597, // 雨林木风YLMF
698760287, // 出星海wrcjs_sp4
307432672, // 花l火
3493108908034540, // S-1-5-21-1726115
1158046953, // VistaChrome108
727892489, // Windows2003R2
1243577821, // hyq061221
],
blacklist: [
2102787368, // Silence默不作声 按理说休学的他不应该出现在这
//- 组1/关键词:system -//
493998035, // SYSTEM-RAMOS-ZDY 唯一真神,他的脸皮比医院设备还要厚
702028797, // JERRY-SYSTEM
631731585, // system-bootmgr-L
501355555, // MS-SYSTEM
1865727084, // SYSTEM-WinPE-CHD
1162296488, // System3206
1531948091, // SYSTEM_Win11_RE
392697653, // System-i386
313342814, // SYSTEM-GREE-GZN
1546428456, // SYSTEM-WIN-EDGE
//- 组2/关键词:bsod -//
451475014, // STR-BSOD
1511907771, // MEMZ-BSOD
1975308950, // BSOD-MEMZ
397847418, // 蓝屏钙BSOD
1776025003, // 蓝瓶钙BSoD
1007224506, // EXPLORER-BSOD
1175873768, // BSOD-Winme
2032637936, // BSOD-SYSTEM
1933399514, // win11_BSOD
1641461034, // DEEPIN_BSOD2_CMD
//- 组3/关键词:memz -//
21927744, // 360MEMZ
1353783215, // MEMZ-Chrome
412777837, // 注册表MEMZ
457692234, // 奇怪的MEMZ
298993710, // 注册表编辑器MEMZ
413269076, // Cmd_MEMZ
649846967, // Win7MEMZ-BX
498912953, // AMD_MEMZ
390483853, // 炒鸡360MEMZ
362451533, // NC_Memz
//- 组5/关键词:Aero -//
435972058, // WindowsAero毛玻璃
1452376557, // 没有Aero就没有灵魂
1911529131, // Aero8m
1321946754, // 没有Aero的Windows7
//- 组5/关键词:setup -//
589370259, // setup-windows安装
2050076822, // Windows-Setup
1549141274, // system-setup
692755897, // Setup-Official
483574120, // setup安装程序
1031408618, // Deewin-Setup
671918906, // win95setup
//- 组6/关键词:Start -//
524501321, // Start-hs888
2030178992, // Start-BME
//- 自定义组/依照个人判断 -//
1157923020, // 仗义的老班长
401094700, // 旮沓曼_gt428
356882513, // 被重组吃掉的虚拟桌面
1151325757, // SYSTEM-OPS-LJY
1304244190, // System-NBNB
504179884, // MYB_CKLS
1776456802, // 奇怪的MEMZ的小号
1534842751, // 爱WinPE的MEMZ
2112060594, // WINPE-SYSTEM
1439352366, // SYSTEM-WINPE-EXE
678414222, // Windows-regedit
505199229, // SYSTEM_PHILI
652188355, // 一个windows爱好者
1863175083, // 半不了世的空城
1736202379, // 胡桃玩VM
1322183332, // WindowsCEMEMZ新账号
414666753, // 桌面窗口管理器_DWM
698760287, // 出星海wrcjs_sp4
307432672, // 花l火
3493108908034540, // S-1-5-21-1726115
1158046953, // VistaChrome108
727892489, // Windows2003R2
1243577821, // hyq061221
]
},
// 指定仙家军UID与仙话的XianLists太容易误杀,故此处删除
]
// 空间动态api
const spaceApiUrl = 'https://api.bilibili.com/x/polymer/web-dynamic/v1/feed/space?&host_mid='
const followingApiUrl = 'https://api.bilibili.com/x/relation/followings?vmid='
const searchIcon = `<svg width="12" height="12" viewBox="0 0 17 17" fill="none" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" clip-rule="evenodd" d="M16.3451 15.2003C16.6377 15.4915 16.4752 15.772 16.1934 16.0632C16.15 16.1279 16.0958 16.1818 16.0525 16.2249C15.7707 16.473 15.4456 16.624 15.1854 16.3652L11.6848 12.8815C10.4709 13.8198 8.97529 14.3267 7.44714 14.3267C3.62134 14.3267 0.5 11.2314 0.5 7.41337C0.5 3.60616 3.6105 0.5 7.44714 0.5C11.2729 0.5 14.3943 3.59538 14.3943 7.41337C14.3943 8.98802 13.8524 10.5087 12.8661 11.7383L16.3451 15.2003ZM2.13647 7.4026C2.13647 10.3146 4.52083 12.6766 7.43624 12.6766C10.3517 12.6766 12.736 10.3146 12.736 7.4026C12.736 4.49058 10.3517 2.1286 7.43624 2.1286C4.50999 2.1286 2.13647 4.50136 2.13647 7.4026Z" fill="currentColor"></path></svg>`
const checked = {}
const checking = {}
var printed = false
// 监听用户ID元素出现
waitForKeyElements(".user-name", installCheckButton);
waitForKeyElements(".sub-user-name", installCheckButton);
waitForKeyElements(".user .name", installCheckButton);
waitForKeyElements(".user-card .btn-box .like", installCheckButton);
console.log("开启B站用户成分检查器...")
let dom = ''
// 添加检查按钮
function installCheckButton(element) {
let node = $(`<div style="display: inline-block;" class="composition-checkable"><div class="composition-badge-control"><a class="composition-name-control" title="点击查看已识别用户">${searchIcon}</a></div></div>`)
if (element.attr("data-user-id") || element.attr("data-usercard-mid")){
node.on('click', function () {
checkComposition(element, node.find(".composition-name-control"))
})
node.click()
element.after(node)
} else if (element.attr("mid")) {
node.on('click', function () {
checkComposition(element, node.find(".composition-name-control"))
})
node.click()
node.css({"margin":"8px 5px"});
element.parent().after(node);
}
}
// 添加标签
function installComposition(id, element, setting) {
let node = $(`<div style="display: inline-block;"><div class="composition-badge">
<a class="composition-name" title="点击查看已识别用户">${setting.displayName}</a>
<img src="${setting.displayIcon}" class="composition-icon">
</div></div>`)
node.on('click', function () {
check.showAllUser()
})
if (element.attr("data-user-id") || element.attr("data-usercard-mid")){
element.after(node)
} else if (element.attr("mid")) {
node.css({"margin":"10px 0"});
element.parent().after(node);
}
}
// 检查标签
function checkComposition(element, loadingElement) {
// 用户ID
let userID = element.attr("data-user-id") || element.attr("data-usercard-mid") || element.attr("mid")
// 用户名
let name = element.text().charAt(0) == "@" ? element.text().substring(1) : element.text()
if (checked[userID] != undefined && GM_getValue(userID, undefined) != undefined ) {
// 已经缓存过了
let found = GM_getValue(userID, undefined)//checked[userID]
if (found.length > 0) {
for (let setting of found) {
installComposition(userID, element, setting)
}
loadingElement.parent().remove()
} else {
console.log(`检测到 ${name} ${userID} 的成分为 无`)
loadingElement.text('路人')
loadingElement.on('click', function () {
check.showAllUser()
})
}
} else if (checking[userID] != undefined) {
// 检查中
if (checking[userID].indexOf(element) < 0)
checking[userID].push(element)
} else {
checking[userID] = [element]
console.log("正在检查用户 " + name + " 的成分...");
new Promise(async (resolve, reject) => {
try {
// 找到的匹配内容
let found = []
let spaceRequest = request({
data: "",
url: spaceApiUrl + userID,
})
let bingRequest = requestbing({
data: "",
url: 'https://www.bing.com/search?q="'+name+'" %2B'+userID.toString()+' site:bilibili.com',
})
let vidreq = requestbing({
data: "",
url: 'https://space.bilibili.com/'+userID.toString(),
})
let artreq = requestbing({
data: "",
url: 'https://space.bilibili.com/'+userID.toString()+'/article',
})
let favreq = requestbing({
data: "",
url: 'https://space.bilibili.com/'+userID.toString()+'/favlist',
})
let followingRequest = request({
data: "",
url: followingApiUrl + userID,
})
try {
//console.log(spaceApiUrl + userID)
try{
let spaceContent = await spaceRequest
if (!printed) {
console.log(spaceContent)
printed = true
}
// 动态内容检查
if (spaceContent.code == 0) {
// 解析并拼接动态数据
let st = JSON.stringify(spaceContent.data.items)
//alert(bingContent)
for (let setting of checkers) {
// 检查动态内容
if (setting.keywords) {
if (setting.keywords.find(keyword => st.includes(keyword))) {
if (found.indexOf(setting) < 0)
found.push(setting)
continue;
}
}
}
}
}catch(error){
console.log("no")
}
let bingContent = await bingRequest
let vidContent = await vidreq
let artContent = await artreq
let favContent = await favreq
// 动态内容检查
if (vidContent.code == 0) {
// 解析并拼接动态数据
let st = vidContent+" "+artContent+" "+favContent+" "+element.parent().attr('text')+" "+name
//alert(bingContent)
for (let setting of checkers) {
// 检查动态内容
if (setting.keywords) {
if (setting.keywords.find(keyword => st.includes(keyword))) {
if (found.indexOf(setting) < 0)
found.push(setting)
continue;
}
}
}
}
// 动态内容检查
if (bingContent.code == 0) {
// 解析并拼接动态数据
let st = bingContent
for (let setting of checkers) {
// 检查动态内容
if (setting.keywords) {
if (setting.keywords.find(keyword => st.includes(keyword))) {
if (found.indexOf(setting) < 0 && setting.displayName != "兔友" && setting.displayName != "Minecraft")
found.push(setting)
continue;
}
}
}
}
} catch(error) {
console.error(`获取 ${name} ${userID} 的动态失败`, error)
}
try {
//console.log(followingApiUrl + userID)
let followingContent = await followingRequest
// 可能无权限
let following = followingContent.code == 0 ? followingContent.data.list.map(it => it.mid) : []
if (following) {
for (let setting of checkers) {
// 检查关注列表
if (setting.followings)
for (let mid of setting.followings) {
if (following.indexOf(mid) >= 0) {
if (found.indexOf(setting) < 0)
found.push(setting)
continue;
}
}
}
}
} catch(error) {
console.error(`获取 ${name} ${userID} 的关注列表失败`, error)
}
try {
for (let setting of checkers) {
// 检查关注列表
if (setting.blacklist)
for (let mid of setting.blacklist) {
if (userID.indexOf(mid) >= 0) {
if (found.indexOf(setting) < 0)
found.push(setting)
continue;
}
}
}
} catch(error) {
console.error(`获取 ${name} ${userID} 是否在命中名单失败`, error)
}
// 添加标签
if (found.length > 0) {
// 输出日志
// console.log(`检测到 ${name} ${userID} 的成分为 `, found.map(it => it.displayName))
dom += `<span>昵称: ${name}<br>UID: ${userID}<br>成分: ${found.map(it => it.displayName)}<br>主页: <a href="https://space.bilibili.com/${userID}" target="_blank" style="color: #fb7299;">space.bilibili.com/${userID}</a></span><br><br>`;
checked[userID] = found
// 给所有用到的地方添加标签
for (let element of checking[userID]) {
for (let setting of found) {
installComposition(userID, element, setting)
}
}
loadingElement.parent().remove()
} else {
// console.log(`检测到 ${name} ${userID} 的成分为 无`)
loadingElement.text('路人')
loadingElement.on('click', function () {
check.showAllUser()
})
}
checked[userID] = found
GM_setValue(userID, found)
delete checking[userID]
resolve(found)
} catch (error) {
console.error(`检测 ${name} ${userID} 的成分失败`, error)
loadingElement.text('失败')
loadingElement.on('click', function () {
check.showAllUser()
})
delete checking[userID]
reject(error)
}
})
}
}
dom = '<div id="Identified"><span id="tips">注:因判断关键词较为广泛,可能会出现识别错误的现象<br>脚本还在测试阶段,喜欢的话还请留下你的评论</span><br><br>' + dom + '</div>';
let check = {
showAllUser() {
Swal.fire({
title: '已识别用户',
html: dom,
icon: 'info',
heightAuto: false,
scrollbarPadding: false,
showCloseButton: true,
confirmButtonText: '关闭'
})
},
}
GM_registerMenuCommand("查看所有已识别用户", () => {
check.showAllUser();
});
function request(option) {
return new Promise((resolve, reject) => {
let requestFunction = GM_xmlhttpRequest ? GM_xmlhttpRequest : GM.xmlHttpRequest
requestFunction({
method: "get",
headers: {
'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36'
},
...option,
onload: (response) => {
let res = JSON.parse(response.responseText)
resolve(res)
},
onerror: (error) => {
reject(error);
}
});
})
}
function requestbing(option) {
return new Promise((resolve, reject) => {
let requestFunction = GM_xmlhttpRequest ? GM_xmlhttpRequest : GM.xmlHttpRequest
requestFunction({
method: "get",
headers: {
'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36'
},
...option,
onload: (response) => {
let res = response.responseText
resolve(res)
},
onerror: (error) => {
reject(error);
}
});
})
}
/*--- waitForKeyElements(): A utility function, for Greasemonkey scripts,
that detects and handles AJAXed content.
Usage example:
waitForKeyElements (
"div.comments"
, commentCallbackFunction
);
//--- Page-specific function to do what we want when the node is found.
function commentCallbackFunction (jNode) {
jNode.text ("This comment changed by waitForKeyElements().");
}
IMPORTANT: This function requires your script to have loaded jQuery.
*/
function waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector) {
var targetNodes, btargetsFound;
if (typeof iframeSelector == "undefined")
targetNodes = $(selectorTxt);
else
targetNodes = $(iframeSelector).contents()
.find(selectorTxt);
if (targetNodes && targetNodes.length > 0) {
btargetsFound = true;
targetNodes.each(function () {
var jThis = $(this);
var alreadyFound = jThis.data('alreadyFound') || false;
if (!alreadyFound) {
//--- Call the payload function.
var cancelFound = actionFunction(jThis);
if (cancelFound) btargetsFound = false;
else jThis.data('alreadyFound', true);
}
});
} else {
btargetsFound = false;
}
//--- Get the timer-control variable for this selector.
var controlObj = waitForKeyElements.controlObj || {};
var controlKey = selectorTxt.replace(/[^\w]/g, "_");
var timeControl = controlObj[controlKey];
//--- Now set or clear the timer as appropriate.
if (btargetsFound && bWaitOnce && timeControl) {
//--- The only condition where we need to clear the timer.
clearInterval(timeControl);
delete controlObj[controlKey]
} else {
//--- Set a timer, if needed.
if (!timeControl) {
timeControl = setInterval(function () {
waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector);
}, 300);
controlObj[controlKey] = timeControl;
}
}
waitForKeyElements.controlObj = controlObj;
}
})