仙家军成分查询Helper

用于标记仙家军和动态转发仙以及使用仙话术的b站用户。可能存在误伤,请注意辨别。标记仅供参考,不建议直接作为“依据”使用。

  1. // ==UserScript==
  2. // @name 仙家军成分查询Helper
  3. // @namespace www.bilibili.com
  4. // @version 1.7.3
  5. // @description 用于标记仙家军和动态转发仙以及使用仙话术的b站用户。可能存在误伤,请注意辨别。标记仅供参考,不建议直接作为“依据”使用。
  6. // @author Darknights
  7. // @match *://*.bilibili.com/*
  8. // @match *://*.biligame.com/detail/?id=*
  9. // @exclude *://message.bilibili.com/*
  10. // @exclude *://manga.bilibili.com/*
  11. // @exclude *://www.bilibili.com/correspond/*
  12. // @exclude *://www.bilibili.com/page-proxy/*
  13. // @exclude *://live.bilibili.com/*
  14. // @exclude *://search.bilibili.com/*
  15. // @icon *://static.hdslb.com/images/favicon.ico
  16. // @connect bilibili.com
  17. // @connect biligame.com
  18. // @connect fastly.jsdelivr.net
  19. // @connect raw.githubusercontent.com
  20. // @grant GM_xmlhttpRequest
  21. // @grant GM_info
  22. // @grant GM_registerMenuCommand
  23. // @grant GM_unregisterMenuCommand
  24. // @grant GM_setValue
  25. // @grant GM_getValue
  26. // @license MIT
  27. // @run-at document-end
  28. // ==/UserScript==
  29.  
  30. 'use strict';
  31.  
  32. // 以下为网络名单
  33. let xianLists = [];
  34. let xianFavList = [];
  35. let wordLists = [];
  36. let xianLeakList = [];
  37. let ignoreList = [];
  38. let aidList = [];
  39.  
  40. //以下为本地名单,请在脚本编辑器-存储区自行添加,关键词列表均为字符串形式,注意要转义反斜杠
  41. // 大部分为仙,少数可能有误判
  42. const localXianList = GM_getValue("localXianList", []);
  43.  
  44. // 转发者常见仙的,包含且不限于一些up主/被仙缠上的人等等
  45. const localXianFavList = GM_getValue("localXianFavList", []);
  46.  
  47. // 无视官号和无关号的动态,防止匹配到关键词浪费标签
  48. const localIgnoreList = GM_getValue("localIgnoreList", []);
  49.  
  50. // 仙可能会用的词汇
  51. const localXianWordList = GM_getValue("localXianWordList", []);
  52.  
  53. // 被开盒者隐私信息,需要特殊处理故与关键词列表区分
  54. const localXianLeakList = GM_getValue("localXianLeakList", []);
  55.  
  56. // 辅助,因为有些正则匹配返回值为空
  57. const localAidList = GM_getValue("localAidList", []);
  58.  
  59. const recordMap = new Map();
  60. const uidSet = new Set();
  61.  
  62. const xianTags = [["仙", "#11DD77"], ["仙Ⅰ", "#11DD77"], ["仙Ⅱ", "#11DD77"], ["仙Ⅲ", "#11DD77"]];
  63. const localXianTag = ["仙(本地)", "#11DD77"];
  64.  
  65. const xianRepostTags = [["转发仙:", "#1E971E"], ["转发仙Ⅰ:", "#1E971E"], ["转发仙Ⅱ:", "#1E971E"], ["转发仙Ⅲ:", "#1E971E"]];
  66. const localXianRepostTag = ["转发仙(本地):", "#1E971E"];
  67.  
  68. const favRepostTag = ["转发:", "#2C9EFF"];
  69. const localFavRepostTag = ["转发(本地):", "#2C9EFF"];
  70.  
  71. const wordTags = [["命中:", "#04AEAB"], ["命中Ⅰ:", "#04AEAB"], ["命中Ⅱ:", "#04AEAB"], ["命中Ⅲ:", "#04AEAB"]];
  72. const localXianWordTag = ["命中(本地):", "#04AEAB"];
  73.  
  74. const errorTag = ["出错", "#FF3434"];
  75.  
  76. const captchaTag = ["出错,点此消除", "#FF3434"];
  77.  
  78. const refreshTag = ["然后点此🔄", "#FF7B00"];
  79.  
  80. const newVerTag = ["*已有新版本*", "#990CD0"];
  81.  
  82. const BLOG_URL = "https://api.bilibili.com/x/polymer/web-dynamic/v1/feed/space?&host_mid=";
  83.  
  84. const BILI_URL = "https://t.bilibili.com/";
  85.  
  86. const SCRIPT_URL = "https://gf.qytechs.cn/zh-CN/scripts/467197";
  87.  
  88. const PRIVATE_TIPS = "*已隐藏,注意可能是无关话题被匹配*";
  89.  
  90. const XIAN_MATCH_TIPS = ["*未定级的仙或其拥护者、跟风者,可能存在误判,请注意辨别*", "*主要在评论区活动的仙,或仙的拥护者、跟风者,可能存在误判,请注意辨别*", "*此类仙呈抱团趋势,大量制造暗区动态,或拥有较多粉丝,有一定号召力*", "*此类仙大量制造暗区视频、专栏,或存在开盒行为,或是粉丝量极高,已然成为仙的意见领袖*"];
  91.  
  92. const NEW_VERSION_TIPS = "*点击跳转安装页,更多功能尽在新版本*";
  93.  
  94. const BIG_V_CLASS = ".bili-avatar-icon-business,.bili-avatar-icon-personal,.bili-avatar-icon--business,.bili-avatar-icon--personal,.local-3,.local-4";
  95.  
  96. const CheckType = {
  97. Profile: "PROFILE",
  98. Comment: "COMMENT",
  99. At: "AT",
  100. Reference: "REFER",
  101. Repo: "REPO",
  102. Follow: "FOLLOW",
  103. GameComment: "GAME",
  104. Like: "LIKE"
  105. }
  106.  
  107. const urlSourceDic = {
  108. githubusercontent: "https://raw.githubusercontent.com/Darknights1750/XianLists/main/xianLists.json",
  109. jsdelivr: "https://fastly.jsdelivr.net/gh/Darknights1750/XianLists@main/xianLists.json"
  110. }
  111.  
  112. const statusDic = {
  113. 0: "脚本已暂停⚠️",
  114. 1: "脚本运行中✅"
  115. }
  116.  
  117. const loadingSvg = `<svg t="1734440955984" class="loading-xian" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5163" id="mx_n_1734440955985" width="18" height="18"><path d="M512 61.44a40.96 40.96 0 0 1 40.96 40.96v122.88a40.96 40.96 0 1 1-81.92 0V102.4a40.96 40.96 0 0 1 40.96-40.96z" fill="#06dbff" p-id="5164"></path><path d="M737.28 121.79456a40.96 40.96 0 0 1 14.99136 55.95136l-61.44 106.43456a40.96 40.96 0 1 1-70.94272-40.96l61.44-106.43456A40.96 40.96 0 0 1 737.28 121.79456z" fill="#06dbff" p-id="5165"></path><path d="M902.20544 286.72a40.96 40.96 0 0 1-14.99136 55.95136l-106.43456 61.44a40.96 40.96 0 0 1-40.96-70.94272l106.43456-61.44a40.96 40.96 0 0 1 55.95136 14.99136z" fill="#06dbff" p-id="5166"></path><path d="M962.56 512a40.96 40.96 0 0 1-40.96 40.96h-122.88a40.96 40.96 0 1 1 0-81.92h122.88a40.96 40.96 0 0 1 40.96 40.96z" fill="#06dbff" p-id="5167"></path><path d="M902.20544 737.28a40.96 40.96 0 0 1-55.95136 14.99136l-106.43456-61.44a40.96 40.96 0 1 1 40.96-70.94272l106.43456 61.44A40.96 40.96 0 0 1 902.20544 737.28z" fill="#06dbff" p-id="5168"></path><path d="M737.28 902.20544a40.96 40.96 0 0 1-55.95136-14.99136l-61.44-106.43456a40.96 40.96 0 0 1 70.94272-40.96l61.44 106.43456A40.96 40.96 0 0 1 737.28 902.20544z" fill="#06dbff" p-id="5169"></path><path d="M512 962.56a40.96 40.96 0 0 1-40.96-40.96v-122.88a40.96 40.96 0 1 1 81.92 0v122.88a40.96 40.96 0 0 1-40.96 40.96z" fill="#06dbff" p-id="5170"></path><path d="M286.72 902.20544a40.96 40.96 0 0 1-14.99136-55.95136l61.44-106.43456a40.96 40.96 0 1 1 70.94272 40.96l-61.44 106.43456a40.96 40.96 0 0 1-55.95136 14.99136z" fill="#06dbff" p-id="5171"></path><path d="M121.79456 737.28a40.96 40.96 0 0 1 14.99136-55.95136l106.43456-61.44a40.96 40.96 0 0 1 40.96 70.94272l-106.43456 61.44A40.96 40.96 0 0 1 121.79456 737.28z" fill="#06dbff" p-id="5172"></path><path d="M61.44 512a40.96 40.96 0 0 1 40.96-40.96h122.88a40.96 40.96 0 1 1 0 81.92H102.4a40.96 40.96 0 0 1-40.96-40.96z" fill="#06dbff" p-id="5173"></path><path d="M121.79456 286.72a40.96 40.96 0 0 1 55.95136-14.99136l106.43456 61.44a40.96 40.96 0 1 1-40.96 70.94272l-106.43456-61.44A40.96 40.96 0 0 1 121.79456 286.72z" fill="#06dbff" p-id="5174"></path><path d="M286.72 121.79456a40.96 40.96 0 0 1 55.95136 14.99136l61.44 106.43456a40.96 40.96 0 0 1-70.94272 40.96l-61.44-106.43456A40.96 40.96 0 0 1 286.72 121.79456z" fill="#06dbff" p-id="5175"></path></svg>`
  118.  
  119. let updateTime;
  120. let onlineVersion;
  121. let isLatestVersion = false;
  122. let clearAllMenuId;
  123. let urlSourceMenuId;
  124. let statusMenuId;
  125.  
  126.  
  127. const commandClearTags = function (element) {
  128. Array.prototype.slice.call(element.getElementsByClassName('xian')).forEach(item => item.remove());
  129. Array.prototype.slice.call(element.getElementsByClassName('xian-fail')).forEach(item => item.remove());
  130. }
  131. const commandClearAllTags = function () {
  132. commandClearTags(document);
  133. }
  134.  
  135. const commandUrlSource = function () {
  136. GM_unregisterMenuCommand(urlSourceMenuId);
  137. if ("jsdelivr" === GM_getValue("urlSource", "jsdelivr")) {
  138. GM_setValue("urlSource", "githubusercontent");
  139. } else {
  140. GM_setValue("urlSource", "jsdelivr");
  141. }
  142. urlSourceMenuId = GM_registerMenuCommand("切换数据源(刷新生效)|当前" + GM_getValue("urlSource", "jsdelivr"), commandUrlSource);
  143. }
  144.  
  145. const commandStatus = function () {
  146. GM_unregisterMenuCommand(statusMenuId);
  147. if (1 === GM_getValue("status", 1)) {
  148. GM_setValue("status", 0);
  149. } else {
  150. GM_setValue("status", 1);
  151. }
  152. statusMenuId = GM_registerMenuCommand("暂停/启动脚本|当前" + statusDic[GM_getValue("status", 1)], commandStatus);
  153. }
  154.  
  155. // 初始化存储区、菜单选项
  156. const initSettings = function () {
  157. // 0:不开启,1:开启
  158. if (null === GM_getValue("timeInterval", null)) GM_setValue("timeInterval", 2500); // 标签处理间隔时间 单位:ms
  159. if (null === GM_getValue("testLog", null)) GM_setValue("testLog", 0); // 是否开启调试日志
  160. if (null === GM_getValue("previewLength", null)) GM_setValue("previewLength", 60); // 文本预览长度
  161. if (null === GM_getValue("usingCheckProfile", null)) GM_setValue("usingCheckProfile", 1); // 是否监测个人主页UID
  162. if (null === GM_getValue("usingCheckComments", null)) GM_setValue("usingCheckComments", 1); // 是否监测评论区
  163. if (null === GM_getValue("usingCheckRepos", null)) GM_setValue("usingCheckRepos", 1); // 是否监测转发区
  164. if (null === GM_getValue("usingCheckReferences", null)) GM_setValue("usingCheckReferences", 1); // 是否监测动态被转发者
  165. if (null === GM_getValue("usingCheckAts", null)) GM_setValue("usingCheckAts", 1); // 是否监测@他人
  166. if (null === GM_getValue("usingCheckFollows", null)) GM_setValue("usingCheckFollows", 1); // 是否监测关注/粉丝列表
  167. if (null === GM_getValue("usingCheckGameComments", null)) GM_setValue("usingCheckGameComments", 1); // 是否监测游戏评价区
  168. if (null === GM_getValue("usingCheckLikes", null)) GM_setValue("usingCheckLikes", 1); // 是否监测动态转赞区
  169. if (null === GM_getValue("localXianList", null)) GM_setValue("localXianList", []);
  170. if (null === GM_getValue("localXianFavList", null)) GM_setValue("localXianFavList", []);
  171. if (null === GM_getValue("localIgnoreList", null)) GM_setValue("localIgnoreList", []);
  172. if (null === GM_getValue("localXianWordList", null)) GM_setValue("localXianWordList", []);
  173. if (null === GM_getValue("localXianLeakList", null)) GM_setValue("localXianLeakList", []);
  174. if (null === GM_getValue("localAidList", null)) GM_setValue("localAidList", []);
  175. if (null === GM_getValue("captchaUrl", null)) GM_setValue("captchaUrl", "https://space.bilibili.com/208259/dynamic"); // 输验证码跳转的个人主页,默认叔叔
  176. if (null === GM_getValue("fanLimit", null)) GM_setValue("fanLimit", 250000); // up主粉丝阈值,超过将不再进行匹配(对<仙>无效)
  177. clearAllMenuId = GM_registerMenuCommand("清空本页所有标签", commandClearAllTags);
  178. urlSourceMenuId = GM_registerMenuCommand("切换数据源(刷新生效)|当前" + GM_getValue("urlSource", "jsdelivr"), commandUrlSource);
  179. statusMenuId = GM_registerMenuCommand("暂停/启动脚本|当前" + statusDic[GM_getValue("status", 1)], commandStatus);
  180. }
  181.  
  182.  
  183. const log = function (message) {
  184. return GM_getValue("testLog", 0) ? console.log(message) : null;
  185. };
  186.  
  187. const spawnHtml = function (data, text) {
  188. return `<a class="xian" style='color: ${data[1]} !important' title='${text}' target='_blank' onclick="event.stopPropagation()">&lt;${data[0]}&gt;</a>`;
  189. }
  190.  
  191. const spawnErrorHtml = function (data, text) {
  192. return `<a class="xian-error" style='color: ${data[1]} !important' target='_blank' onclick="event.stopPropagation()">&lt;${data[0]}:${text}&gt;</a>`;
  193. }
  194.  
  195. const spawnCaptchaHtml = function (data) {
  196. return `<a class="xian-fail" style='color: ${data[1]} !important' href=${GM_getValue("captchaUrl", "https://space.bilibili.com/208259/dynamic")} target='_blank' onclick="event.stopPropagation()">&lt;${data[0]}&gt;</a>`;
  197. }
  198.  
  199. const spawnRefreshHtml = function (data) {
  200. return `<a class="xian-fail" style='color: ${data[1]} !important' target='_blank' onclick="event.stopPropagation();refreshTags();">&lt;${data[0]}&gt;</a>`;
  201. }
  202.  
  203. const spawnHtmlWithRef = function (data, word, link, text) {
  204. return `<a class="xian" style='color: ${data[1]} !important' href='${link}' title='${text}' target='_blank' onclick="event.stopPropagation()">&lt;${data[0]}${word}&gt;</a>`;
  205. }
  206.  
  207. // 检测是不是新版
  208. const isNew = function () {
  209. if (location.host === 'space.bilibili.com') {
  210. return true;
  211. }
  212. if (document.getElementsByClassName('item goback').length > 0) {
  213. return true;
  214. }
  215. if (document.getElementsByClassName('app-v1').length > 0) {
  216. return true;
  217. }
  218. if (document.getElementsByClassName('opus-detail').length > 0) {
  219. return true;
  220. }
  221. if (document.getElementsByClassName('bgc').length > 0) {
  222. return true;
  223. }
  224. return false;
  225. };
  226.  
  227. const sleep = (delay) => new Promise((resolve) => setTimeout(resolve, delay));
  228.  
  229. const getXianListOnline = function () {
  230. return new Promise(resolve => {
  231. GM_xmlhttpRequest({
  232. method: "GET",
  233. url: urlSourceDic[GM_getValue("urlSource", "jsdelivr")],
  234. headers: {
  235. 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36'
  236. },
  237. onload: res => {
  238. if (res.status === 200) {
  239. resolve(JSON.parse(res.responseText));
  240. } else {
  241. resolve(JSON.parse('{"xianList":[],"xianFavList":[],"wordLv1List":[],"wordLv2List":[],"wordLv3List":[]}'));
  242. log('[xian-helper]获取远程列表失败!请检查网络情况');
  243. log(res);
  244. }
  245. },
  246. onerror: res => {
  247. log('[xian-helper]访问列表失败!请检查网络情况');
  248. log(res);
  249. }
  250. });
  251. });
  252. }
  253.  
  254. const createScriptFun = function () {
  255. let refreshTagsScript = document.createElement('script');
  256. refreshTagsScript.type = "text/javascript";
  257. refreshTagsScript.innerText = `const refreshTags=function(){Array.prototype.slice.call(document.getElementsByClassName('xian-fail')).forEach(item=>item.remove());}`;
  258. document.head.appendChild(refreshTagsScript);
  259. }
  260.  
  261. const compareVersions = function (curVer, netVer) {
  262. var curArr = curVer.split('.');
  263. var netArr = netVer.split('.');
  264. for (var i = 0; i < Math.max(curArr.length, netArr.length); i++) {
  265. var curNum = parseInt(curArr[i] || 0);
  266. var netNum = parseInt(netArr[i] || 0);
  267. if (curNum < netNum) {
  268. return 1;
  269. }
  270. if (curNum > netNum + 10 || !/^\d+$/.test(curNum) || !/^\d+$/.test(netNum)) {
  271. return 2;
  272. }
  273. if (curNum > netNum) {
  274. return -1;
  275. }
  276. }
  277. return 0;
  278. }
  279.  
  280. const fillLists = async function () {
  281. let json = await getXianListOnline();
  282. xianLists = [
  283. json.xianList,
  284. json.xianLv1List,
  285. json.xianLv2List,
  286. json.xianLv3List
  287. ];
  288. xianFavList = json.xianFavList;
  289. ignoreList = [...localIgnoreList, ...json.ignoreList];
  290. wordLists = [
  291. json.wordLv1List.map((item) => new RegExp(item)),
  292. json.wordLv2List.map((item) => new RegExp(item)),
  293. json.wordLv3List.map((item) => new RegExp(item))
  294. ];
  295. xianLeakList = json.xianLeakList.map((item) => new RegExp(item));
  296. aidList = json.aidList.map((item) => new RegExp(item));
  297. aidList = [...aidList, ...localAidList];
  298. updateTime = json.updateTime;
  299. onlineVersion = json.version;
  300. log(`[xian-helper]>>List update time: ${updateTime}`);
  301. }
  302.  
  303. const runHelper = function () {
  304.  
  305. /* Functions */
  306. const isBlank = function (str) {
  307. if (!str || /^\s*$/.test(str)) return true;
  308. return false;
  309. }
  310.  
  311. const getUidLevel = function (uid, isFav = false, isLocal = false) {
  312. if (isLocal) {
  313. const usingList = isFav ? localXianFavList : localXianList;
  314. return usingList.indexOf(uid) > -1 ? 0 : -1;
  315. }
  316. if (isFav) {
  317. return xianFavList.indexOf(uid) > -1 ? 0 : -1;
  318. }
  319. for (let level = 3; level >= 0; level--) {
  320. const xianList = xianLists[level];
  321. if (xianList.indexOf(uid) > -1) return level;
  322. }
  323. return -1;
  324. }
  325.  
  326. //忽略认证用户和大up主,但不忽略名单里的。html方法只能找到主页、评论和关注的
  327. const isBigUpByHtml = function (checkType, uid) {
  328. if (getUidLevel(uid) > -1) return false;
  329. if (checkType === CheckType.Profile) {
  330. const dynAvater = document.querySelector(`.bili-dyn-item__avatar`);
  331. if (dynAvater) {
  332. const bigV = dynAvater.querySelector(BIG_V_CLASS);
  333. if (bigV) return true;
  334. } else {
  335. const bigV = document.querySelector(`.h-user`).querySelector(BIG_V_CLASS);
  336. if (bigV) return true;
  337. }
  338. const fanCount = document.getElementById("n-fs");
  339. if (fanCount) {
  340. const count = Number(fanCount.computedName?.replace(/\,/g, ""));
  341. return (count ?? 0) >= GM_getValue("fanLimit", 250000);
  342. }
  343. }
  344. if (checkType === CheckType.Comment) {
  345. const bigVNew = document.querySelector(`[data-user-id="${uid}"]`)?.querySelector(BIG_V_CLASS);
  346. const bigVOld = document.querySelector(`[data-usercard-mid="${uid}"]`)?.querySelector(BIG_V_CLASS);
  347. if (bigVNew || bigVOld) return true;
  348. }
  349. if (checkType === CheckType.Follow) {
  350. const bigV = document.querySelector(`[href="//space.bilibili.com/${uid}"]`)?.querySelector(BIG_V_CLASS);
  351. if (bigV) return true;
  352. }
  353. return false;
  354. }
  355.  
  356. //忽略认证用户和大up主,但不忽略名单里的。
  357. const isBigUpByJson = function (moduleAuthor) {
  358. if (getUidLevel(String(moduleAuthor.mid)) > -1) return false;
  359. const verify = moduleAuthor?.official_verify?.type;
  360. return verify != null && verify > -1;
  361. }
  362.  
  363. const getUid = function (htmlEntity, checkType) {
  364. if (checkType === CheckType.Profile) {
  365. return window.location.href.match(/(?<=space\.bilibili\.com\/)\d+/)[0];
  366. }
  367. if (checkType === CheckType.Comment) {
  368. return htmlEntity.dataset?.userId ?? htmlEntity.dataset?.usercardMid ?? htmlEntity.href?.replace(/[^\d]/g, "");
  369. // return htmlEntity.dataset?.userId ?? htmlEntity.dataset?.usercardMid ?? htmlEntity.href?.replace(/[^\d]/g, "") ?? htmlEntity.parentElement?.previousElementSibling?.href?.replace(/[^\d]/g, "");
  370. }
  371. if (checkType === CheckType.Repo) {
  372. return htmlEntity._profile.uid;
  373. }
  374. if (checkType === CheckType.At) {
  375. if (htmlEntity.dataset.oid) return htmlEntity.dataset.oid;
  376. if (htmlEntity.dataset.userId) return htmlEntity.dataset.userId;
  377. if (htmlEntity.dataset.usercardMid) return htmlEntity.dataset.usercardMid;
  378. }
  379. if (checkType === CheckType.Reference) {
  380. return htmlEntity._profile?.uid ?? htmlEntity.href?.replace(/[^\d]/g, "");
  381. }
  382. if (checkType === CheckType.Follow) {
  383. return htmlEntity.parentElement.href.replace(/[^\d]/g, "");
  384. }
  385. if (checkType === CheckType.GameComment) {
  386. return htmlEntity.href.replace(/[^\d]/g, "");
  387. }
  388. if (checkType === CheckType.Like) {
  389. return htmlEntity.parentElement.previousElementSibling._profile.uid;
  390. }
  391. return null;
  392. }
  393.  
  394. const getName = function (htmlEntity, checkType) {
  395. if (checkType === CheckType.Profile) {
  396. return htmlEntity.textContent;
  397. }
  398. if (checkType === CheckType.Comment) {
  399. return htmlEntity.textContent.trim().replace(/[@:]/g, "");
  400. }
  401. if (checkType === CheckType.Repo) {
  402. return htmlEntity.textContent;
  403. }
  404. if (checkType === CheckType.At) {
  405. return htmlEntity.textContent.trim().replace(/[@:]/g, "");
  406. }
  407. if (checkType === CheckType.Reference) {
  408. return htmlEntity.textContent;
  409. }
  410. if (checkType === CheckType.Follow) {
  411. return htmlEntity.textContent;
  412. }
  413. if (checkType === CheckType.GameComment) {
  414. return htmlEntity.textContent;
  415. }
  416. if (checkType === CheckType.Like) {
  417. return htmlEntity.textContent.slice(0, -2);
  418. }
  419. return null;
  420. }
  421.  
  422. const getCommentList = function () {
  423. let oldArray = Array.from(document.querySelectorAll(".user-name,.sub-user-name,.user>.name"));
  424. // let oldArray = document.querySelectorAll(".user-name,.sub-user-name,.user>.name,.bili-user-profile-view__info__uname");
  425. let headerArray1 = Array.from(document.querySelector(".bili-header")?.querySelectorAll(".user-name,.sub-user-name,.user>.name") ?? []);
  426. let headerArray2 = Array.from(document.querySelector(".nav-user-center") ?? []);
  427. let headerArray = [...headerArray1, ...headerArray2];
  428. return oldArray.filter(item => !headerArray.includes(item));
  429. }
  430.  
  431. const getRepoList = function () {
  432. return document.getElementsByClassName('bili-dyn-forward-item__uname');
  433. }
  434.  
  435. const getReferenceList = function () {
  436. return document.querySelectorAll('.dyn-orig-author__name,.original-card-content .username');
  437. }
  438.  
  439. const getLikeList = function () {
  440. return document.getElementsByClassName('reaction-item__name');
  441. }
  442.  
  443. const getFollowList = function () {
  444. return Array.from(document.getElementsByClassName('fans-name'));
  445. }
  446.  
  447. const getGameCommentList = function () {
  448. return Array.from(document.querySelectorAll("a.user-name"));
  449. }
  450.  
  451. const getAtList = function () {
  452. const lst = new Set();
  453. for (let c of document.getElementsByClassName('jump-link user')) {
  454. lst.add(c);
  455. }
  456. for (let c of document.getElementsByClassName('bili-rich-text-module at')) {
  457. lst.add(c);
  458. }
  459. for (let c of document.querySelectorAll('.text-con > a,.text > a')) {
  460. if (c.dataset.usercardMid) lst.add(c);
  461. }
  462. return Array.from(lst);
  463. }
  464.  
  465. const spliceText = function (moduleDynamic) {
  466. let fullTextArr = [];
  467. if (moduleDynamic.topic && !isBlank(moduleDynamic.topic.name)) {
  468. fullTextArr.push(moduleDynamic.topic.name);
  469. }
  470. if (moduleDynamic.desc && !isBlank(moduleDynamic.desc.text)) {
  471. fullTextArr.push(moduleDynamic.desc.text);
  472. }
  473. if (moduleDynamic.major) {
  474. if (moduleDynamic.major.archive) {
  475. if (!isBlank(moduleDynamic.major.archive.title)) {
  476. fullTextArr.push(moduleDynamic.major.archive.title);
  477. }
  478. if (!isBlank(moduleDynamic.major.archive.desc)) {
  479. fullTextArr.push(moduleDynamic.major.archive.desc);
  480. }
  481. }
  482. if (moduleDynamic.major.article) {
  483. if (!isBlank(moduleDynamic.major.article.title)) {
  484. fullTextArr.push(moduleDynamic.major.article.title);
  485. }
  486. if (!isBlank(moduleDynamic.major.article.desc)) {
  487. fullTextArr.push(moduleDynamic.major.article.desc);
  488. }
  489. }
  490. if (moduleDynamic.major.live && !isBlank(moduleDynamic.major.live.title)) {
  491. fullTextArr.push(moduleDynamic.major.live.title);
  492. }
  493. }
  494. if (moduleDynamic.additional && moduleDynamic.additional.ugc && !isBlank(moduleDynamic.additional.ugc.title)) {
  495. fullTextArr.push(moduleDynamic.additional.ugc.title);
  496. }
  497. return fullTextArr.join('//');
  498. }
  499.  
  500. const previewText = function (text, index, len) {
  501. if (!text) return '';
  502. const left = Math.max(0, index - len);
  503. const right = Math.min(text.length, index + len);
  504. let textPart = '';
  505. if (left > 0) {
  506. textPart += '...';
  507. }
  508. textPart += text.substring(left, right).replace(/\n|\r/g, '').trim();
  509. if (right < text.length) {
  510. textPart += '...';
  511. }
  512. return textPart;
  513. }
  514.  
  515. const findRepost = function (items, ownId, matchedDynamicList, isFav, isLocal) {
  516. for (let i = 0; i < items.length; i++) {
  517. const item = items[i];
  518. if (isBigUpByJson(item.modules.module_author)) {
  519. matchedDynamicList.push(String(item.id_str));
  520. continue;
  521. }
  522. if (item.orig) {
  523. if (!isFav && isBigUpByJson(item.orig.modules.module_author)) {
  524. matchedDynamicList.push(String(item.orig.id_str));
  525. continue;
  526. }
  527. const origId = String(item.orig.modules.module_author.mid);
  528. if (origId === ownId) continue;
  529. const uidLevel = getUidLevel(origId, isFav, isLocal);
  530. if (uidLevel > -1) {
  531. const origName = String(item.orig.modules.module_author.name);
  532. const ownFullText = spliceText(item.modules.module_dynamic);
  533. const origFullText = spliceText(item.orig.modules.module_dynamic);
  534. const ownTextPart = previewText(ownFullText, 0, GM_getValue("previewLength", 60));
  535. const origTextPart = previewText(origFullText, 0, GM_getValue("previewLength", 60));
  536. const bothTextPart = `${ownTextPart}//@${origName}:${origTextPart}`;
  537. matchedDynamicList.push(String(item.id_str));
  538. matchedDynamicList.push(String(item.orig.id_str));
  539. return [origId, origName, String(item.id_str), bothTextPart, uidLevel];
  540. }
  541. }
  542. }
  543. return null;
  544. }
  545.  
  546. const hearOne = function (text, usingWordList, level) {
  547. for (const word of usingWordList) {
  548. const matchRes = text.match(word);
  549. if (matchRes) {
  550. let matchStr = matchRes[0];
  551. let matchIndex = matchRes.index;
  552. if (matchStr === '') {
  553. for (const aidWord of aidList) {
  554. const matchAid = text.match(aidWord);
  555. if (matchAid) {
  556. matchStr = matchAid[0];
  557. matchIndex = matchAid.index;
  558. break;
  559. }
  560. }
  561. }
  562. matchStr = matchStr.replace(/\n|\r/g, ' ').trim();
  563. return [matchStr, matchIndex + matchStr.length / 2, level];
  564. }
  565. }
  566. return null;
  567. }
  568.  
  569. const hear = function (text, name, isLocal) {
  570. if (isBlank(text) || ignoreList.indexOf(name) > -1) return null;
  571. const usingLeakList = isLocal ? localXianLeakList : xianLeakList;
  572. if (!isLocal) {
  573. for (let level = 2; level >= 0; level--) {
  574. const wordList = wordLists[level];
  575. const matchRes = hearOne(text, wordList, level);
  576. if (matchRes) return matchRes;
  577. }
  578. } else {
  579. const matchRes = hearOne(text, localXianWordList, -1);
  580. if (matchRes) return matchRes;
  581. }
  582. for (const word of usingLeakList) {
  583. const matchRes = text.match(word);
  584. if (matchRes) return ['可能是盒隐私', -1, -1];
  585. }
  586. return null;
  587. }
  588.  
  589. /**
  590. * 查找关键词
  591. * @param {} items 动态列表
  592. * @returns [关键词,动态id,动态片段]
  593. */
  594. const findWord = function (items, ownName, matchedDynamicList, isLocal) {
  595. for (let i = 0; i < items.length; i++) {
  596. const item = items[i];
  597. //忽略认证用户 但不忽略名单里的
  598. if (isBigUpByJson(item.modules.module_author)) {
  599. matchedDynamicList.push(String(item.id_str));
  600. continue;
  601. }
  602. let level = -1;
  603. let origName;
  604. let ownFullText = spliceText(item.modules.module_dynamic);
  605. let origFullText;
  606. let ownTextPart;
  607. let origTextPart;
  608. let returnWord; // 关键词
  609. let returnPart; // 预览文本
  610. if (matchedDynamicList.indexOf(String(item.id_str)) > -1) {
  611. ownFullText = null;
  612. }
  613. if (item.orig) {
  614. origName = String(item.orig.modules.module_author.name);
  615. origFullText = spliceText(item.orig.modules.module_dynamic);
  616. if (isBigUpByJson(item.orig.modules.module_author)) {
  617. matchedDynamicList.push(String(item.orig.id_str));
  618. origFullText = null;
  619. }
  620. if (matchedDynamicList.indexOf(String(item.orig.id_str)) > -1) {
  621. origFullText = null;
  622. }
  623. }
  624. const ownMatch = hear(ownFullText, ownName, isLocal);
  625. const origMatch = hear(origFullText, origName, isLocal);
  626. const ownLevel = ownMatch ? ownMatch[2] : -2;
  627. const origLevel = origMatch ? origMatch[2] : -2;
  628. if (ownLevel === -2 && origLevel === -2) {
  629. continue;
  630. }
  631. if (ownLevel < origLevel) {
  632. level = origLevel;
  633. returnWord = '🔁' + origMatch[0];
  634. } else {
  635. level = ownLevel;
  636. returnWord = ownMatch[0];
  637. }
  638. const ownIndex = ownMatch ? ownMatch[1] : 0;
  639. const origIndex = origMatch ? origMatch[1] : 0;
  640. ownTextPart = ownIndex < 0 ? PRIVATE_TIPS : previewText(ownFullText, ownIndex, GM_getValue("previewLength", 60) / 2);
  641. returnPart = ownTextPart;
  642. if (!isBlank(origName)) {
  643. origTextPart = origIndex < 0 ? PRIVATE_TIPS : previewText(origFullText, origIndex, GM_getValue("previewLength", 60) / 2);
  644. returnPart = ownTextPart + `//@${origName}:${origTextPart}`;
  645. }
  646. return [returnWord, String(item.id_str), returnPart, level + 1];
  647. }
  648. return null;
  649. }
  650.  
  651. //检查记录
  652. const findRecord = async function (uid, name) {
  653. let oldTag;
  654. if (recordMap.has(uid)) {
  655. oldTag = recordMap.get(uid);
  656. uidSet.delete(uid);
  657. if (oldTag) {
  658. log(`[xian-helper]>>Record:${name}@UID-${uid}>>find>>${oldTag.replaceAll(/<\/?a.*?>/g, "").replaceAll(/&gt;&lt;/g, "、").replaceAll(/&.t;/g, "")}`);
  659. }
  660. } else if (uidSet.has(uid)) {
  661. await sleep(500);
  662. oldTag = findRecord(uid, name);
  663. } else {
  664. uidSet.add(uid);
  665. }
  666. return oldTag;
  667. }
  668.  
  669. function isContain(dom) {
  670. const totalHeight = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight;
  671. const totalWidth = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth;
  672. // 当滚动条滚动时,top、left、bottom、right时刻会发生改变
  673. const { top, right, bottom, left } = dom.getBoundingClientRect();
  674. return left >= 0 && top >= 0 && right <= totalWidth && bottom <= totalHeight;
  675. }
  676.  
  677. const checkEntity = async function (htmlEntity, checkType) {
  678. if (htmlEntity.innerHTML.indexOf(`<span class="xian`) < 0) {
  679. htmlEntity.textContent = htmlEntity.textContent.trim();
  680. if (isContain(htmlEntity)) {
  681. if (htmlEntity.innerHTML.indexOf(`class="loading-xian"`) < 0) {
  682. htmlEntity.innerHTML += loadingSvg;
  683. }
  684. let xianSpan = document.createElement('span');
  685. xianSpan.className = 'xian';
  686. htmlEntity.appendChild(xianSpan);
  687. if (compareVersions(GM_info.script.version, onlineVersion) > 0) {
  688. xianSpan.innerHTML += spawnHtmlWithRef(newVerTag, '', SCRIPT_URL, NEW_VERSION_TIPS);
  689. }
  690. const uid = String(getUid(htmlEntity, checkType));
  691. if (!/^\d+$/.test(uid)) {
  692. log(`[xian-helper]传入UID格式错误:${uid}`);
  693. xianSpan.innerHTML += spawnErrorHtml(errorTag, 'UID格式错误');
  694. Array.prototype.slice.call(htmlEntity.getElementsByClassName('loading-xian')).forEach(item => item.remove());
  695. return;
  696. }
  697. if (isBigUpByHtml(checkType, uid)) {
  698. Array.prototype.slice.call(htmlEntity.getElementsByClassName('loading-xian')).forEach(item => item.remove());
  699. return;
  700. }
  701. const name = getName(htmlEntity, checkType).trim();
  702. if (ignoreList.indexOf(name) > -1) {
  703. Array.prototype.slice.call(htmlEntity.getElementsByClassName('loading-xian')).forEach(item => item.remove());
  704. return;
  705. }
  706. let oldTag = await findRecord(uid, name);
  707.  
  708. if (typeof oldTag === 'string') {
  709. xianSpan.innerHTML += oldTag;
  710. Array.prototype.slice.call(htmlEntity.getElementsByClassName('loading-xian')).forEach(item => item.remove());
  711. } else {
  712. let newTag = '';
  713. const uidLevel = getUidLevel(uid);
  714. if (uidLevel > -1) {
  715. log(`[xian-helper]>>Find Target Lv${uidLevel}:${name}@UID-${uid}>>${checkType}`);
  716. newTag += spawnHtml(xianTags[uidLevel], XIAN_MATCH_TIPS[uidLevel]);
  717. } else if (localXianList.indexOf(uid) > -1) {
  718. log(`[xian-helper]>>Find Local Target:${name}@UID-${uid}>>${checkType}`);
  719. newTag += spawnHtml(localXianTag, XIAN_MATCH_TIPS[0]);
  720. }
  721. GM_xmlhttpRequest({
  722. method: "get",
  723. url: BLOG_URL + uid,
  724. headers: {
  725. 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36'
  726. },
  727. onload: res => {
  728. if (res.status === 200) {
  729. const dynamicJson = JSON.parse(res.response).data;
  730. if (dynamicJson) {
  731. if (dynamicJson.items) {
  732. let matchedDynamicList = [];
  733. const repostMatch = findRepost(dynamicJson.items, uid, matchedDynamicList, false, false);
  734. if (repostMatch) {
  735. log(`[xian-helper]>>Find Repost Lv${repostMatch[4]}:${name}@UID-${uid}>>repost>>${repostMatch[1]}@UID-${repostMatch[0]}>>${checkType}`);
  736. const fixedText = repostMatch[1].length > 12 ? repostMatch[1].slice(0, 9) + '...' : repostMatch[1];
  737. newTag += spawnHtmlWithRef(xianRepostTags[repostMatch[4]], fixedText, BILI_URL + repostMatch[2], repostMatch[3]);
  738. } else {
  739. const localRepostMatch = findRepost(dynamicJson.items, uid, matchedDynamicList, false, true);
  740. if (localRepostMatch) {
  741. log(`[xian-helper]>>Find Local Repost:${name}@UID-${uid}>>repost>>${repostMatch[1]}@UID-${repostMatch[0]}>>${checkType}`);
  742. const fixedText = repostMatch[1].length > 12 ? repostMatch[1].slice(0, 9) + '...' : repostMatch[1];
  743. newTag += spawnHtmlWithRef(localXianRepostTag, fixedText, BILI_URL + repostMatch[2], repostMatch[3]);
  744. }
  745. }
  746. const favRepostMatch = findRepost(dynamicJson.items, uid, matchedDynamicList, true, false);
  747. if (favRepostMatch) {
  748. log(`[xian-helper]>>Find Fav:${name}@UID-${uid}>>repost>>${favRepostMatch[1]}@UID-${favRepostMatch[0]}>>${checkType}`);
  749. const fixedText = favRepostMatch[1].length > 12 ? favRepostMatch[1].slice(0, 9) + '...' : favRepostMatch[1];
  750. newTag += spawnHtmlWithRef(favRepostTag, fixedText, BILI_URL + favRepostMatch[2], favRepostMatch[3]);
  751. } else {
  752. const localFavRepostMatch = findRepost(dynamicJson.items, uid, matchedDynamicList, true, true);
  753. if (localFavRepostMatch) {
  754. log(`[xian-helper]>>Find Local Fav:${name}@UID-${uid}>>repost>>${favRepostMatch[1]}@UID-${favRepostMatch[0]}>>${checkType}`);
  755. const fixedText = favRepostMatch[1].length > 12 ? favRepostMatch[1].slice(0, 9) + '...' : favRepostMatch[1];
  756. newTag += spawnHtmlWithRef(localFavRepostTag, fixedText, BILI_URL + favRepostMatch[2], favRepostMatch[3]);
  757. }
  758. }
  759. const wordMatch = findWord(dynamicJson.items, name, matchedDynamicList, false);
  760. if (wordMatch) {
  761. log(`[xian-helper]>>Find Word:${name}@UID-${uid}>>say>>${wordMatch[0]}>>${checkType}`);
  762. const fixedText = wordMatch[0].length > 12 ? wordMatch[0].slice(0, 9) + '...' : wordMatch[0];
  763. newTag += spawnHtmlWithRef(wordTags[wordMatch[3]], fixedText, BILI_URL + wordMatch[1], wordMatch[2]);
  764. } else {
  765. const localWordMatch = findWord(dynamicJson.items, name, matchedDynamicList, true);
  766. if (localWordMatch) {
  767. log(`[xian-helper]>>Find Local Word:${name}@UID-${uid}>>say>>${wordMatch[0]}>>${checkType}`);
  768. const fixedText = wordMatch[0].length > 12 ? wordMatch[0].slice(0, 9) + '...' : wordMatch[0];
  769. newTag += spawnHtmlWithRef(localXianWordTag, fixedText, BILI_URL + wordMatch[1], wordMatch[2]);
  770. }
  771. }
  772. }
  773. xianSpan.innerHTML += newTag;
  774. recordMap.set(uid, newTag);
  775. } else {
  776. xianSpan.className = 'xian-fail';
  777. newTag = newTag.replace(/"xian"/g, '"xian-fail"');
  778. xianSpan.innerHTML += newTag;
  779. xianSpan.innerHTML += spawnCaptchaHtml(captchaTag);
  780. xianSpan.innerHTML += spawnRefreshHtml(refreshTag);
  781. uidSet.delete(uid);
  782. log('[xian-helper]仙家军成分查询Helper get dynamic fail...');
  783. log(htmlEntity);
  784. log(res);
  785. }
  786. } else {
  787. xianSpan.className = 'xian-error';
  788. newTag = newTag.replace(/"xian"/g, '"xian-error"');
  789. xianSpan.innerHTML += newTag;
  790. xianSpan.innerHTML += spawnErrorHtml(errorTag,"网络请求异常");
  791. log('[xian-helper]仙家军成分查询Helper request fail...');
  792. log(htmlEntity);
  793. log(res);
  794. }
  795. Array.prototype.slice.call(htmlEntity.getElementsByClassName('loading-xian')).forEach(item => item.remove());
  796. },
  797. });
  798. }
  799. }
  800. } else {
  801. Array.prototype.slice.call(htmlEntity.getElementsByClassName('loading-xian')).forEach(item => item.remove());
  802. }
  803. }
  804.  
  805. const checkComments = function () {
  806. const commentlist = getCommentList();
  807. if (commentlist && commentlist.length > 0) {
  808. commentlist.forEach(htmlEntity => {
  809. checkEntity(htmlEntity, CheckType.Comment);
  810. });
  811. }
  812. }
  813.  
  814. const checkRepos = function () {
  815. const repolist = getRepoList();
  816. if (repolist && repolist.length > 0) {
  817. repolist.forEach(htmlEntity => {
  818. checkEntity(htmlEntity, CheckType.Repo);
  819. });
  820. }
  821. }
  822.  
  823. const checkAts = function () {
  824. const atList = getAtList();
  825. if (atList && atList.length > 0) {
  826. atList.forEach(htmlEntity => checkEntity(htmlEntity, CheckType.At));
  827. }
  828. }
  829.  
  830. const checkReferences = function () {
  831. const referenceList = getReferenceList();
  832. if (referenceList && referenceList.length > 0) {
  833. referenceList.forEach(htmlEntity => checkEntity(htmlEntity, CheckType.Reference));
  834. }
  835. }
  836.  
  837. const checkFollows = function () {
  838. const followList = getFollowList();
  839. if (followList && followList.length > 0) {
  840. followList.forEach(htmlEntity => checkEntity(htmlEntity, CheckType.Follow));
  841. }
  842. }
  843.  
  844. const checkProfile = async function () {
  845. let htmlEntity = document.getElementById('h-name');
  846. if (htmlEntity) checkEntity(htmlEntity, CheckType.Profile);
  847. }
  848.  
  849. const checkGameComments = async function () {
  850. const gameCommentList = getGameCommentList();
  851. if (gameCommentList && gameCommentList.length > 0) {
  852. gameCommentList.forEach(htmlEntity => checkEntity(htmlEntity, CheckType.GameComment));
  853. }
  854. }
  855.  
  856. const checkLikes = async function () {
  857. const LikeList = getLikeList();
  858. if (LikeList && LikeList.length > 0) {
  859. LikeList.forEach(htmlEntity => checkEntity(htmlEntity, CheckType.Like));
  860. }
  861. }
  862.  
  863. log(`[xian-helper]start finding targets...`);
  864.  
  865. setInterval(() => {
  866. if (GM_getValue("status", 1)) {
  867. if (location.host === "www.biligame.com") {
  868. if (GM_getValue("usingCheckGameComments", 1)) {
  869. checkGameComments();
  870. }
  871. } else {
  872. if (GM_getValue("usingCheckComments", 1)) {
  873. checkComments();
  874. }
  875. if (GM_getValue("usingCheckRepos", 1)) {
  876. checkRepos();
  877. }
  878. if (GM_getValue("usingCheckReferences", 1)) {
  879. checkReferences();
  880. }
  881. if (GM_getValue("usingCheckAts", 1)) {
  882. checkAts();
  883. }
  884. if (GM_getValue("usingCheckLikes", 1)) {
  885. checkLikes();
  886. }
  887. if (location.host === "space.bilibili.com") {
  888. if (GM_getValue("usingCheckProfile", 1)) {
  889. checkProfile();
  890. }
  891. if (GM_getValue("usingCheckFollows", 1)) {
  892. checkFollows();
  893. }
  894. }
  895. }
  896. } else {
  897. Array.prototype.slice.call(document.body.getElementsByClassName('loading-xian')).forEach(item => item.remove());
  898. }
  899. const zhuangbans = document.querySelectorAll('.sailing');
  900. for (const zhuangban of zhuangbans) {
  901. if (zhuangban.style.pointerEvents !== "none") {
  902. zhuangban.style.pointerEvents = "none";
  903. }
  904. }
  905. }, GM_getValue("timeInterval", 2500));
  906.  
  907. }
  908.  
  909. const fitBiliHelper = function () {
  910. if (!document.getElementById("helper-card")) {
  911. const args = Array.from(arguments).slice(0, arguments.length);
  912. setTimeout(fitBiliHelper, GM_getValue("timeInterval", 2500));
  913. return;
  914. }
  915. const target = document.querySelector('.helper-card-face.face');
  916. const observer = new MutationObserver(function (mutations) {
  917. mutations.map(function (mutation) {
  918. commandClearTags(document.getElementById("helper-card"));
  919. });
  920. });
  921.  
  922. observer.observe(target, { attributeFilter: ["src"] });
  923. }
  924.  
  925. const start = async function () {
  926. log(`[xian-helper]仙家军成分查询Helper v${GM_info.script.version},启动!
  927. [xian-helper]>>isNew: ${isNew()}
  928. [xian-helper]>>Loading: ${window.location.href}
  929. [xian-helper]>>urlSource: ${GM_getValue("urlSource", "jsdelivr")}`);
  930. initSettings();
  931. createScriptFun();
  932. await fillLists();
  933. runHelper();
  934. fitBiliHelper();
  935. }
  936.  
  937. start();

QingJ © 2025

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