// ==UserScript==
// @name Bilibili评论成分识别&屏蔽
// @namespace analyzer.bilibili
// @version 2.15
// @description 标注成分&自动屏蔽
// @author 星之所向i
// @match *://www.bilibili.com/video/*
// @match *://www.bilibili.com/festival/*
// @match *://www.bilibili.com/read/*
// @match *://www.bilibili.com/blackboard/*
// @match *://www.bilibili.com/list/watchlater*
// @match *://www.bilibili.com/list/ml*
// @match *://www.bilibili.com/v/topic/*
// @match *://www.bilibili.com/bangumi/*
// @match *://www.bilibili.com/opus/*
// @match *://t.bilibili.com/*
// @match *://space.bilibili.com/*/dynamic*
// @match *://live.bilibili.com/*
// @match *//live.bilibili.com/blanc/*?liteVersion=*
// @exclude *://www.bilibili.com/video/online.html*
// @exclude *://www.bilibili.com/
// @exclude *://live.bilibili.com/p/*
// @exclude *//t.bilibili.com/h5/dynamic/*
// @icon https://static.hdslb.com/images/favicon.ico
// @connect bilibili.com
// @grant GM_xmlhttpRequest
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_notification
// @noframes
// @run-at document-end
// @license MIT
// ==/UserScript==
const script = async () => {
'use strict';
let
/** MODE 模式
* @type { '自动' | '静默' }
*/
MODE = '自动',
/** 检测功能启用状态
* @type { boolean }
*/
DET = true,
/** 屏蔽功能启用状态
* @type { boolean }
*/
DEL = false,
/** 屏蔽样式
* @type { '完全删除' | '消息屏蔽' }
*/
DEL_STYLE = '消息屏蔽',
/** 自动模式下每秒最大检测次数, 不含已检测过成分的用户
* @type { number }
*/
LIMIT_EVENT = 10,
/** 标签收纳阈值(字节), 超过自动收纳
* @type { number }
*/
TAG_MAX_LENGTH = 20,
/** 渲染最短间隔时间(毫秒)
* @type { number }
*/
INTERVAL = 0,
/** 检测依据 粉丝牌/关注列表/投稿列表/动态空间/评论文本
* @type { { [ key: string ]: boolean} }
*/
DET_BASE = {
DET_BASE_MedalWall: true,
DET_BASE_FOLLOWINGS: true,
DET_BASE_VIDEOS: true,
DET_BASE_DYNAMIC: true,
DET_BASE_COMMENT: true,
},
/** 粉丝牌/关注列表/投稿列表/动态空间API 过载状态
* @type { { [ key: string ]: { Timer: number | null, state: boolean, timestamp: number } } }
*/
API_OVERLOAD = {
DET_BASE_MedalWall: { Timer: null, state: false, timestamp: 0 },
DET_BASE_FOLLOWINGS: { Timer: null, state: false, timestamp: 0 },
DET_BASE_VIDEOS: { Timer: null, state: false, timestamp: 0 },
DET_BASE_DYNAMIC: { Timer: null, state: false, timestamp: 0 },
}
/** <===================================================================================设置面板==========================================================================================> */
let navPanelVis = false, ctab = true, ckey = undefined, ciframe = undefined, animationState = false, init = false, newRuleKeysSet = new Set(), deleteRuleKeysSet = new Set(), newAntiRuleSets = new Set(), deleteAntiRuleSets = new Set(), settingTemp = {}, timerTab1 = null, timerTab2 = null, timerFoot = null, timerFoot2 = null, timerAnimaltion = null
/** 重新封装history行为
* @param { string } type histroy行为类型
* @returns { (...args: any[]) => any }
*/
const bindHistoryEvent = (type) => {
const historyEvent = history[type]
return function () {
const newEvent = historyEvent.apply(this, arguments)
const e = new Event(type)
e.arguments = arguments
window.dispatchEvent(e)
return newEvent
}
}
/** 等待
* @param { number } wait 等待时长
* @returns { Promise<void> }
*/
const delay = (wait) => new Promise((r) => setTimeout(r, wait))
/** 基于消息队列的事件节流器 */
class Throttler {
/** 构造器参数
* @param { number } limit 单位时间内事件执行的最大次数
* @param { number } interval 单位时间长度
* @param { ( object ) => boolean } callback 回调函数
*/
constructor(limit, interval, callback) {
this._limit = limit
this._count = 0
this._interval = interval
this._ispending = true
this._ftimestamp = Date.now() + interval
this._timer = null
this._callback = callback
this._queue = []
this._queue.__proto__ = Object.create(Array.prototype)
this._queue.__proto__.push = (...args) => { Array.prototype.push.call(this._queue, ...args); if (this._ispending && this._count < this._limit) this.deal() }
this._queue.__proto__.unshift = (...args) => { Array.prototype.unshift.call(this._queue, ...args); if (this._ispending && this._count < this._limit) this.deal() }
}
/** 加入队列
* @param { object } data 处理回调事件所需的数据对象
* @param { boolean } isJump 插队到消息队列最前端
* @returns { void }
*/
enqueue = (data, isJump) => isJump ? this._queue.unshift(data) : this._queue.push(data)
/** 从队列最前面移除并返回元素
* @returns { object }
*/
dequeue = () => this._queue.shift()
/** 重置计数与时间戳
* @returns { void }
*/
reset = () => {
this._count = 0
this._ftimestamp = Date.now() + this._interval
}
/** 重置限制次数
* @returns { void }
*/
resetlimit = (newLimit) => {
this._limit = newLimit
this.reset()
}
/** 重置队列
* @returns { void }
*/
resetqueue = () => this._queue.length = 0
/** 处理事件
* @returns { boolean }
*/
deal = () => {
if (Date.now() >= this._ftimestamp) this.reset()
if (this._queue.length > 0 && this._count < this._limit) {
let item = this.dequeue()
if (item) {
if (this._callback(item)) this._count++
return true
} else {
return false
}
} else {
return false
}
}
/** 定时处理消息队列中额定的事件
* @returns { void }
*/
start = () => {
setTimeout(async () => {
while (true) {
this._ispending = false
while (this.deal()) { }
this._ispending = true
await delay(this._ftimestamp - Date.now())
}
}, 0)
}
}
/** 节流
* @param { function } func 函数体
* @param { number } wait 时间间隔
* @param { { leading?: boolean, trailing?: boolean } } options 参数配置
* @returns { [ (...args: any[]) => void, () => void ] }
*/
const _throttle = (func, wait, options = {}) => {
if (typeof (func) !== 'function' || typeof (wait) !== 'number' || typeof (options) !== 'object') return () => { }
let context, args, timer, option = { leading: false, trailing: true, ...options }
return [function () {
context = this
args = arguments
if (option.leading && !timer) {
func.apply(context, args)
if (!option.trailing) {
timer = setTimeout(() => timer = null, wait)
}
}
if (!timer && option.trailing) {
timer = setTimeout(() => {
func.apply(context, args)
timer = null
}, wait)
}
}, () => { clearTimeout(timer); timer = null }]
}
/** 防抖
* @param { function } func 函数体
* @param { number } wait 时间间隔
* @param { { leading?: boolean, maxWait?: number, trailing?: boolean } } options 参数配置
* @returns { [ (...args: any[]) => void, () => void ] }
*/
const _debounce = (func, wait, options = {}) => {
if (typeof (func) !== 'function' || typeof (wait) !== 'number' || typeof (options) !== 'object') return () => { }
let context, args, timer, maxTimr, option = { leading: false, maxWait: undefined, trailing: true, ...options }
return [function () {
context = this
args = arguments
if (option.leading && !timer) {
func.apply(context, args)
}
if (timer) {
clearTimeout(timer)
timer = null
}
if (option.maxWait !== undefined && !maxTimr) {
maxTimr = setTimeout(() => {
func.apply(context, args)
clearTimeout(timer)
timer = null
maxTimr = null
}, option.maxWait)
}
if (!timer && option.trailing) {
timer = setTimeout(() => {
func.apply(context, args)
clearTimeout(maxTimr)
timer = null
}, wait)
}
}, () => { clearTimeout(timer); timer = null }]
}
/** 系统提示
* @param { string } text 提示文本
* @param { string } title 提示标题
* @param { () => void } callback 提示点击回调
* @returns { void }
*/
const showNotification = (text, title, timeout = 0, callback = () => { }) => {
GM_notification({
text,
title,
timeout,
onclick: () => {
window.focus()
callback()
}
})
}
/** 监听对象并在被监听键值改变时执行回调函数
* @param { object } obj 监听对象
* @param { string } key 监听键名
* @param { ( newvalue: string | number ) => void } callback 监听键值变化时调用的回调函数
* @returns { void }
*/
const watch = (obj, key, callback) => {
if (typeof (obj) !== 'object' || typeof (key) !== 'string' || typeof (callback) !== 'function') return
let value = obj[key]
Object.defineProperty(obj, key, {
configurable: true,
enumerable: true,
get: () => value,
set: (newValue) => {
value = newValue
callback(newValue)
}
})
}
/** API过载后禁用15分钟
* @param { string } api 需要禁用的API名
* @param { Element | null } ele 需要修改的元素
* @param { number } time 禁用时间, 默认15分钟
* @param { number } timestamp 禁用截止时间戳
* @returns { void }
*/
const banAPI = (api, ele = null, time = 15 * 60 * 1000, timestamp = Date.now(), isAlert = true) => {
if (isAlert && DET_BASE[api]) showNotification(`检测到API『${api}』频繁调用被B站暂时拉黑, 已自动禁用${(time / 60000).toFixed(2)}分钟`, 'API禁用提醒')
API_OVERLOAD[api].state = true
if (API_OVERLOAD[api].Timer === null) {
API_OVERLOAD[api].timestamp = timestamp
API_OVERLOAD[api].Timer = setTimeout(() => recoverAPI(api, ele, ele?.className), time)
}
if (DET_BASE[api]) {
DET_BASE[api] = false
if (ele) ele.className = 'statusBAN'
}
GM_setValue('xzsx_bilibili_detector_apioverload', JSON.stringify(API_OVERLOAD))
}
/** 恢复API
* @param { string } api 需要恢复的API名
* @param { Element | null } ele 需要修改的元素
* @param { string } preClass 需要修改的元素样式
* @returns { void }
*/
const recoverAPI = (api, ele = null, preClass = undefined) => {
DET_BASE[api] = true
API_OVERLOAD[api].state = false
API_OVERLOAD[api].timestamp = Date.now()
clearTimeout(API_OVERLOAD[api].Timer)
API_OVERLOAD[api].Timer = null
if (ele !== null && preClass !== undefined) ele.className = JSON.parse(JSON.stringify(preClass))
}
/** 深拷贝配置参数
* @param { object } source 配置参数对象
* @returns { object }
*/
const deepCloneRules = (source) => {
let clone = JSON.parse(JSON.stringify(source))
Object.keys(clone.detect).forEach((key) => {
clone.detect[key]['keywords'] = clone.detect[key].hasOwnProperty('keywords') ? new Set([...source.detect[key].keywords]) : new Set()
clone.detect[key]['antikeywords'] = clone.detect[key].hasOwnProperty('antikeywords') ? new Set([...source.detect[key].antikeywords]) : new Set()
})
return clone
}
/** 从本地存储读取配置信息
* @returns { object }
*/
const initRules = () => {
/** 初始配置 */
const initConf = {
blackList:
{
'原神': '基于成分',
'明日方舟': '基于成分',
'王者荣耀': '基于成分',
'用户昵称': '基于昵称',
/** ... */
},
detect: {
'原神': { 'color': '#000000', 'keywords': new Set(['原神', '莴苣某人', '你的影月月']), 'antikeywords': new Set(['原P', '原批']) }, /** 最好不要填写容易匹配英文单词的关键词 */
'明日方舟': { 'color': '#73ff00', 'keywords': new Set(['明日方舟', 'Wan顽子']), 'antikeywords': new Set(['粥÷', '粥畜']) },
'王者荣耀': { 'color': '#ff00f7', 'keywords': new Set(['王者荣耀', 'AG超玩会王者荣耀梦泪']), 'antikeywords': new Set(['农P', '农批']) },
'战双帕弥什': { 'color': '#f50000', 'keywords': new Set(['战双帕弥什']), 'antikeywords': new Set(['双畜']) },
'崩坏3': { 'color': '#8000ff', 'keywords': new Set(['崩坏3第一偶像爱酱']), 'antikeywords': new Set(['幻官']) },
'Asoul': { 'color': '#000000', 'keywords': new Set(['A-SOUL_Official', '嘉然今天吃什么', '向晚大魔王', '贝拉kira', '乃琳Queen', '顶碗人', '音乐珈', '奶淇淋', '贝极星']), 'antikeywords': new Set(['A畜']) },
'幻塔': { 'color': '#66dbd9', 'keywords': new Set(['幻塔']), 'antikeywords': new Set(['幻官']) },
'棺人痴': { 'color': '#fed06c', 'keywords': new Set(['東雪蓮Official', '東雪蓮', '东雪莲', '棺人痴']), 'antikeywords': new Set(['眠大佐']) },
/** ... */
}
}
try {
let conf = GM_getValue('xzsx_bilibili_detector'), state = GM_getValue('xzsx_bilibili_detector_state'), overload = GM_getValue('xzsx_bilibili_detector_apioverload')
/** 从本地存储恢复配置列表 */
if (typeof (conf) === 'string' && conf) {
conf = JSON.parse(conf)
if (Object.keys(conf).length > 0) {
if (Array.isArray(conf.blackList)) {
let obj = {}
conf.blackList.forEach((key) => obj[key] = '基于昵称')
conf.blackList = obj
}
Object.keys(conf?.detect).forEach((key) => {
conf.detect[key]['keywords'] = conf.detect[key].hasOwnProperty('keywords') ? new Set([...conf.detect[key].keywords.length > 0 ? conf.detect[key].keywords : []]) : new Set()
conf.detect[key]['antikeywords'] = conf.detect[key].hasOwnProperty('antikeywords') ? new Set([...conf.detect[key].antikeywords.length > 0 ? conf.detect[key].antikeywords : []]) : new Set()
})
}
}
/** 初始配置 */
if (!conf || Object.keys(conf).length === 0) conf = initConf
/** 从本地存储恢复状态 */
if (typeof (state) === 'string' && state) {
state = JSON.parse(state)
MODE = state?.MODE ?? (state?.MODE === '自动' || state?.MODE === '静默') ? state?.MODE : '自动'
DET = state?.DET ?? (typeof (state?.DET) === "boolean" ? state?.DET : true)
DEL = state?.DEL ?? (typeof (state?.DEL) === "boolean" ? state?.DEL : false)
DEL_STYLE = state?.DEL_STYLE ?? (state?.DEL_STYLE === '消息屏蔽' || state?.DEL_STYLE === '完全删除') ? state?.DEL_STYLE : '消息屏蔽'
TAG_MAX_LENGTH = state?.TAG_MAX_LENGTH ?? (typeof (state?.TAG_MAX_LENGTH) === "number" ? state?.TAG_MAX_LENGTH : 20)
INTERVAL = state?.INTERVAL ?? (typeof (state?.INTERVAL) === "number" ? state?.INTERVAL : 0)
LIMIT_EVENT = state?.LIMIT_EVENT ?? (typeof (state?.LIMIT_EVENT) === "number" ? state?.LIMIT_EVENT : 10)
DET_BASE['DET_BASE_MedalWall'] = state?.DET_BASE_MedalWall ?? (typeof (state?.DET_BASE_MedalWall) === "boolean" ? state?.DET_BASE_MedalWall : true)
DET_BASE['DET_BASE_FOLLOWINGS'] = state?.DET_BASE_FOLLOWINGS ?? (typeof (state?.DET_BASE_FOLLOWINGS) === "boolean" ? state?.DET_BASE_FOLLOWINGS : true)
DET_BASE['DET_BASE_VIDEOS'] = state?.DET_BASE_VIDEOS ?? (typeof (state?.DET_BASE_VIDEOS) === "boolean" ? state?.DET_BASE_VIDEOS : true)
DET_BASE['DET_BASE_DYNAMIC'] = state?.DET_BASE_DYNAMIC ?? (typeof (state?.DET_BASE_DYNAMIC) === "boolean" ? state?.DET_BASE_DYNAMIC : true)
DET_BASE['DET_BASE_COMMENT'] = state?.DET_BASE_COMMENT ?? (typeof (state?.DET_BASE_COMMENT) === "boolean" ? state?.DET_BASE_COMMENT : true)
}
/** 从本地存储恢复API禁用状态 */
if (typeof (overload) === 'string' && overload) {
overload = JSON.parse(overload)
Object.keys(overload).forEach((api) => {
if (overload[api]?.timestamp) {
let passed = Date.now() - overload[api].timestamp
if (overload[api]?.state && passed < 15 * 60 * 1000) banAPI(api, null, 15 * 60 * 1000 - passed, overload[api].timestamp, false)
}
})
}
return [conf, deepCloneRules(conf)]
} catch (error) {
return [initConf, deepCloneRules(initConf)]
}
}
/** 标记/屏蔽规则
* @type { { blackList: [ key: string ]: string, detect: { [ key: string ]: { color: string, keywords: Set<string>, antikeywords: Set<string> } } } }
*/
let [rulesApply, rules] = initRules()
/** 构建事件节流器实例 */
const _T = new Throttler(LIMIT_EVENT, 1000, ({ uid, user, isdeep, forceSync }) => {
// 查重
if (filte(uid, user)) return false
// 异步处理
setTimeout(() => handleStr(uid, user, isdeep, forceSync), 0)
return true
})
/** 启动事件节流器 */
_T.start()
/** 获取正确的iframe
* 当用户元素存在于iframe内时需要调用
* @returns { HTMLIFrameElement | undefined } iframe存在则返回实例,反之则返回undefined
*/
const getCurIframe = () => {
let matches = []
for (const iframe of document.getElementsByTagName('iframe')) {
matches = iframe.attributes.src.value.match(/\/\/(.*?)\/([a-zA-Z|\d])+\/.*$/)
if (matches !== null) {
if (matches.length > 2 && matches[2] === 'p') continue
if (matches.length > 1 && matches[1] === 'live.bilibili.com') return iframe
}
}
return undefined
}
/** 添加css样式
* @type { HTMLStyleElement }
*/
const style = document.createElement('style')
/** 面板相关css样式
* @type { string }
*/
const css_panel =
`
.nav-show{display:block;outline:0;height:24px;width:40px;margin-bottom:12px;transition:all 0.3s;cursor:pointer;text-align:center;padding:0 4px;position:fixed;bottom:4px;right:6px;color:white;background-color:rgb(75,182,206);box-sizing:border-box;box-shadow:0px 0px 2px #525252;border:0px;user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:14px;transition:all 0.2s;-o-transition:all 0.2s;-moz-transition:all 0.2s;-webkit-transition:all 0.2s;transform:scale(1);-o-transform:scale(1);-ms-transform:scale(1);-moz-transform:scale(1);-webkit-transform:scale(1);appearance:button;-moz-appearance:button;-webkit-appearance:button;z-index:99999;cursor:pointer;opacity:1}
.nav-show:hover{transform:scale(0.8);-o-transform:scale(0.8);-ms-transform:scale(0.8);-moz-transform:scale(0.8);-webkit-transform:scale(0.8);opacity:0.5}
.nav-panel{width:250px;height:425px;display:none;align-items:center;flex-direction:column;justify-content:flex-start;position:fixed;bottom:12px;right:90px;background-color:white;box-shadow:0px 0px 4px #6a6767;user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;z-index:999999;overflow:hidden}
.nav-tabs{display:flex;width:100%;justify-content:space-around;box-shadow:0px 2px 4px #ddd}
.nav-tabs-btn1{flex-grow:1;height:36px;background-color:rgba(22,51,194,0.821);color:white;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;border:0;cursor:pointer;filter:brightness(100%);-o-filter:brightness(100%);-moz-filter:brightness(100%);-webkit-filter:brightness(100%);z-index:1}
.nav-tabs-btn1:hover{opacity:0.8}
.nav-tabs-btn2{flex-grow:1;height:36px;background-color:rgba(49,182,203,0.821);color:white;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;border:0;cursor:pointer;filter:brightness(50%);-o-filter:brightness(50%);-moz-filter:brightness(50%);-webkit-filter:brightness(50%);z-index:1}
.nav-tabs-btn2:hover{opacity:0.8}
.nav-list{display:inline-flex;flex-direction:column;height:calc(100% - 36px);width:100%;overflow:auto}
.rule-item-btn{border:1px solid #d9d9d9;outline:0;height:24px;padding:0 4px;font-size:14px;border-radius:2px;color:#fff;background:#1890ff;border-color:#1890ff;text-shadow:0 -1px 0 rgba(0,0,0,.12);box-shadow:0 2px 0 rgba(0,0,0,.045);appearance:button;-moz-appearance:button;-webkit-appearance:button;cursor:pointer}
.rule-item-btn:hover{opacity:0.8}
.addrule_form{position:absolute;top:36px;width:100%;height:calc(100% - 72px);padding:0;display:none;flex-direction:column;justify-content:flex-start;background-color:rgb(255,255,255);user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;z-index:1;animation:showEditPanel 0.5s forwards;-o-animation:showEditPanel 0.5s forwards;-moz-animation:showEditPanel 0.5s forwards;-webkit-animation:showEditPanel 0.5s forwards;clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%);-o-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%);-moz-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%);-webkit-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%)}
@keyframes showEditPanel{from{clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%)}
to{clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
}@-o-keyframes showEditPanel{from{-o-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%)}
to{-o-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
}@-moz-keyframes showEditPanel{from{-moz-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%)}
to{-moz-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
}@-webkit-keyframes showEditPanel{from{-webkit-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%)}
to{-webkit-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
}.addrule_form-hide{position:absolute;top:36px;width:100%;height:calc(100% - 72px);padding:0;display:none;flex-direction:column;justify-content:flex-start;background-color:rgb(255,255,255);user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;z-index:1;animation:hideEditPanel 0.5s forwards;-o-animation:hideEditPanel 0.5s forwards;-moz-animation:hideEditPanel 0.5s forwards;-webkit-animation:hideEditPanel 0.5s forwards;clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%);-o-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%);-moz-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%);-webkit-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
@keyframes hideEditPanel{from{clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
to{clip-path:polygon(157% 0,157% 0,100% 100%,100% 100%)}
}@-o-keyframes hideEditPanel{from{-o-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
to{-o-clip-path:polygon(157% 0,157% 0,100% 100%,100% 100%)}
}@-moz-keyframes hideEditPanel{from{-moz-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
to{-moz-clip-path:polygon(157% 0,157% 0,100% 100%,100% 100%)}
}@-webkit-keyframes hideEditPanel{from{-webkit-clip-path:polygon(0 0,157% 0,100% 100%,-57% 100%)}
to{-webkit-clip-path:polygon(157% 0,157% 0,100% 100%,100% 100%)}
}.addrule_item1{display:flex;justify-content:space-between;align-items:center;padding:8px;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px}
.rule_keywords{height:50%;width:94%;flex-wrap:wrap;padding:8px;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;overflow-x:hidden;overflow-y:auto}
.tagname_input{border:0;color:black;outline-style:none;height:37px;width:195px;font-size:32px;transition:color 0.5s;-o-transition:color 0.5s;-moz-transition:color 0.5s;-webkit-transition:color 0.5s}
.tagcolor_input{height:30px;width:30px;cursor:pointer}
.xzsx_detect_tagsample_container{display:flex;justify-content:flex-start;align-items:center;height:36px;padding-left:8px;padding-right:8px;padding-top:4px;padding-bottom:2px;box-shadow:0px 0px 8px #ddd inset;overflow:hidden}
.xzsx_detect_tagsample{display:inline-flex;height:14px;line-height:14px;position:relative;box-sizing:content-box !important;padding:1px 2px 1px 14px;margin:1px 4px 1px 0;color:black;white-space:nowrap;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑 !important;font-size:14px;font-weight:600 !important;border-radius:6px 4px 4px 6px;box-shadow:0 0 6px #ddd;text-size-adjust:100%;-ms-text-size-adjust:100%;-moz-text-size-adjust:100%;-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent;transition:all 1s;-o-transition:all 1s;-moz-transition:all 1s;-webkit-transition:all 1s;overflow-wrap:break-word;cursor:pointer}
.xzsx_detect_tagsample:hover{opacity:0.8}
.xzsx_detect_tagsample::before{background:#fff;border-radius:10px;box-shadow:0 1px rgba(0,0,0,0.25) inset;content:'';height:6px;left:4px;position:absolute;width:6px;top:calc(50% - 3px)}
.add_items{display:inline-flex;justify-content:center;align-items:center}
.tagkeyword_input{width:128px;border:0;border-bottom:0.5px solid;outline-style:none;font-size:24px}
.add_icon{display:inline-flex;justify-content:center;align-items:center;height:10px;width:10px;color:rgb(0,0,0);transform:scale(2);-o-transform:scale(2);-ms-transform:scale(2);-moz-transform:scale(2);-webkit-transform:scale(2);transform-origin:25% 55%;-o-transform-origin:25% 55%;-ms-transform-origin:25% 55%;-moz-transform-origin:25% 55%;-webkit-transform-origin:25% 55%;cursor:pointer}
.add_icon:hover{opacity:0.5}
.keyword{display:inline-flex;font-size:24px;padding:4px;animation:showEditPanel 0.5s forwards;-o-animation:showEditPanel 0.5s forwards;-moz-animation:showEditPanel 0.5s forwards;-webkit-animation:showEditPanel 0.5s forwards;clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%);-o-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%);-moz-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%);-webkit-clip-path:polygon(0 0,0 0,-57% 100%,-57% 100%)}
.keyword:hover{background-color:#ddd}
.keyword_delete{position:relative;top:-3px;right:0;width:20px;height:20px;border-radius:50%;text-align:center;transform:scale(0.6) translate(0%,0%);-o-transform:scale(0.6) translate(0%,0%);-ms-transform:scale(0.6) translate(0%,0%);-moz-transform:scale(0.6) translate(0%,0%);-webkit-transform:scale(0.6) translate(0%,0%);transform-origin:center;-o-transform-origin:center;-ms-transform-origin:center;-moz-transform-origin:center;-webkit-transform-origin:center;font-size:20px;color:#837171;line-height:18px;background-color:#b2bfc67a;cursor:pointer}
.keyword_delete:hover{color:white;background-color:crimson}
.keyword_delete:active{color:white;background-color:#f0f}
.rule-item{display:flex;justify-content:space-between;align-items:center;padding:2px 8px 2px 8px;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;transform:scaleX(1);-o-transform:scaleX(1);-ms-transform:scaleX(1);-moz-transform:scaleX(1);-webkit-transform:scaleX(1);transition:all 0.2s;-o-transition:all 0.2s;-moz-transition:all 0.2s;-webkit-transition:all 0.2s;clip-path:polygon(0 0,115% 0,100% 100%,-15% 100%);-o-clip-path:polygon(0 0,115% 0,100% 100%,-15% 100%);-moz-clip-path:polygon(0 0,115% 0,100% 100%,-15% 100%);-webkit-clip-path:polygon(0 0,115% 0,100% 100%,-15% 100%)}
.rule-item:hover{background-color:#ddd;transform:scaleX(0.95);-o-transform:scaleX(0.95);-ms-transform:scaleX(0.95);-moz-transform:scaleX(0.95);-webkit-transform:scaleX(0.95)}
.rule-item-detect-add{font-size:18px;text-align:center;padding:2px 8px 2px 8px;border-left:0;border-right:0;color:blue;cursor:pointer}
.rule-item-detect-add:hover{background-color:#ddd}
.rule-item-blacklist-add{display:none;text-align:center;padding:2px 8px 2px 8px;border-left:0;border-right:0}
.rule-item-blacklist-add-input{height:32px;width:90px;border:0;border-bottom:0.5px solid;outline-style:none;font-size:24px}
.rule-item-blacklist-select{height:32.8px;width:90px;border:0;border-bottom:0.5px solid;outline-style:none;font-size:24px}
.rule-item-blacklist-addicon{display:inline-flex;justify-content:center;align-items:center;height:10px;width:10px;color:rgb(0,0,0);transform:scale(3);-o-transform:scale(3);-ms-transform:scale(3);-moz-transform:scale(3);-webkit-transform:scale(3);transform-origin:25% 5%;-o-transform-origin:25% 5%;-moz-transform-origin:25% 5%;-webkit-transform-origin:25% 5%;cursor:pointer}
.rule-item-blacklist-addicon:hover{opacity:0.5}
.rule-item-blacklist-base{display:flex;justify-content:center;align-items:center;padding:2px 8px 2px 8px;margin-right:4px;color:#299bda;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;transform:scaleX(1);-o-transform:scaleX(1);-ms-transform:scaleX(1);-moz-transform:scaleX(1);-webkit-transform:scaleX(1);transition:all 0.2s;-o-transition:all 0.2s;-moz-transition:all 0.2s;-webkit-transition:all 0.2s;cursor:pointer}
.rule-item-blacklist-base:hover{background-color:#ddd;transform:scaleX(0.95);-o-transform:scaleX(0.95);-moz-transform:scaleX(0.95);-webkit-transform:scaleX(0.95)}
.nav-footer{display:flex;width:100%;position:relative;background-color:red;justify-content:flex-start;box-shadow:0px -2px 4px #ddd}
.nav-footer:hover{opacity:0.8}
.confirm_btn{flex-grow:1;height:36px;background-color:rgb(80,130,220,0.7);color:white;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:20px;border:0;cursor:pointer;z-index:1}
.add_btn{flex-grow:1;height:36px;background-color:rgba(60,234,57,0.821);color:white;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:20px;border:0;cursor:pointer;z-index:1}
.alter_btn{flex-grow:1;height:36px;background-color:rgba(154,36,160,0.821);color:white;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:20px;border:0;cursor:pointer;z-index:1}
.nav-tabs-confirm_btn:hover{opacity:0.8}
.changed-signel1{height:8px;width:8px;border-radius:50%;background-color:white;position:absolute;left:105px;top:14px;z-index:1}
.changed-signel2{height:8px;width:8px;border-radius:50%;background-color:white;position:absolute;right:10px;top:14px;z-index:1}
.statuslayout{height:12px;width:12px;position:absolute;left:8px;top:12px;display:flex;justify-content:center;align-items:center}
.statuslayout2{margin-right:6px;margin-left:8px}
.statusP{height:12px;width:12px;background-color:rgb(33,247,140);border-radius:50%}
.statusE{height:12px;width:12px;background-color:rgb(243,28,28);border-radius:50%}
.statusBAN{height:12px;width:12px;background-color:rgb(246,255,0);border-radius:50%}
.statusPcircle{position:absolute;width:12px;height:12px;border:2px solid rgb(33,247,140);border-radius:50%;animation:Processing 1.2s infinite ease-in-out;-o-animation:Processing 1.2s infinite ease-in-out;-moz-animation:Processing 1.2s infinite ease-in-out;-webkit-animation:Processing 1.2s infinite ease-in-out;opacity:0}
@keyframes Processing{0%{width:12px;height:12px;opacity:0.5}
100%{width:24px;height:24px;opacity:0}
}@-o-keyframes Processing{0%{width:12px;height:12px;opacity:0.5}
100%{width:24px;height:24px;opacity:0}
}@-moz-keyframes Processing{0%{width:12px;height:12px;opacity:0.5}
100%{width:24px;height:24px;opacity:0}
}@-webkit-keyframes Processing{0%{width:12px;height:12px;opacity:0.5}
100%{width:24px;height:24px;opacity:0}
}.process{visibility:hidden;position:absolute;height:36px;width:0%;background-color:rgb(0 53 255);transition:width 0.5s;-o-transition:width 0.5s;-moz-transition:width 0.5s;-webkit-transition:width 0.5s}
.applysuccess{position:absolute;right:36px;top:6px;z-index:2;transform:scale(1.5);-o-transform:scale(1.5);-ms-transform:scale(1.5);-moz-transform:scale(1.5);-webkit-transform:scale(1.5);clip-path:polygon(0 0,0 0,0 100%,0 100%);-o-clip-path:polygon(0 0,0 0,0 100%,0 100%);-moz-clip-path:polygon(0 0,0 0,0 100%,0 100%);-webkit-clip-path:polygon(0 0,0 0,0 100%,0 100%);animation:success 1.5s infinite;-o-animation:success 1.5s infinite;-moz-animation:success 1.5s infinite;-webkit-animation:success 1.5s infinite}
@keyframes success{0%{clip-path:polygon(0 0,0 0,0 100%,0 100%)}
25%{clip-path:polygon(0% 0%,100% 0%,100% 100%,0% 100%)}
100%{clip-path:polygon(100% 0%,100% 0%,100% 100%,100% 100%)}
}@-o-keyframes success{0%{-o-clip-path:polygon(0 0,0 0,0 100%,0 100%)}
25%{-o-clip-path:polygon(0% 0%,100% 0%,100% 100%,0% 100%)}
100%{-o-clip-path:polygon(100% 0%,100% 0%,100% 100%,100% 100%)}
}@-moz-keyframes success{0%{-moz-clip-path:polygon(0 0,0 0,0 100%,0 100%)}
25%{-moz-clip-path:polygon(0% 0%,100% 0%,100% 100%,0% 100%)}
100%{-moz-clip-path:polygon(100% 0%,100% 0%,100% 100%,100% 100%)}
}@-webkit-keyframes success{0%{-webkit-clip-path:polygon(0 0,0 0,0 100%,0 100%)}
25%{-webkit-clip-path:polygon(0% 0%,100% 0%,100% 100%,0% 100%)}
100%{-webkit-clip-path:polygon(100% 0%,100% 0%,100% 100%,100% 100%)}
}.side-btn{display:flex;justify-content:center;align-items:center;position:fixed;right:355px;color:white;border:0;padding:4px;background-color:rgba(22,51,194,0.821);box-shadow:0px 0px 4px #575555;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:16px;cursor:pointer;user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;transform:scale(1);-o-transform:scale(1);-ms-transform:scale(1);-moz-transform:scale(1);-webkit-transform:scale(1);transition:all 0.2s;-o-transition:all 0.2s;-moz-transition:all 0.2s;-webkit-transition:all 0.2s;z-index:999999}
.side-btn:hover{transform:scale(0.8);-o-transform:scale(0.8);-ms-transform:scale(0.8);-moz-transform:scale(0.8);-webkit-transform:scale(0.8);opacity:0.8}
.side-btn-mode{bottom:380px}
.side-btn-import{bottom:340px}
.side-btn-export{bottom:300px}
.navPanel-resetBtn{display:flex;justify-content:center;align-items:center;position:fixed;height:25px;width:25px;bottom:430px;right:56px;color:white;border-radius:0%;border:0;padding:2px;background-color:#11cc83bd;box-shadow:0px 0px 4px #575555;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;cursor:pointer;user-select:none;transform:scale(1);-o-transform:scale(1);-ms-transform:scale(1);-moz-transform:scale(1);-webkit-transform:scale(1);transition:all 0.5s;-o-transition:all 0.5s;-moz-transition:all 0.5s;-webkit-transition:all 0.5s;z-index:999999}
.navPanel-resetBtn:hover{transform:scale(0.8);-o-transform:scale(0.8);-ms-transform:scale(0.8);-moz-transform:scale(0.8);-webkit-transform:scale(0.8);opacity:0.8;border-radius:35%}
.resetBtn-icon{height:25px;width:25px;transform:rotate(0deg);-webkit-transform:rotate(0deg);transform-origin:50% 50%;-webkit-transform-origin:50% 50%;transition:all 0.5s;-o-transition:all 0.5s;-moz-transition:all 0.5s;-webkit-transition:all 0.5s}
.resetBtn-icon:hover{transform:rotate(360deg);-o-transform:rotate(360deg);-ms-transform:rotate(360deg);-moz-transform:rotate(360deg);-webkit-transform:rotate(360deg)}
.block_btn{display:inline-flex;position:relative;left:12px;bottom:-4px;cursor:pointer}
.show-animation{width:344px;height:450px;position:fixed;bottom:12px;right:53px;clip-path:polygon(0 0,0 0,0 100%,0 100%);-o-clip-path:polygon(0 0,0 0,0 100%,0 100%);-moz-clip-path:polygon(0 0,0 0,0 100%,0 100%);-webkit-clip-path:polygon(0 0,0 0,0 100%,0 100%);animation:showPanel 0.5s forwards;-o-animation:showPanel 0.5s forwards;-moz-animation:showPanel 0.5s forwards;-webkit-animation:showPanel 0.5s forwards;opacity:1;z-index:99999}
@keyframes showPanel{from{clip-path:polygon(0 0,100% 0,100% 0,0 0);opacity:1}
to{clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
}@-o-keyframes showPanel{from{-o-clip-path:polygon(0 0,100% 0,100% 0,0 0);opacity:1}
to{-o-clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
}@-moz-keyframes showPanel{from{-moz-clip-path:polygon(0 0,100% 0,100% 0,0 0);opacity:1}
to{-moz-clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
}@-webkit-keyframes showPanel{from{-webkit-clip-path:polygon(0 0,100% 0,100% 0,0 0);opacity:1}
to{-webkit-clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
}.hide-animation{width:342px;height:450px;position:fixed;bottom:12px;right:57px;clip-path:polygon(0 0,0 0,0 100%,0 100%);-o-clip-path:polygon(0 0,0 0,0 100%,0 100%);-moz-clip-path:polygon(0 0,0 0,0 100%,0 100%);-webkit-clip-path:polygon(0 0,0 0,0 100%,0 100%);animation:hidePanel 0.5s forwards;-o-animation:hidePanel 0.5s forwards;-moz-animation:hidePanel 0.5s forwards;-webkit-animation:hidePanel 0.5s forwards;z-index:99999}
@keyframes hidePanel{from{clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
to{clip-path:polygon(0 100%,100% 100%,100% 100%,0 100%);opacity:1}
}@-o-keyframes hidePanel{from{-o-clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
to{-o-clip-path:polygon(0 100%,100% 100%,100% 100%,0 100%);opacity:1}
}@-moz-keyframes hidePanel{from{-moz-clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
to{-moz-clip-path:polygon(0 100%,100% 100%,100% 100%,0 100%);opacity:1}
}@-webkit-keyframes hidePanel{from{-webkit-clip-path:polygon(0 0,100% 0,100% 100%,0 100%);opacity:1}
to{-webkit-clip-path:polygon(0 100%,100% 100%,100% 100%,0 100%);opacity:1}
}.setting-show{display:inline-flex;flex-direction:column;justify-content:flex-start;align-items:center;position:absolute;right:0;top:36px;width:50%;height:0px;background-color:rgba(49,182,203,0.821);z-index:2;overflow:hidden;transition:all 1s;-o-transition:all 1s;-moz-transition:all 1s;-webkit-transition:all 1s}
.setting-btn{display:flex;align-items:center;height:30px;width:100%;padding:0px;background-color:rgba(22,51,194,0.821);color:white;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;border:0;cursor:pointer;transform:scale(1);-o-transform:scale(1);-ms-transform:scale(1);-moz-transform:scale(1);-webkit-transform:scale(1);transition:all 0.2s;-o-transition:all 0.2s;-moz-transition:all 0.2s;-webkit-transition:all 0.2s}
.setting-btn:hover{transform:scale(0.8);-o-transform:scale(0.8);-ms-transform:scale(0.8);-moz-transform:scale(0.8);-webkit-transform:scale(0.8);opacity:0.8}
.setting-text{display:flex;justify-content:center;align-items:center;flex-direction:column;height:68px;width:100%;background-color:rgba(22,51,194,0.821);color:white;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-size:18px;border:0;cursor:pointer}
.setting-text:hover{opacity:0.8}
input[type=range]::-webkit-slider-thumb{appearance:none;-moz-appearance:none;-webkit-appearance:none;height:14px;width:14px;border-radius:50%;background:rgb(5,165,189);cursor:pointer}
input[type=range][id=xzsx_length]{appearance:none;-moz-appearance:none;-webkit-appearance:none;height:8px;width:100%;margin-bottom:8px;outline:0;background:none;background:-webkit-linear-gradient(rgb(93,214,65),rgb(93,214,65))no-repeat,#fb0f0f;background-size:50% 100%}
input[type=range][id=xzsx_interval]{appearance:none;-moz-appearance:none;-webkit-appearance:none;height:8px;width:100%;margin-bottom:8px;outline:0;background:none;background:-webkit-linear-gradient(#fb0f0f,#fb0f0f)no-repeat,rgb(93,214,65);background-size:10% 100%}
input[type=range][id=xzsx_limitevent]{appearance:none;-moz-appearance:none;-webkit-appearance:none;height:8px;width:100%;margin-bottom:8px;outline:0;background:none;background:-webkit-linear-gradient(rgb(93,214,65),rgb(93,214,65))no-repeat,#fb0f0f;background-size:60% 100%}
.nav-list::-webkit-scrollbar{height:8px;width:8px}
.nav-list::-webkit-scrollbar-thumb{background-color:rgb(80,130,220,1)}
.nav-list::-webkit-scrollbar-thumb:hover{background-color:rgb(80,130,220,0.5)}
.nav-list::-webkit-scrollbar-track-piece{background:transparent}
#rule_keywords::-webkit-scrollbar{height:8px;width:8px}
#rule_keywords::-webkit-scrollbar-thumb{background-color:rgb(80,130,220,1)}
#rule_keywords::-webkit-scrollbar-thumb:hover{background-color:rgb(80,130,220,0.5)}
#rule_keywords::-webkit-scrollbar-track-piece{background:transparent}
#rule_antikeywords::-webkit-scrollbar{height:8px;width:8px}
#rule_antikeywords::-webkit-scrollbar-thumb{background-color:rgb(80,130,220,1)}
#rule_antikeywords::-webkit-scrollbar-thumb:hover{background-color:rgb(80,130,220,0.5)}
#rule_antikeywords::-webkit-scrollbar-track-piece{background:transparent}
.xzsx_detect_tag-trackpanel-show{width:250px;height:200px;display:flex;flex-direction:column;justify-content:flex-start;position:absolute;background-color:white;box-shadow:0px 0px 4px #6a6767;user-select:none;-o-user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);-o-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);-moz-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);-webkit-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);animation:showTrack 0.5s forwards;-o-animation:showTrack 0.5s forwards;-moz-animation:showTrack 0.5s forwards;-webkit-animation:showTrack 0.5s forwards;z-index:999999}
@keyframes showTrack{from{clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%)}
to{clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
}@-o-keyframes showTrack{from{-o-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%)}
to{-o-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
}@-moz-keyframes showTrack{from{-moz-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%)}
to{-moz-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
}@-webkit-keyframes showTrack{from{-webkit-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%)}
to{-webkit-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
}.xzsx_detect_tag-medalwall-show{height:100px;width:fit-content;max-width:400px;display:flex;flex-direction:column;position:absolute;background-color:white;box-shadow:0px 0px 4px #6a6767;user-select:none;-o-user-select:none;-ms-user-select:none;-moz-user-select:none;-webkit-user-select:none;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-weight:550;clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);-o-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);-moz-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);-webkit-clip-path:polygon(-5% -5%,-5% -5%,-5% 105%,-5% 105%);animation:showTrack 0.5s forwards;-o-animation:showTrack 0.5s forwards;-moz-animation:showTrack 0.5s forwards;-webkit-animation:showTrack 0.5s forwards;z-index:999999;overflow:auto}
.xzsx_detect_tag-trackpanel-hide{width:250px;height:200px;display:flex;flex-direction:column;justify-content:flex-start;position:absolute;background-color:white;box-shadow:0px 0px 4px #6a6767;user-select:none;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-weight:550;clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);-o-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);-moz-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);-webkit-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);animation:hideTrack 0.5s forwards;-o-animation:hideTrack 0.5s forwards;-moz-animation:hideTrack 0.5s forwards;-webkit-animation:hideTrack 0.5s forwards;z-index:999999;overflow:auto}
@keyframes hideTrack{from{clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
to{clip-path:polygon(105% -5%,105% -5%,105% 105%,105% 105%)}
}@-o-keyframes hideTrack{from{-o-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
to{-o-clip-path:polygon(105% -5%,105% -5%,105% 105%,105% 105%)}
}@-moz-keyframes hideTrack{from{-moz-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
to{-moz-clip-path:polygon(105% -5%,105% -5%,105% 105%,105% 105%)}
}@-webkit-keyframes hideTrack{from{-webkit-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%)}
to{-webkit-clip-path:polygon(105% -5%,105% -5%,105% 105%,105% 105%)}
}.xzsx_detect_tag-medalwall-hide{height:100px;width:fit-content;max-width:400px;display:flex;flex-direction:column;position:absolute;background-color:white;box-shadow:0px 0px 4px #6a6767;user-select:none;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑;font-weight:550;clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);-o-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);-moz-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);-webkit-clip-path:polygon(-5% -5%,105% -5%,105% 105%,-5% 105%);animation:hideTrack 0.5s forwards;-o-animation:hideTrack 0.5s forwards;-moz-animation:hideTrack 0.5s forwards;-webkit-animation:hideTrack 0.5s forwards;z-index:999999;overflow:auto}
.xzsx_detect_tag-medalwall-title{display:inline-flex;align-items:center;height:24px;width:100%;padding-left:8px;padding-top:4px;padding-bottom:4px;color:#1890ff;font-size:18px;font-style:normal;font-weight:600;box-shadow:0px 0px 4px #ddd;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;-o-text-overflow:ellipsis;-ms-text-overflow:ellipsis;cursor:pointer}
.xzsx_detect_tag-medalwall-container{position:relative}
.xzsx_detect_tag-trackpanel-content::-webkit-scrollbar{height:8px;width:8px}
.xzsx_detect_tag-trackpanel-content::-webkit-scrollbar-thumb{background-color:rgb(80,130,220,1)}
.xzsx_detect_tag-trackpanel-content::-webkit-scrollbar-thumb:hover{background-color:rgb(80,130,220,0.5)}
.xzsx_detect_tag-trackpanel-content::-webkit-scrollbar-track-piece{background:transparent}
#xzsx_medalwall::-webkit-scrollbar{height:8px;width:8px}
#xzsx_medalwall::-webkit-scrollbar-thumb{background-color:rgb(80,130,220,1)}
#xzsx_medalwall::-webkit-scrollbar-thumb:hover{background-color:rgb(80,130,220,0.5)}
#xzsx_medalwall::-webkit-scrollbar-track-piece{background:transparent}
.xzsx_detect_tag-trackpanel-title{display:flex;align-items:center;height:35px;padding-left:8px;padding-top:4px;padding-bottom:4px;color:#1890ff;font-size:18px;font-style:normal;font-weight:600;box-shadow:0px 0px 4px #ddd;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;-o-text-overflow:ellipsis;-ms-text-overflow:ellipsis;cursor:pointer}
.xzsx_detect_tag-trackpanel-content{height:200px;padding:8px;overflow:auto}
.xzsx_detect_tag-trackpanel-texttitle{font-weight:600;font-size:20px;margin-bottom:4px}
.xzsx_detect_tag-trackpanel-anti{margin-bottom:8px;color:red}
.xzsx_detect_tag-trackpanel-fans{margin-bottom:8px;color:green}
.xzsx_detect_tag-trackpanel-source{font-size:18px;color:rgba(22,51,194,0.821);font-weight:600}
.xzsx_detect_tag-trackpanel-text{font-size:16px;color:rgb(64,63,63);font-weight:400;padding-left:4px;margin-top:4px;margin-bottom:4px;box-shadow:0 0 4px #ddd}
`
/** 成分标签相关css样式
* @type { string }
*/
const css_tag =
`
.tag_container{display:inline-flex;align-items:center;flex-wrap:wrap;position:relative;vertical-align:middle;overflow:hidden;padding:1px 0px 1px 0px;margin-left:4px;transform:translateX(-50px);-o-transform:translateX(-50px);-ms-transform:translateX(-50px);-moz-transform:translateX(-50px);-webkit-transform:translateX(-50px);animation:showtag 0.5s ease-out forwards;-o-animation:showtag 0.5s ease-out forwards;-moz-animation:showtag 0.5s ease-out forwards;-webkit-animation:showtag 0.5s ease-out forwards;opacity:0}
@keyframes showtag{from{transform:translateX(-50px);opacity:0}
to{transform:translateX(0px);opacity:1}
}@-o-keyframes showtag{from{-o-transform:translateX(-50px);opacity:0}
to{-o-transform:translateX(0px);opacity:1}
}@-moz-keyframes showtag{from{-moz-transform:translateX(-50px);opacity:0}
to{-moz-transform:translateX(0px);opacity:1}
}@-webkit-keyframes showtag{from{-webkit-transform:translateX(-50px);opacity:0}
to{-webkit-transform:translateX(0px);opacity:1}
}.xzsx_detect_tag-medalwall-medalBtn{display:inline-flex;justify-content:center;align-items:center;height:100%;width:fit-content;margin-right:4px;box-shadow:0 0 6px #ddd;cursor:pointer}
.xzsx_detect_tag-medalwall-medalBtn:hover{opacity:0.8}
.xzsx_detect_tag{display:inline-flex;height:14px;line-height:14px;position:relative;box-sizing:content-box !important;padding:1px 2px 1px 14px;margin:1px 4px 1px 0;color:black;white-space:nowrap;font-family:"Microsoft YaHei","Microsoft Sans Serif","Microsoft SanSerf",微软雅黑 !important;font-size:14px;font-weight:600 !important;border-radius:6px 4px 4px 6px;box-shadow:0 0 6px #ddd;text-size-adjust:100%;-webkit-tap-highlight-color:transparent;overflow-wrap:break-word;cursor:pointer}
.xzsx_detect_tag:hover{opacity:0.8}
.xzsx_detect_tag::before{background:#fff;border-radius:10px;box-shadow:0 1px rgba(0,0,0,0.25) inset;content:'';height:6px;left:4px;position:absolute;width:6px;top:calc(50% - 3px)}
.icon-expend{height:24px;transform:rotate(90deg);-o-transform:rotate(90deg);-moz-transform:rotate(90deg);-webkit-transform:rotate(90deg);cursor:pointer}
.icon-expend:hover{opacity:0.8}
.icon-deepcheck{display:none;height:18px;width:18px;margin-left:4px}
.icon-deepcheck:hover{opacity:0.8}
.icon-deepcheck-hide{display:unset;height:18px;width:18px;transform:scale(1);-o-transform:scale(1);-moz-transform:scale(1);-webkit-transform:scale(1);transition:all 0.2s;-o-transition:all 0.2s;-moz-transition:all 0.2s;-webkit-transition:all 0.2s;cursor:pointer;z-index:1}
svg.check-container{background:transparent;position:absolute;height:18px;width:18px;left:0px}
circle.check-circle{fill:transparent;stroke:rgb(42,226,223);stroke-width:1.5px;stroke-dasharray:50.3,50.3;stroke-dashoffset:50.3;transition:all 1s;-o-transition:all 1s;-moz-transition:all 1s;-webkit-transition:all 1s;transform-origin:center;-o-transform-origin:center;-moz-transform-origin:center;-webkit-transform-origin:center}
circle:hover{stroke-dashoffset:0}
.svg-expend{height:25px;width:25px}
.svg-deepcehck{height:18px;width:18px;color:#9400D3}
`
style.innerHTML = css_panel + css_tag
/** 设置面板
* @type { HTMLDivElement }
*/
const nav = document.createElement('div')
nav.innerHTML =
`
<button class='nav-show'>
配置
</button>
<div id='panel-animation' class='hide-animation'>
<div class='navPanel-resetBtn' style='display:${navPanelVis ? '' : 'none'}'>
<svg fill="#FFFFFF" t="1636097794549" class="resetBtn-icon" viewBox="0 0 512 512" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5713">
<path d="M245.11,393.317c19.5,0,38.5-3.8,56.6-11.3c36.6-15.1,65-43.6,80.2-80.2s15.2-76.8,0-113.4
c-23-55.6-76.8-91.5-137-91.5c-19.5,0-38.5,3.8-56.6,11.3c-75.5,31.2-111.5,118.1-80.2,193.6
C131.11,357.317,184.91,393.317,245.11,393.317z M195.91,126.517c15.7-6.5,32.2-9.8,49-9.8c52.1,0,98.7,31.1,118.7,79.3
c13.1,31.7,13.1,66.6,0,98.2c-13.1,31.7-37.8,56.3-69.5,69.5c-15.7,6.5-32.2,9.8-49,9.8c-52.1,0-98.7-31.1-118.7-79.3
C99.31,228.817,130.51,153.517,195.91,126.517z"/>
<path d="M14.41,328.017c1.5,4,5.3,6.6,9.3,6.6c1.1,0,2.2-0.2,3.3-0.6c5.1-1.8,7.8-7.5,6-12.7
c-29.3-81.9-9.8-171.4,51-233.6c42-43,98.3-67.1,158.4-67.8c60-0.7,116.9,22,159.9,64l-25.3,0.3c-5.5,0.1-9.8,4.5-9.8,10
c0.1,5.4,4.5,9.8,9.9,9.8h0.1l49.2-0.6c2.6,0,5.1-1.1,7-3c1.8-1.9,2.8-4.4,2.8-7l-0.6-49.2c-0.1-5.5-4.5-10-10-9.8
c-5.5,0.1-9.8,4.5-9.8,10l0.3,25.3c-46.8-45.7-108.5-70.4-173.9-69.7c-65.4,0.9-126.7,27-172.4,73.8
c-32.7,33.5-55.1,75.4-64.6,121C-4.09,239.317-0.89,285.317,14.41,328.017z"/>
<path d="M406.01,402.417c-86.7,88.8-229.5,90.4-318.3,3.8l25.3-0.3c5.5-0.1,9.8-4.5,9.8-10
c-0.1-5.4-4.5-9.8-9.9-9.8h-0.1l-49.2,0.6c-5.5,0.1-9.8,4.5-9.8,10l0.6,49.2c0.1,5.4,4.5,9.8,9.9,9.8h0.1c5.5-0.1,9.8-4.5,9.8-10
l-0.3-25.3c47.6,46.4,109.3,69.6,171.1,69.6c63.7,0,127.3-24.6,175.2-73.6c32.7-33.5,55.1-75.4,64.6-121
c9.3-44.4,6.1-90.5-9.2-133.1c-1.8-5.1-7.5-7.8-12.7-6c-5.1,1.8-7.8,7.5-6,12.7C486.31,250.717,466.81,340.217,406.01,402.417z"/>
</svg>
</div>
<div class='side-btn side-btn-mode' style='display:${navPanelVis ? '' : 'none'}'>${MODE}</div>
<label for="file-upload" class='side-btn side-btn-import' style='display:${navPanelVis ? '' : 'none'}'>
导入
</label>
<input id='file-upload' name='config-upload' type='file' accept='.json,.txt' style='display:none' />
<div class='side-btn side-btn-export' style='display:${navPanelVis ? '' : 'none'}'>导出</div>
<div class='nav-panel'>
<div id='setting-panel' class='setting-show'>
<button id='DEL_STYLE' class='setting-btn' style="padding-left: 26px;">${DEL_STYLE}</button>
<button id='DET_BASE_MedalWall' class='setting-btn'>
<div class='statuslayout2'>
<div id='status_MedalWall' class='${DET_BASE['DET_BASE_MedalWall'] ? 'statusP' : 'statusE'}'></div>
</div>
粉丝牌
</button>
<button id='DET_BASE_FOLLOWINGS' class='setting-btn'>
<div class='statuslayout2'>
<div id='status_FOLLOWINGS' class='${DET_BASE['DET_BASE_FOLLOWINGS'] ? 'statusP' : 'statusE'}'></div>
</div>
关注列表
</button>
<button id='DET_BASE_VIDEOS' class='setting-btn'>
<div class='statuslayout2'>
<div id='status_VIDEOS' class='${DET_BASE['DET_BASE_VIDEOS'] ? 'statusP' : 'statusE'}'></div>
</div>
投稿列表
</button>
<button id='DET_BASE_DYNAMIC' class='setting-btn'>
<div class='statuslayout2'>
<div id='status_DYNAMIC' class='${DET_BASE['DET_BASE_DYNAMIC'] ? 'statusP' : 'statusE'}'></div>
</div>
动态内容
</button>
<button id='DET_BASE_COMMENT' class='setting-btn'>
<div class='statuslayout2'>
<div id='status_COMMENT' class='${DET_BASE['DET_BASE_COMMENT'] ? 'statusP' : 'statusE'}'></div>
</div>
评论内容
</button>
<div id='TAG_MAX_LENGTH' class='setting-text'>
收纳阈值
<span>${TAG_MAX_LENGTH}</span>
<input id='xzsx_length' type="range" min="1" max="50" step="1" value="${TAG_MAX_LENGTH}"/>
</div>
<div id='INTERVAL' class='setting-text'>
渲染间隔
<span>${INTERVAL}</span>
<input id='xzsx_interval' type="range" min="0" max="5000" step="250" value="${INTERVAL}"/>
</div>
<div id='LIMIT_EVENT' class='setting-text'>
事件节流
<span>${LIMIT_EVENT}</span>
<input id='xzsx_limitevent' type="range" min="1" max="25" step="1" value="${LIMIT_EVENT}"/>
</div>
</div>
<div class='nav-tabs'>
<button class='nav-tabs-btn1'>
检测成分
<div id='status1' class='statuslayout'>
<div id='status1_circleE' class='statusE' style='display:${DET ? 'none' : ''}'></div>
<div id='status1_circleP' class='statusP' style='display:${DET ? '' : 'none'}'></div>
<div id='status1_effect' class='statusPcircle' style='display:${DET ? '' : 'none'}'></div>
</div>
</button>
<div class='changed-signel1' style='display:none'></div>
<div id='process1' class='process' style="left:0%;"></div>
<button class='nav-tabs-btn2'>
自动屏蔽
<div id='status2' class='statuslayout'>
<div id='status2_circleE' class='statusE' style='display:${DEL ? 'none' : ''}'></div>
<div id='status2_circleP' class='statusP' style='display:${DEL ? '' : 'none'}'></div>
<div id='status2_effect' class='statusPcircle' style='display:${DEL ? '' : 'none'}'></div>
</div>
</button>
<div class='changed-signel2' style='display:none'></div>
<div id='process2' class='process' style="left:50%;"></div>
</div>
<div class='addrule_form'>
<div class='addrule_item1'>
<input id='tagname' type="text" class='tagname_input' placeholder='标签名' />
<input id="tagcolor" type="color" class='tagcolor_input' value='#008000' />
</div>
<div class='xzsx_detect_tagsample_container'>
<div class='xzsx_detect_tagsample'></div>
</div>
<div id='rule_keywords' class='rule_keywords'>
<div class='add_items'>
<input id='keyword' type="text" class='tagkeyword_input' placeholder='粉丝关键词' />
<div id='keywords-add' class='add_icon'>+</div>
</div>
</div>
<div id='rule_antikeywords' class='rule_keywords'>
<div class='add_items'>
<input id='antikeyword' type="text" class='tagkeyword_input' placeholder='黑子关键词' />
<div id='antikeywords-add' class='add_icon'>+</div>
</div>
</div>
</div>
<div class='nav-list'>
<div class='rule-item-detect-add'>+</div>
<div class='rule-item-blacklist-add'>
<input type="text" placeholder='关键词' class='rule-item-blacklist-add-input' style="display:unset"/>
<select name='rule-item-blacklist-select' class='rule-item-blacklist-select' style="display:none" required></select>
<div class='rule-item-blacklist-base'>基于昵称</div>
<div id='blacklist-add' class='rule-item-blacklist-addicon'>+</div>
</div>
</div>
<div class='nav-footer'>
<div class='applysuccess' style="display:none">
<svg fill="white" width="24px" height="24px" viewBox="0 0 36 36" version="1.1">
<path d="M13.72,27.69,3.29,17.27a1,1,0,0,1,1.41-1.41l9,9L31.29,7.29a1,1,0,0,1,1.41,1.41Z"></path>
<rect x="0" y="0" width="36" height="36" fill-opacity="0" />
</svg>
</div>
<div id='footer-process' class='process'></div>
<button class='confirm_btn'>应 用</button>
<button class='add_btn' style="display:none">新 增 规 则</button>
<button class='alter_btn' style="display:none">修 改 规 则</button>
</div>
</div>
</div>
<div id='xzsx_trackpanel' class='xzsx_detect_tag-trackpanel-hide' style='display:none'></div>
<div id='xzsx_medalwall' class='xzsx_detect_tag-medalwall-hide'></div>
`
document.querySelector('head').appendChild(style)
document.querySelector('body').appendChild(nav)
/** <===================================================================================获取实例==========================================================================================>
* @type { Element }
*/
let
/** 面板显示/隐藏动画 */
animation = document.querySelector('#panel-animation'),
/** 侧边悬浮按钮 */
navBtn = document.querySelector('.nav-show'),
/** 设置面板 */
navPanel = document.querySelector('.nav-panel'),
/** 成分溯源面板 */
trackPanel = document.querySelector('#xzsx_trackpanel'),
/** 勋章墙 */
medalWall = document.querySelector('#xzsx_medalwall'),
/** 侧边悬浮按钮 */
resetBtn = document.querySelector('.navPanel-resetBtn'),
/** 切换模式按钮 */
navMode = document.querySelector('.side-btn.side-btn-mode'),
/** 导入按钮 */
importBtn = document.querySelector('.side-btn.side-btn-import'),
/** 上传文件 */
uploader = document.querySelector('#file-upload'),
/** 侧边悬浮按钮 */
exportBtn = document.querySelector('.side-btn.side-btn-export'),
/** 检测成分按钮 */
navTabsBtn1 = document.querySelector('.nav-tabs-btn1'),
/** 自动屏蔽按钮 */
navTabsBtn2 = document.querySelector('.nav-tabs-btn2'),
/** 设置面板 */
settingPanel = document.querySelector('#setting-panel'),
/** 屏蔽样式按钮 */
DEL_STYLE_BTN = document.querySelector('#DEL_STYLE'),
/**检测依据-粉丝牌 */
DET_BASE_MedalWall = document.querySelector('#DET_BASE_MedalWall'),
/** 检测依据-关注列表 */
DET_BASE_FOLLOWINGS = document.querySelector('#DET_BASE_FOLLOWINGS'),
/** 检测依据-投稿列表 */
DET_BASE_VIDEOS = document.querySelector('#DET_BASE_VIDEOS'),
/** 检测依据-动态空间 */
DET_BASE_DYNAMIC = document.querySelector('#DET_BASE_DYNAMIC'),
/** 检测依据-评论文本 */
DET_BASE_COMMENT = document.querySelector('#DET_BASE_COMMENT'),
/** 标签最大长度文本 */
TAG_MAX_LENGTH_TEXT = document.querySelector('#TAG_MAX_LENGTH'),
/** 标签最大长度滑动条 */
TAG_MAX_LENGTH_SLIDER = document.querySelector('#xzsx_length'),
/** 渲染间隔文本 */
INTERVAL_TEXT = document.querySelector('#INTERVAL'),
/** 渲染间隔滑动条 */
INTERVAL_SLIDER = document.querySelector('#xzsx_interval'),
/** 事件节流文本 */
LIMIT_EVENT_TEXT = document.querySelector('#LIMIT_EVENT'),
/** 事件节流滑动条 */
LIMIT_EVENT_SLIDER = document.querySelector('#xzsx_limitevent'),
/** 检测成分启用状态标识1 */
status1_circleP = document.querySelector('#status1_circleP'),
/** 检测成分启用状态标识2 */
status1_circleE = document.querySelector('#status1_circleE'),
/** 检测成分启用状态标识3 */
status1_effect = document.querySelector('#status1_effect'),
/** 自动屏蔽启用状态标识1 */
status2_circleP = document.querySelector('#status2_circleP'),
/** 自动屏蔽启用状态标识2 */
status2_circleE = document.querySelector('#status2_circleE'),
/** 自动屏蔽启用状态标识3 */
status2_effect = document.querySelector('#status2_effect'),
/** 检测依据启用状态标识-粉丝牌 */
status_MedalWall = document.querySelector('#status_MedalWall'),
/** 检测依据启用状态标识-关注列表 */
status_FOLLOWINGS = document.querySelector('#status_FOLLOWINGS'),
/** 检测依据启用状态标识-投稿列表 */
status_VIDEOS = document.querySelector('#status_VIDEOS'),
/** 检测依据启用状态标识-动态内容(), */
status_DYNAMIC = document.querySelector('#status_DYNAMIC'),
/** 检测依据启用状态标识-评论内容 */
status_COMMENT = document.querySelector('#status_COMMENT'),
/** 检测成分长按进度条 */
process1 = document.querySelector('#process1'),
/** 自动屏蔽长按进度条 */
process2 = document.querySelector('#process2'),
/** 检测成分修改标识 */
changedSignel1 = document.querySelector('.changed-signel1'),
/** 自动屏蔽修改标识 */
changedSignel2 = document.querySelector('.changed-signel2'),
/** 规则列表 */
navRuleList = document.querySelector('.nav-list'),
/** 检测成分添加/修改规则界面 */
addRuleForm = document.querySelector('.addrule_form'),
/** 检测成分添加/修改规则界面 添加粉丝关键词右侧<+>号按钮 */
addKeyWordBtn = document.querySelector('#keywords-add'),
/** 检测成分添加/修改规则界面 添加黑子关键词右侧<+>号按钮 */
addAntiKeyWordBtn = document.querySelector('#antikeywords-add'),
/** 检测成分添加/修改规则界面 添加粉丝关键词列表 */
keywords = document.querySelector('#rule_keywords'),
/** 检测成分添加/修改规则界面 添加黑子关键词列表 */
antikeywords = document.querySelector('#rule_antikeywords'),
/** 检测成分添加/修改规则界面 标签名 */
tagname = document.querySelector('#tagname'),
/** 检测成分添加/修改规则界面 标签颜色 */
tagcolor = document.querySelector('#tagcolor'),
/** 检测成分添加/修改规则界面 标签样式预览 */
tagsample = document.querySelector('.xzsx_detect_tagsample'),
/** 检测成分添加/修改规则界面 添加粉丝关键词左侧文字输入框 */
newKeyWord = document.querySelector('#keyword'),
/** 检测成分添加/修改规则界面 添加黑子关键词左侧文字输入框 */
newAntiKeyWord = document.querySelector('#antikeyword'),
/** 检测成分规则列表 添加按钮 */
detectAdd_btn = document.querySelector('.rule-item-detect-add'),
/** 黑名单规则列表 添加关键词容器 */
blacklistAdd_item = document.querySelector('.rule-item-blacklist-add'),
/** 黑名单规则列表 添加关键词选择框 */
blacklistAdd_select = document.querySelector('.rule-item-blacklist-select'),
/** 黑名单规则列表 添加关键词右侧<+>号按钮 */
blacklistAdd_btn = document.querySelector('#blacklist-add'),
/** 黑名单规则列表 添加关键词屏蔽依据 */
blacklistAdd_base = document.querySelector('.rule-item-blacklist-base'),
/** 黑名单规则列表 添加关键词左侧文字输入框 */
blacklistAdd_keyword = document.querySelector('.rule-item-blacklist-add-input'),
/** 页脚 进度条 */
footerProcess = document.querySelector('#footer-process'),
/** 页脚 应用按钮 */
confirm_btn = document.querySelector('.confirm_btn'),
/** 页脚 应用成功动画 */
confirm_success = document.querySelector('.applysuccess'),
/** 页脚 新增规则按钮 */
add_btn = document.querySelector('.add_btn'),
/** 页脚 修改规则按钮 */
alter_btn = document.querySelector('.alter_btn')
/** 批量监听下拉设置栏DOM元素的变化并绑定数据
* @type { () => void }
*/
const batchWatch = () => {
/**
* @type { { watchObj: object, watchKey: string, target: { ele: Element | null, key: string, callback: () => string | number }[] }
*/
const watchParams = [
{ watchObj: settingTemp, watchKey: 'DEL_STYLE', target: { ele: DEL_STYLE_BTN, key: 'textContent', callback: (newValue) => newValue } },
{ watchObj: settingTemp, watchKey: 'TAG_MAX_LENGTH', target: { ele: TAG_MAX_LENGTH_TEXT.childNodes[1], key: 'textContent', callback: (newValue) => newValue } },
{ watchObj: settingTemp, watchKey: 'INTERVAL', target: { ele: INTERVAL_TEXT.childNodes[1], key: 'textContent', callback: (newValue) => newValue } },
{ watchObj: settingTemp, watchKey: 'LIMIT_EVENT', target: { ele: LIMIT_EVENT_TEXT.childNodes[1], key: 'textContent', callback: (newValue) => newValue } },
{ watchObj: settingTemp, watchKey: 'DET_BASE_MedalWall', target: { ele: status_MedalWall, key: 'className', callback: (newValue) => newValue ? 'statusP' : 'statusE' } },
{ watchObj: settingTemp, watchKey: 'DET_BASE_FOLLOWINGS', target: { ele: status_FOLLOWINGS, key: 'className', callback: (newValue) => newValue ? 'statusP' : 'statusE' } },
{ watchObj: settingTemp, watchKey: 'DET_BASE_VIDEOS', target: { ele: status_VIDEOS, key: 'className', callback: (newValue) => newValue ? 'statusP' : 'statusE' } },
{ watchObj: settingTemp, watchKey: 'DET_BASE_DYNAMIC', target: { ele: status_DYNAMIC, key: 'className', callback: (newValue) => newValue ? 'statusP' : 'statusE' } },
{ watchObj: settingTemp, watchKey: 'DET_BASE_COMMENT', target: { ele: status_COMMENT, key: 'className', callback: (newValue) => newValue ? 'statusP' : 'statusE' } },
]
watchParams.forEach((v) => watch(v.watchObj, v.watchKey, (newValue) => v.target.ele[v.target.key] = v.target.callback(newValue)))
}
/** 获取集合1相对于集合2的差集
* @param { Set } set1 集合1
* @param { Set } set2 集合2
* @returns { Set } 差集结果
*/
const _difference = (set1, set2) => new Set([...set1].filter((v) => !set2.has(v)))
/** 重置DOM状态
* @type { () => void }
*/
const resetAll = () => {
navMode.textContent = MODE
status1_circleP.style.display = DET ? '' : 'none'
status1_circleE.style.display = DET ? 'none' : ''
status1_effect.style.display = DET ? '' : 'none'
process1.style.visibility = 'hidden'
status2_circleP.style.display = DEL ? '' : 'none'
status2_circleE.style.display = DEL ? 'none' : ''
status2_effect.style.display = DEL ? '' : 'none'
process2.style.visibility = 'hidden'
DEL_STYLE_BTN.textContent = DEL_STYLE
TAG_MAX_LENGTH_SLIDER.value = parseInt(TAG_MAX_LENGTH)
INTERVAL_SLIDER.value = parseInt(INTERVAL)
LIMIT_EVENT_SLIDER.value = parseInt(LIMIT_EVENT)
TAG_MAX_LENGTH_TEXT.childNodes[1].textContent = parseInt(TAG_MAX_LENGTH)
INTERVAL_TEXT.childNodes[1].textContent = parseInt(INTERVAL)
LIMIT_EVENT_TEXT.childNodes[1].textContent = parseInt(LIMIT_EVENT)
status_MedalWall.className = API_OVERLOAD['DET_BASE_MedalWall'].state ? 'statusBAN' : DET_BASE['DET_BASE_MedalWall'] ? 'statusP' : 'statusE'
status_FOLLOWINGS.className = API_OVERLOAD['DET_BASE_FOLLOWINGS'].state ? 'statusBAN' : DET_BASE['DET_BASE_FOLLOWINGS'] ? 'statusP' : 'statusE'
status_VIDEOS.className = API_OVERLOAD['DET_BASE_VIDEOS'].state ? 'statusBAN' : DET_BASE['DET_BASE_VIDEOS'] ? 'statusP' : 'statusE'
status_DYNAMIC.className = API_OVERLOAD['DET_BASE_DYNAMIC'].state ? 'statusBAN' : DET_BASE['DET_BASE_DYNAMIC'] ? 'statusP' : 'statusE'
status_COMMENT.className = DET_BASE['DET_BASE_COMMENT'] ? 'statusP' : 'statusE'
}
/** <===================================================================================事件绑定==========================================================================================> */
/** 阻止点击事件冒泡 */
[navPanel, trackPanel, medalWall, importBtn, uploader].forEach(ele => ele.onclick = e => e.stopPropagation())
/** 控制面板可见性 */
navBtn.onclick = (e) => {
e.stopPropagation()
animationState = !animationState
animation.className = animationState ? 'show-animation' : 'hide-animation'
clearTimeout(timerAnimaltion)
if (navPanelVis && !animationState) {
timerAnimaltion = setTimeout(() => {
navPanelVis = false
navPanel.style.display = 'none'
resetBtn.style.display = 'none'
navMode.style.display = 'none'
importBtn.style.display = 'none'
exportBtn.style.display = 'none'
blacklistAdd_keyword.value = ''
blacklistAdd_select.style.display = 'none'
blacklistAdd_keyword.style.display = 'unset'
blacklistAdd_base.textContent = '基于昵称'
}, 500)
} else {
[rulesApply, rules] = initRules()
resetAll()
renderMainPanel()
navPanelVis = true, ctab = true, ckey = undefined, newRuleKeysSet.clear(), deleteRuleKeysSet.clear(), newAntiRuleSets.clear(), deleteAntiRuleSets.clear(), settingTemp = {}
batchWatch()
navPanel.style.display = 'flex'
resetBtn.style.display = 'flex'
navMode.style.display = 'flex'
importBtn.style.display = 'flex'
exportBtn.style.display = 'flex'
changedSignel1.style.display = 'none'
changedSignel2.style.display = 'none'
navTabsBtn1.click()
}
}
/** 设置面板鼠标移开相关操作 */
navPanel.onmouseleave = () => { settingPanel.style.height = '0px' }
/** 点击空白处相关操作 */
document.addEventListener('click', () => { trackPanel.className = 'xzsx_detect_tag-trackpanel-hide'; medalWall.className = 'xzsx_detect_tag-medalwall-hide'; medalWall.setAttribute('key', null) })
/** 执行显示隐藏面板各一次 */
const [_dreset, __] = _throttle(() => { navBtn.click() }, 500, { leading: true })
/** 恢复设置 */
resetBtn.onclick = (e) => { e.stopPropagation(); _dreset() }
/** 切换模式 */
navMode.onclick = (e) => {
e.stopPropagation()
let MODE_TEMP = settingTemp.hasOwnProperty('MODE') ? settingTemp['MODE'] === '自动' ? '静默' : '自动' : MODE === '自动' ? '静默' : '自动'
navMode.textContent = MODE_TEMP
settingTemp['MODE'] = MODE_TEMP
}
/** 导入 */
uploader.onchange = async (e) => {
e.stopPropagation()
try {
let conf = JSON.parse(await e.target.files[0].text())
if (Object.keys(conf).length > 0) {
GM_setValue('xzsx_bilibili_detector', JSON.stringify({ blackList: conf.blackList, detect: conf.detect }))
GM_setValue('xzsx_bilibili_detector_state', JSON.stringify(conf.state))
resetBtn.click()
e.target.value = ''
}
} catch (error) {
console.log(error)
}
}
/** 导出 */
exportBtn.onclick = (e) => {
e.stopPropagation()
let temp = {}, state = {
MODE, DET, DEL, DEL_STYLE, TAG_MAX_LENGTH, INTERVAL, LIMIT_EVENT,
DET_BASE_MedalWall: DET_BASE['DET_BASE_MedalWall'],
DET_BASE_FOLLOWINGS: DET_BASE['DET_BASE_FOLLOWINGS'],
DET_BASE_VIDEOS: DET_BASE['DET_BASE_VIDEOS'],
DET_BASE_DYNAMIC: DET_BASE['DET_BASE_DYNAMIC'],
DET_BASE_COMMENT: DET_BASE['DET_BASE_COMMENT']
}
Object.keys(rulesApply.detect).forEach((key) => {
temp[key] = { 'color': rulesApply.detect[key].color, 'keywords': [...rulesApply.detect[key].keywords], 'antikeywords': [...rulesApply.detect[key].antikeywords] }
})
let json = JSON.stringify({ blackList: rulesApply.blackList, detect: temp, state })
let blob = new Blob([json], { type: "octet/stream" })
let url = window.URL.createObjectURL(blob)
let a = document.createElement("a");
a.href = url
a.download = 'xzsx_bilibili_detector.json'
a.click()
a.remove()
window.URL.revokeObjectURL(url)
}
/** 检测成分按钮单击相关操作 */
navTabsBtn1.onclick = () => {
ctab = true
addRuleForm.style.display = 'none'
detectAdd_btn.style.display = ''
blacklistAdd_item.style.display = 'none'
confirm_btn.style.display = ''
add_btn.style.display = 'none'
alter_btn.style.display = 'none'
navTabsBtn1.style.filter = 'brightness(100%)'
navTabsBtn2.style.filter = 'brightness(50%)'
newRuleKeysSet.clear()
deleteRuleKeysSet.clear()
newAntiRuleSets.clear()
deleteAntiRuleSets.clear()
renderMainPanel()
}
/** 检测成分按钮长按相关操作 */
navTabsBtn1.onmousedown = () => {
if (ctab) {
process1.style.visibility = 'visible'
process1.style.width = '50%'
timerTab1 = setTimeout(() => {
let temp = settingTemp.hasOwnProperty('DET') ? !settingTemp['DET'] : !DET
settingTemp['DET'] = temp
changedSignel1.style.display = ''
status1_circleP.style.display = temp ? '' : 'none'
status1_circleE.style.display = temp ? 'none' : ''
status1_effect.style.display = 'none'
process1.style.visibility = 'hidden'
process1.style.width = '0%'
}, 500)
}
}
/** 检测成分按钮松开相关操作 */
navTabsBtn1.onmouseup = () => {
if (ctab) {
process1.style.width = '0%'
clearTimeout(timerTab1)
}
}
/** 检测成分按钮离开相关操作 */
navTabsBtn1.onmouseout = () => {
if (ctab) {
process1.style.width = '0%'
clearTimeout(timerTab1)
}
}
/** 检测成分按钮标悬浮相关操作 */
navTabsBtn1.onmouseenter = () => { settingPanel.style.height = '0px' }
/** 自动屏蔽按钮单击相关操作 */
navTabsBtn2.onclick = () => {
ctab = false
addRuleForm.style.display = 'none'
detectAdd_btn.style.display = 'none'
blacklistAdd_item.style.display = 'flex'
confirm_btn.style.display = ''
add_btn.style.display = 'none'
alter_btn.style.display = 'none'
navTabsBtn1.style.filter = 'brightness(50%)'
navTabsBtn2.style.filter = 'brightness(100%)'
newRuleKeysSet.clear()
deleteRuleKeysSet.clear()
newAntiRuleSets.clear()
deleteAntiRuleSets.clear()
renderMainPanel()
}
/** 自动屏蔽按钮鼠标聚焦相关操作 */
navTabsBtn2.onmouseenter = () => { if (!ctab) settingPanel.style.height = '378px' }
/** 设置下拉栏按钮鼠标离开相关操作 */
settingPanel.onmouseleave = () => { settingPanel.style.height = '0px' }
/** 规则列表鼠标悬浮相关操作 */
navRuleList.onmouseenter = () => { settingPanel.style.height = '0px' }
/** 修改屏蔽样式 */
DEL_STYLE_BTN.onclick = () => {
let copy = JSON.parse(JSON.stringify(settingTemp))
settingTemp['DEL_STYLE'] = copy.hasOwnProperty('DEL_STYLE') ? copy['DEL_STYLE'] === '消息屏蔽' ? '完全删除' : '消息屏蔽' : DEL_STYLE === '消息屏蔽' ? '完全删除' : '消息屏蔽'
}
/** 修改标签收纳阈值 */
TAG_MAX_LENGTH_SLIDER.onchange = (e) => settingTemp['TAG_MAX_LENGTH'] = parseInt(e.target.value)
/** 修改渲染间隔(毫秒) */
INTERVAL_SLIDER.onchange = (e) => settingTemp['INTERVAL'] = parseInt(e.target.value)
/** 事件节流(每秒) */
LIMIT_EVENT_SLIDER.onchange = (e) => settingTemp['LIMIT_EVENT'] = parseInt(e.target.value)
/** 检测依据-粉丝牌 */
DET_BASE_MedalWall.onclick = () => {
let copy = JSON.parse(JSON.stringify(settingTemp))
settingTemp['DET_BASE_MedalWall'] = copy.hasOwnProperty('DET_BASE_MedalWall') ? !copy['DET_BASE_MedalWall'] : !DET_BASE['DET_BASE_MedalWall']
}
/** 检测依据-关注列表 */
DET_BASE_FOLLOWINGS.onclick = () => {
let copy = JSON.parse(JSON.stringify(settingTemp))
settingTemp['DET_BASE_FOLLOWINGS'] = copy.hasOwnProperty('DET_BASE_FOLLOWINGS') ? !copy['DET_BASE_FOLLOWINGS'] : !DET_BASE['DET_BASE_FOLLOWINGS']
}
/** 检测依据-投稿列表 */
DET_BASE_VIDEOS.onclick = () => {
let copy = JSON.parse(JSON.stringify(settingTemp))
settingTemp['DET_BASE_VIDEOS'] = copy.hasOwnProperty('DET_BASE_VIDEOS') ? !copy['DET_BASE_VIDEOS'] : !DET_BASE['DET_BASE_VIDEOS']
}
/** 检测依据-动态空间 */
DET_BASE_DYNAMIC.onclick = () => {
let copy = JSON.parse(JSON.stringify(settingTemp))
settingTemp['DET_BASE_DYNAMIC'] = copy.hasOwnProperty('DET_BASE_DYNAMIC') ? !copy['DET_BASE_DYNAMIC'] : !DET_BASE['DET_BASE_DYNAMIC']
}
/** 检测依据-评论文本 */
DET_BASE_COMMENT.onclick = () => {
let copy = JSON.parse(JSON.stringify(settingTemp))
settingTemp['DET_BASE_COMMENT'] = copy.hasOwnProperty('DET_BASE_COMMENT') ? !copy['DET_BASE_COMMENT'] : !DET_BASE['DET_BASE_COMMENT']
}
/** 自动屏蔽按钮长按相关操作 */
navTabsBtn2.onmousedown = () => {
if (!ctab) {
process2.style.visibility = 'visible'
process2.style.width = '50%'
timerTab2 = setTimeout(() => {
let temp = settingTemp.hasOwnProperty('DEL') ? !settingTemp['DEL'] : !DEL
settingTemp['DEL'] = temp
changedSignel2.style.display = ''
status2_circleP.style.display = temp ? '' : 'none'
status2_circleE.style.display = temp ? 'none' : ''
status2_effect.style.display = 'none'
process2.style.visibility = 'hidden'
process2.style.width = '0%'
}, 500)
}
}
/** 自动屏蔽按钮松开相关操作 */
navTabsBtn2.onmouseup = () => {
if (!ctab) {
process2.style.width = '0%'
clearTimeout(timerTab2)
}
}
/** 自动屏蔽按钮离开相关操作 */
navTabsBtn2.onmouseout = () => {
if (!ctab) {
process2.style.width = '0%'
clearTimeout(timerTab2)
}
}
/** 检测成分添加规则按钮相关操作 */
detectAdd_btn.onclick = () => {
add_btn.style.display = ''
alter_btn.style.display = 'none'
confirm_btn.style.display = 'none'
renderEditPanel(undefined)
}
/** 自动屏蔽添加规则按钮相关操作 */
blacklistAdd_btn.onclick = () => {
if (blacklistAdd_base.textContent === '基于成分' && blacklistAdd_select.value.trim().length > 0 && !rules.blackList.hasOwnProperty(blacklistAdd_select.value)) {
changedSignel2.style.display = ''
rules.blackList[blacklistAdd_select.value.trim()] = blacklistAdd_base.textContent
blacklistAdd_select.querySelector(`option[value="${blacklistAdd_select.value.trim()}"]`).remove()
renderMainPanel()
} else if (blacklistAdd_base.textContent === '基于昵称') {
let keyword = blacklistAdd_keyword.value?.trim() ?? ''
if (!rules.blackList.hasOwnProperty(keyword) && keyword.length > 0) {
changedSignel2.style.display = ''
rules.blackList[keyword] = blacklistAdd_base.textContent
blacklistAdd_keyword.value = ''
renderMainPanel()
}
}
}
/** 自动屏蔽屏蔽依据按钮相关操作 */
blacklistAdd_base.onclick = () => {
blacklistAdd_base.textContent = blacklistAdd_base.textContent === '基于昵称' ? '基于成分' : '基于昵称'
if (blacklistAdd_base.textContent === '基于成分') {
let detectListKeys = Object.keys(rules.detect), blackListKeys = Object.keys(rules.blackList), htmlStr = ''
detectListKeys.filter((key) => !blackListKeys.includes(key)).forEach((v) => htmlStr += `<option value='${v.trim()}'>${v.trim()}</option>`)
blacklistAdd_select.innerHTML = htmlStr
blacklistAdd_select.style.display = 'unset'
blacklistAdd_keyword.style.display = 'none'
} else if (blacklistAdd_base.textContent === '基于昵称') {
blacklistAdd_select.style.display = 'none'
blacklistAdd_keyword.style.display = 'unset'
}
}
/** 黑子关键词输入框回车直接触发添加 */
blacklistAdd_keyword.onkeydown = (e) => { e.keyCode === 13 && blacklistAdd_btn.click() }
/** 检测成分添加/修改页面 拾色器相关操作 */
tagcolor.onchange = (e) => {
tagname.style.color = e.target.value
changedSignel1.style.display = ''
tagsample.style.color = getContrastColor(e.target.value)
tagsample.style.backgroundColor = e.target.value
tagsample.style.borderColor = e.target.value + '80'
tagsample.style.backgroundImage = `linear-gradient(45deg, ${e.target.value}, ${e.target.value + '80'})`
}
/** 检测成分添加/修改页面 标签名相关操作 */
tagname.onchange = (e) => {
changedSignel1.style.display = ''
tagname.title = e.target.value
tagsample.title = e.target.value
tagsample.textContent = e.target.value
}
/** 检测成分添加/修改页面 添加粉丝关键词按钮<+>相关操作 */
addKeyWordBtn.onclick = () => {
let keyword = newKeyWord.value?.trim() ?? '', exist = false
for (const dom of keywords.querySelectorAll('.keyword')) {
if (dom.firstChild.textContent === keyword) {
exist = true
break
}
}
if (keyword.length > 0 && !exist) {
newRuleKeysSet.add(keyword)
newKeyWord.value = ''
changedSignel1.style.display = ''
let keywordDom = document.createElement('div')
keywordDom.className = 'keyword'
keywordDom.style.color = 'green'
keywordDom.innerHTML = `${keyword}<div class="keyword_delete">x</div>`
keywordDom.children[0].onclick = () => {
newRuleKeysSet.has(keyword) ? newRuleKeysSet.delete(keyword) : deleteRuleKeysSet.add(keyword)
keywordDom.remove(keywordDom.children[0])
changedSignel1.style.display = ''
}
keywords.insertBefore(keywordDom, keywords.lastChild.previousSibling)
}
}
/** 粉丝关键词输入框回车直接触发添加 */
newKeyWord.onkeydown = (e) => { e.keyCode === 13 && addKeyWordBtn.click() }
/** 检测成分添加/修改页面 添加黑子关键词按钮<+>相关操作 */
addAntiKeyWordBtn.onclick = () => {
let keyword = newAntiKeyWord.value?.trim() ?? '', exist = false
for (const dom of antikeywords.querySelectorAll('.keyword')) {
if (dom.firstChild.textContent === keyword) {
exist = true
break
}
}
if (keyword.length > 0 && !exist) {
newAntiRuleSets.add(keyword)
newAntiKeyWord.value = ''
changedSignel1.style.display = ''
let keywordDom = document.createElement('div')
keywordDom.className = 'keyword'
keywordDom.style.color = 'rgb(243,28,28)'
keywordDom.innerHTML = `${keyword}<div class="keyword_delete">x</div>`
keywordDom.children[0].onclick = () => {
newAntiRuleSets.has(keyword) ? newAntiRuleSets.delete(keyword) : deleteAntiRuleSets.add(keyword)
keywordDom.remove(keywordDom.children[0])
changedSignel1.style.display = ''
}
antikeywords.insertBefore(keywordDom, antikeywords.lastChild.previousSibling)
}
}
/** 黑子关键词输入框回车直接触发添加 */
newAntiKeyWord.onkeydown = (e) => { e.keyCode === 13 && addAntiKeyWordBtn.click() }
/** 页脚应用按钮长按相关操作 */
confirm_btn.onmousedown = async () => {
clearTimeout(timerFoot)
clearTimeout(timerFoot2)
footerProcess.style.visibility = 'visible'
footerProcess.style.width = '100%'
timerFoot = setTimeout(async () => {
let curDocument = domain === 'live.bilibili.com' ? ciframe !== undefined ? ciframe.contentWindow.document : document : document
curDocument.querySelectorAll('.tag_container').forEach(tag => tag.remove())
userTags = {}
userTagsHash = {}
userBlackList.clear()
uidset.clear()
changedSignel1.style.display = 'none'
changedSignel2.style.display = 'none'
footerProcess.style.visibility = 'hidden'
footerProcess.style.width = '0%'
blacklistAdd_keyword.value = ''
trackPanel.className = 'xzsx_detect_tag-trackpanel-hide'
medalWall.className = 'xzsx_detect_tag-trackpanel-hide'
medalWall.setAttribute('key', null)
let settingTempCopy = JSON.parse(JSON.stringify(settingTemp))
settingTemp = {}
if (settingTempCopy.hasOwnProperty('MODE')) MODE = settingTempCopy['MODE']
if (settingTempCopy.hasOwnProperty('DET')) DET = settingTempCopy['DET']
if (settingTempCopy.hasOwnProperty('DEL')) DEL = settingTempCopy['DEL']
if (DET) status1_effect.style.display = ''
if (DEL) status2_effect.style.display = ''
if (settingTempCopy.hasOwnProperty('DEL_STYLE')) DEL_STYLE = settingTempCopy['DEL_STYLE']
if (settingTempCopy.hasOwnProperty('TAG_MAX_LENGTH')) TAG_MAX_LENGTH = settingTempCopy['TAG_MAX_LENGTH']
if (settingTempCopy.hasOwnProperty('INTERVAL')) INTERVAL = settingTempCopy['INTERVAL']
if (settingTempCopy.hasOwnProperty('LIMIT_EVENT')) LIMIT_EVENT = settingTempCopy['LIMIT_EVENT']
if (settingTempCopy.hasOwnProperty('DET_BASE_MedalWall')) DET_BASE['DET_BASE_MedalWall'] = settingTempCopy['DET_BASE_MedalWall']; if (DET_BASE['DET_BASE_MedalWall']) recoverAPI('DET_BASE_MedalWall')
if (settingTempCopy.hasOwnProperty('DET_BASE_FOLLOWINGS')) DET_BASE['DET_BASE_FOLLOWINGS'] = settingTempCopy['DET_BASE_FOLLOWINGS']; if (DET_BASE['DET_BASE_FOLLOWINGS']) recoverAPI('DET_BASE_FOLLOWINGS')
if (settingTempCopy.hasOwnProperty('DET_BASE_VIDEOS')) DET_BASE['DET_BASE_VIDEOS'] = settingTempCopy['DET_BASE_VIDEOS']; if (DET_BASE['DET_BASE_VIDEOS']) recoverAPI('DET_BASE_VIDEOS')
if (settingTempCopy.hasOwnProperty('DET_BASE_DYNAMIC')) DET_BASE['DET_BASE_DYNAMIC'] = settingTempCopy['DET_BASE_DYNAMIC']; if (DET_BASE['DET_BASE_DYNAMIC']) recoverAPI('DET_BASE_DYNAMIC')
if (settingTempCopy.hasOwnProperty('DET_BASE_COMMENT')) DET_BASE['DET_BASE_COMMENT'] = settingTempCopy['DET_BASE_COMMENT']
_T.resetqueue()
_T.reset()
_T.resetlimit(LIMIT_EVENT)
confirm_success.style.display = 'unset'
timerFoot2 = setTimeout(() => confirm_success.style.display = 'none', 1500)
const doms = await getDom('div[xzsx_ischecked="true"],[xzsx_isdeepchecked="true"]')
doms.forEach((dom) => { dom.removeAttribute('xzsx_ischecked'); dom.removeAttribute('xzsx_isdeepchecked') })
render(true)
_render = _throttle((forceSync, isjump) => render(forceSync, isjump), INTERVAL, { leading: false, trailing: true })[0]
batchWatch()
rulesApply = deepCloneRules(rules)
let temp = {}
Object.keys(rulesApply.detect).forEach((key) => {
temp[key] = { 'color': rulesApply.detect[key].color, 'keywords': [...rulesApply.detect[key].keywords], 'antikeywords': [...rulesApply.detect[key].antikeywords] }
})
GM_setValue('xzsx_bilibili_detector_state', JSON.stringify({
MODE, DET, DEL, DEL_STYLE, TAG_MAX_LENGTH, INTERVAL, LIMIT_EVENT,
DET_BASE_MedalWall: DET_BASE['DET_BASE_MedalWall'],
DET_BASE_FOLLOWINGS: DET_BASE['DET_BASE_FOLLOWINGS'],
DET_BASE_VIDEOS: DET_BASE['DET_BASE_VIDEOS'],
DET_BASE_DYNAMIC: DET_BASE['DET_BASE_DYNAMIC'],
DET_BASE_COMMENT: DET_BASE['DET_BASE_COMMENT']
}))
GM_setValue('xzsx_bilibili_detector', JSON.stringify({
blackList: rulesApply.blackList,
detect: temp
}))
timerFoot = null
}, 500)
}
/** 页脚应用按钮松开相关操作 */
confirm_btn.onmouseup = () => {
footerProcess.style.width = '0%'
clearTimeout(timerFoot)
timerFoot = null
}
/** 页脚新增规则按钮相关操作 */
add_btn.onclick = () => {
ckey = undefined
confirm_btn.style.display = ''
add_btn.style.display = 'none'
alter_btn.style.display = 'none'
addRuleForm.className = 'addrule_form-hide'
setTimeout(() => addRuleForm.style.display = 'none', 500)
let name = tagname.value?.trim() ?? '', newKeyWords = _difference(newRuleKeysSet, deleteRuleKeysSet), newantikeywords = _difference(newAntiRuleSets, deleteAntiRuleSets)
if (name.length > 0 && newKeyWords.size > 0) {
rules.detect[tagname.value] = { 'color': tagcolor.value, 'keywords': newKeyWords, 'antikeywords': newantikeywords }
newRuleKeysSet.clear()
deleteRuleKeysSet.clear()
newAntiRuleSets.clear()
deleteAntiRuleSets.clear()
renderMainPanel()
}
}
/** 页脚修改规则按钮相关操作 */
alter_btn.onclick = () => {
confirm_btn.style.display = ''
add_btn.style.display = 'none'
alter_btn.style.display = 'none'
addRuleForm.className = 'addrule_form-hide'
setTimeout(() => addRuleForm.style.display = 'none', 500)
_difference(newRuleKeysSet, deleteRuleKeysSet).forEach((addKey) => rules.detect[ckey].keywords.add(addKey))
_difference(deleteRuleKeysSet, newRuleKeysSet).forEach((deleteKey) => rules.detect[ckey].keywords.delete(deleteKey))
_difference(newAntiRuleSets, deleteAntiRuleSets).forEach((addKey) => rules.detect[ckey].antikeywords.add(addKey))
_difference(deleteAntiRuleSets, newAntiRuleSets).forEach((deleteKey) => rules.detect[ckey].antikeywords.delete(deleteKey))
rules.detect[tagname.value] = { 'color': tagcolor.value, 'keywords': rules.detect[ckey].keywords, 'antikeywords': rules.detect[ckey].antikeywords }
ckey !== tagname.value && Reflect.deleteProperty(rules.detect, ckey)
ckey = undefined
changedSignel1.style.display = 'none'
newRuleKeysSet.clear()
deleteRuleKeysSet.clear()
newAntiRuleSets.clear()
deleteAntiRuleSets.clear()
renderMainPanel()
}
/** 渲染面板
* @returns { void }
*/
const renderMainPanel = () => {
navRuleList.querySelectorAll('.rule-item').forEach((c) => c.remove())
let cdatas = ctab ? Object.keys(rules.detect) : Object.keys(rules.blackList)
cdatas.forEach((key) => {
let parent = document.createElement('div')
parent.id = key
parent.className = 'rule-item'
parent.innerHTML =
`
${key}
<div style="display:flex;align-items:center;height:25px;">
${ctab ?
'<button class="rule-item-btn" style="display:none;">编辑</button> '
:
`<button class="rule-item-btn" style="display:none;background-color:#5d71df;border-color:#5d71df;">${rules.blackList[key]}</button> `}
<button class="rule-item-btn" style="display:none;"">删除</button>
</div>
`
/** 构建检测成分面板 */
if (ctab) {
parent.style.color = rules.detect[key].color
parent.children[0].children[0].onclick = () => {
ckey = key
parent.children[0].children[0].style.display = 'none'
parent.children[0].children[1].style.display = 'none'
renderEditPanel(key)
}
parent.children[0].children[1].onclick = () => {
parent.style.clipPath = 'polygon(115% 0, 115% 0, 100% 100%, 100% 100%)'
setTimeout(() => navRuleList.removeChild(parent), 200)
changedSignel1.style.display = ''
Reflect.deleteProperty(rules.detect, key)
rules.blackList[key] && rules.blackList[key] === '基于成分' && Reflect.deleteProperty(rules.blackList, key)
blacklistAdd_select.querySelector(`option[value="${key}"]`)?.remove()
}
/** 构建自动屏蔽面板 */
} else {
parent.children[0].children[0].onclick = () => {
let del_base = parent.children[0].children[0].textContent === '基于成分' ? '基于昵称' : '基于成分'
if (del_base === '基于成分' && rules.detect.hasOwnProperty(key)) {
parent.children[0].children[0].textContent = del_base
rules.blackList[key] = del_base
changedSignel2.style.display = ''
} else if (del_base === '基于昵称') {
parent.children[0].children[0].textContent = del_base
rules.blackList[key] = del_base
changedSignel2.style.display = ''
}
}
parent.children[0].children[1].onclick = () => {
parent.style.clipPath = 'polygon(115% 0, 115% 0, 100% 100%, 100% 100%)'
setTimeout(() => navRuleList.removeChild(parent), 200)
changedSignel2.style.display = ''
Reflect.deleteProperty(rules.blackList, key)
if (blacklistAdd_base.textContent === '基于成分' && parent.children[0].children[0].textContent === '基于成分' && blacklistAdd_select.querySelector(`option[value="${key}"]`) === null) {
let newOption = document.createElement('option')
newOption.value = key
newOption.text = key
blacklistAdd_select.appendChild(newOption)
}
}
}
parent.onmouseenter = () => {
parent.children[0].children[0].style.display = ''
parent.children[0].children[1].style.display = ''
}
parent.onmouseleave = () => {
parent.children[0].children[0].style.display = 'none'
parent.children[0].children[1].style.display = 'none'
}
navRuleList.insertBefore(parent, detectAdd_btn)
})
}
/** 渲染新增/编辑页面
* @param { string } key 当前正在编辑的标签名
* @returns { void }
*/
const renderEditPanel = (key) => {
keywords.querySelectorAll('.keyword').forEach((c) => c.remove())
antikeywords.querySelectorAll('.keyword').forEach((c) => c.remove())
ckey = key
newKeyWord.value = ''
newAntiKeyWord.value = ''
if (key === undefined) {
tagname.title = ''
tagname.value = ''
tagname.style.color = '#000000'
tagcolor.value = '#000000'
add_btn.style.display = ''
alter_btn.style.display = 'none'
} else {
tagname.title = key
tagname.value = key
tagname.style.color = rules.detect[key].color
tagcolor.value = rules.detect[key].color
/** 粉丝关键词 */
rules.detect[key].keywords.forEach((keyword) => {
let keywordDom = document.createElement('div')
keywordDom.className = 'keyword'
keywordDom.style.color = 'green'
keywordDom.innerHTML = `${keyword}<div class="keyword_delete">x</div>`
keywordDom.children[0].onclick = () => {
changedSignel1.style.display = ''
newRuleKeysSet.has(keyword) ? newRuleKeysSet.delete(keyword) : deleteRuleKeysSet.add(keyword)
keywordDom.remove(keywordDom.children[0])
}
keywords.insertBefore(keywordDom, keywords.firstChild)
})
/** 黑子关键词 */
rules.detect[key].antikeywords.forEach((antikeyword) => {
let keywordDom = document.createElement('div')
keywordDom.className = 'keyword'
keywordDom.style.color = 'rgb(243,28,28)'
keywordDom.innerHTML = `${antikeyword}<div class="keyword_delete">x</div>`
keywordDom.children[0].onclick = () => {
changedSignel1.style.display = ''
newAntiRuleSets.has(antikeyword) ? newAntiRuleSets.delete(antikeyword) : deleteAntiRuleSets.add(antikeyword)
keywordDom.remove(keywordDom.children[0])
}
antikeywords.insertBefore(keywordDom, antikeywords.firstChild)
})
add_btn.style.display = 'none'
alter_btn.style.display = ''
}
let curColor = key === undefined ? '#000000' : rules.detect[key].color
tagsample.title = key?.length > 0 ? key : '标签样式预览'
tagsample.textContent = key?.length > 0 ? key : '标签样式预览'
tagsample.style.color = getContrastColor(curColor)
tagsample.style.backgroundColor = curColor
tagsample.style.borderColor = curColor + '80'
tagsample.style.backgroundImage = `linear-gradient(45deg, ${curColor}, ${curColor + '80'})`
confirm_btn.style.display = 'none'
addRuleForm.className = 'addrule_form'
addRuleForm.style.display = 'flex'
}
/** <===================================================================================功能实现==========================================================================================> */
/** 用于去重的UID集合 */
const uidset = new Set(), url = window.location.href
/** 标签缓存
* @type { { [ key: number | string ]: { [key: string]: { fan: { isFans: boolean, sources: { [key: string]: Set<string> } }, anti: { isAnti: boolean, sources: { [key: string]: Set<string> } } } } }}
*/
let userTags = {}, userTagsHash = {}, userBlackList = new Set()
/** 对部分特殊URL进行映射
* @returns { void }
*/
const domainTransition = () => {
switch (true) {
case url.match(/^https?:\/\/www.bilibili.com\/festival.*$/) !== null: return 'www.bilibili.com/festival/'
case url.match(/^https?:\/\/www.bilibili.com\/bangumi.*$/) !== null: return 'www.bilibili.com/bangumi/'
case url.match(/^https?:\/\/www.bilibili.com\/read.*$/) !== null: return 'www.bilibili.com/read/'
case url.match(/^https?:\/\/www.bilibili.com\/blackboard.*$/) !== null: return 'www.bilibili.com/blackboard/'
case url.match(/^https?:\/\/www.bilibili.com\/v\/topic\/.*$/) !== null: return 'www.bilibili.com/v/topic/'
case url.match(/^https?:\/\/t.bilibili.com\/\?spm_id_from=.*$/) !== null: return 'space.bilibili.com'
default: return url.match(/^https?:\/\/(.*?)\/.*$/)[1]
}
}
/** 域名
* @type { string }
*/
const domain = domainTransition()
/** UserAgent生成器 */
class UserAgentGenerator {
constructor() {
this._part1 =
[
{ name: "Mozilla/5.0", children: [] }
]
this._part2 =
[
{
name: "Windows NT ${11}.${10}",
children:
[
{
name: 'Win64; x64',
children:
[
{
name: "rv:${54}.${10}",
children: []
},
{
name: "Trident/${8}.${10}",
children: []
},
]
},
{
name: '',
children:
[
{
name: "rv:${54}.${10}",
children: []
},
{
name: "Trident/${8}.${10}",
children: []
},
]
}
]
},
{
name: 'Macintosh',
children:
[
{
name: "Intel Mac OS X ${11}_${13}_${6}",
children:
[
{
name: "rv:${54}.${10}",
children: []
},
]
}
]
},
{
name: 'X11',
children:
[
{
name: 'Ubuntu',
children:
[
{
name: "rv:${54}.${10}",
children: []
},
]
},
{
name: 'Linux x86_64',
children:
[
{
name: "rv:${54}.${10}",
children: []
},
]
}
]
},
]
this._part3 =
[
{ name: "Gecko/20100101", children: [] },
{ name: "AppleWebKit/5${38}.36", children: [] },
{ name: "", children: [] },
]
this._part4 =
[
{ name: "(KHTML, like Gecko)", children: [] },
{ name: "", children: [] },
]
this._part5 =
[
{ name: "(KHTML, like Gecko)", children: [] },
{ name: "", children: [] },
]
this._part6 =
[
"Firefox/${54}.${10}",
"Chrome/${105}.${10}.${10}.${10}",
"Safari/${538}.${37}",
"OPR/${46}.0.2552.${889}",
'Ubuntu',
"Version/${11}.${10}.${10}",
"Chromium/${59}.0.3029.${111}",
"Mobile/14F89",
"(iPad; U; CPU OS 3_2 like Mac OS X; en-us)",
]
}
generate =
() => this._dfs(this._part1, '')
+ this._dfs(this._part2, '')
+ this._dfs(this._part3, '')
+ this._dfs(this._part4, '')
+ this._dfs(this._part5, '')
+ this._dfs2(this._part6, [], new Set(), Math.floor(Math.random() * this._part6.length))
_dfs = (arr, combine) => {
if (arr.length === 0) return combine
const randomIndex = Math.floor(Math.random() * arr.length)
let str = JSON.parse(JSON.stringify(arr[randomIndex].name)), matches = str.match(/\${(\d+)}/)
while (matches !== null) {
str = str.replace(/\${(\d+)}/, Math.floor(Math.random() * parseInt(matches[1])))
matches = str.match(/\${(\d+)}/)
}
combine += str + (str.length > 0 ? '; ' : '')
return this._dfs(arr[randomIndex].children, combine)
}
_dfs2 = (arr, combine, vis, total) => {
if (combine.length === total) return combine.join('; ')
let randomIndex = Math.floor(Math.random() * arr.length)
while (vis.has(randomIndex)) randomIndex = Math.floor(Math.random() * arr.length)
vis.add(randomIndex)
let str = JSON.parse(JSON.stringify(arr[randomIndex])), matches = str.match(/\${(\d+)}/)
while (matches !== null) {
str = str.replace(/\${(\d+)}/, Math.floor(Math.random() * parseInt(matches[1])))
matches = str.match(/\${(\d+)}/)
}
combine.push(str)
return this._dfs2(arr, combine, vis, total)
}
}
/** UserAgent生成器 */
const Generator = new UserAgentGenerator()
/** 获取网络请求参数
* @param { string } name API名
* @param { string } uid 用户UID
* @param { number } pn API检索页数
* @param { string } offset 主键偏移
* @returns { object } 请求参数对象
*/
const getRequestParams = (name, uid, pn, offset = '') => {
const randomUID = Math.floor(Math.random() * Math.pow(2, 29)), userAgent = Generator.generate()
switch (name) {
/** B站粉丝牌API */
case 'Medal': return {
method: "get",
url: `https://api.live.bilibili.com/xlive/web-ucenter/user/MedalWall?target_id=${uid}`,
headers: { 'user-agent': userAgent, 'referer': 'https://live.bilibili.com/' }
}
/** B站关注列表API */
case 'SubList': return {
method: "get",
url: `https://api.bilibili.com/x/relation/followings?vmid=${uid}&pn=${pn}&ps=50&order=desc&jsonp=jsonp`,
headers: { 'user-agent': userAgent, 'referer': `https://space.bilibili.com/${randomUID}/fans/follow` }
}
/** B站投稿列表API */
case 'Video': return {
method: "get",
url: `https://api.bilibili.com/x/space/arc/search?mid=${uid}&pn=${pn}&ps=50&jsonp=jsonp`,
headers: { 'user-agent': userAgent, 'referer': `https://space.bilibili.com/${randomUID}/video` }
}
/** B站用户动态API */
case 'Dynamic': return {
method: "get",
url: `https://api.bilibili.com/x/polymer/web-dynamic/v1/feed/space?&host_mid=${uid}&offset=${offset}`,
headers: { 'user-agent': userAgent, 'referer': `https://space.bilibili.com/${randomUID}/dynamic` }
}
default: return ''
}
}
/** 封装GM_xmlhttpRequest为Promise方便进行同步操作
* @param { string } params GM_xmlhttpRequest 参数列表
* @returns { Promise<object> }
*/
const GM_Request = (params) => {
return new Promise(resolve => {
GM_xmlhttpRequest({
...params,
onload: res => res.status === 200 ? resolve(JSON.parse(res.response)) : resolve({ 'code': res.status }),
onerror: error => resolve({ 'code': error.code, 'message': error })
})
})
}
/** 判断新旧版本
* @type { boolean }
*/
let version = true, Cookie = document.cookie, goOldVideo = Cookie.match(/(?<=go_old_video=)[-\d]{1,2}/)
if (Cookie && goOldVideo) version = goOldVideo[0] === '-1'
/** 新旧版本-评论根节点class映射
* @type { { [ key: string ]: { true: { mainList: string, subList: string }, false: { mainList: string, subList: string } } } }
*/
const versionMap = {
'www.bilibili.com': { true: { 'mainList': '.reply-list', 'subList': '.sub-reply-list' }, false: { 'mainList': '.comment-list ', 'subList': '.reply-box' } },
'space.bilibili.com': { true: { 'mainList': '.bili-dyn-list__items', 'subList': '.bili-dyn-item' }, false: { 'mainList': '.bili-dyn-list__items', 'subList': '.bili-dyn-item' } },
't.bilibili.com': { true: { 'mainList': '.comment-list ', 'subList': '.reply-box' } , false: { 'mainList': '.comment-list ', 'subList': '.reply-box' } },
'live.bilibili.com': { true: { 'mainList': '.chat-items', 'subList': '.reply-box' }, false: { 'mainList': '.chat-items', 'subList': '.reply-box' } },
'www.bilibili.com/festival/': { true: { 'mainList': '.reply-list,.chat-items', 'subList': '.sub-reply-list,.reply-box' }, false: { 'mainList': '.comment-list ,.chat-items', 'subList': '.reply-box' } },
'www.bilibili.com/bangumi/': { true: { 'mainList': '.comment-list ', 'subList': '.reply-box' }, false: { 'mainList': '.comment-list ', 'subList': '.reply-box' } },
'www.bilibili.com/read/': { true: { 'mainList': '.comment-list ', 'subList': '.reply-box' }, false: { 'mainList': '.comment-list ', 'subList': '.reply-box' } },
'www.bilibili.com/blackboard/': { true: { 'mainList': '.comment-list ', 'subList': '.reply-box' }, false: { 'mainList': '.comment-list ', 'subList': '.reply-box' } },
'www.bilibili.com/v/topic/': { true: { 'mainList': '.comment-list ', 'subList': '.reply-box' }, false: { 'mainList': '.comment-list ', 'subList': '.reply-box' } },
}
/** 用于屏蔽的Class映射表 @视频评论
* @type { { [ key: string ]: Array<string> } }
*/
const videoClassMap = {
'user': ['user', 'reply-item reply-wrap', '.text'],
'user-name': ['.user-info', '.reply-list > .reply-item', '.reply-content,.root-reply'],
'sub-user-name': ['.sub-user-info', '.sub-reply-list > .sub-reply-item', '.reply-content,.sub-reply-content'],
}
/** Class映射表 @动态评论
* @type { { [ key: string ]: Array<string> } }
*/
const dynamicClassMap = {
'user': ['.level-link', '.text text-con reply-item reply-wrap', '.text text-con'],
}
/** Class映射表 @直播评论
* @type { { [ key: string ]: Array<string> } }
*/
const liveClassMap = {
'chat-item danmaku-item ': ['.danmaku-item-left', '.chat-item danmaku-item ', '.danmaku-item-right'],
'chat-item danmaku-item chat-colorful-bubble': ['.danmaku-item-left', '.chat-item danmaku-item ', '.danmaku-item-right'],
'chat-item danmaku-item chat-colorful-bubble chat-emoticon bulge-emoticon': ['.danmaku-item-left', '.chat-item danmaku-item ', '.danmaku-item-right'],
'chat-item danmaku-item chat-emoticon': ['.danmaku-item-left', '.chat-item danmaku-item ', '.danmaku-item-right'],
'chat-item danmaku-item chat-emoticon bulge-emoticon': ['.danmaku-item-left', '.chat-item danmaku-item ', '.danmaku-item-right'],
}
/** 域名Class映射表
* @type { { [ key: string ]: { userDomClass: string, classMap: object } } }
*/
const domainMap = {
'www.bilibili.com': { 'userDomClass': '.user,.user-name,.sub-user-name', 'classMap': videoClassMap },
'space.bilibili.com': { 'userDomClass': '.user', 'classMap': dynamicClassMap },
't.bilibili.com': { 'userDomClass': '.user,.user-name,.sub-user-name', 'classMap': dynamicClassMap },
'live.bilibili.com': { 'userDomClass': '.chat-item,.danmaku-item', 'classMap': liveClassMap },
'www.bilibili.com/festival/': { 'userDomClass': '.user,.user-name,.sub-user-name,.chat-item,.danmaku-item', 'classMap': { ...videoClassMap, ...liveClassMap } },
'www.bilibili.com/bangumi/': { 'userDomClass': '.user,.user-name,.sub-user-name,.chat-item,.danmaku-item', 'classMap': { ...videoClassMap, ...liveClassMap } },
'www.bilibili.com/read/': { 'userDomClass': '.user,.user-name,.sub-user-name,.chat-item,.danmaku-item', 'classMap': { ...videoClassMap, ...liveClassMap } },
'www.bilibili.com/blackboard/': { 'userDomClass': '.user,.user-name,.sub-user-name,.chat-item,.danmaku-item', 'classMap': { ...videoClassMap, ...liveClassMap } },
'www.bilibili.com/v/topic/': { 'userDomClass': '.user,.user-name,.sub-user-name,.chat-item,.danmaku-item', 'classMap': { ...videoClassMap, ...liveClassMap } },
}
/** 获取UID
* @param { Element } v 用户DOM元素
* @returns { string } 返回UID字符串
*/
const getUID = (v) => v?.dataset?.uid || v?.children[0]?.dataset['usercardMid'] || v?.children[0]?.href?.replace(/[^\d]/g, "") || v.dataset['userId']
/** 获取用户昵称
* @param { Element } v 用户DOM元素
* @returns { string } 返回昵称字符串
*/
const getUname = (v) =>
(
{
'live.bilibili.com': v?.dataset?.uname,
'space.bilibili.com': v?.firstChild?.textContent,
'www.bilibili.com': version ? v?.textContent : v?.firstChild?.textContent,
't.bilibili.com': v?.firstChild?.textContent
}[domain]
)
/** 轮询获取DOM元素
* @param { string } className 类名
* @param { number } timeout 轮询周期(毫秒), 默认值为1000
* @param { boolean } isDelay 是否立刻执行一次
* @returns { Promise<Array<Element>> } 返回DOM数组
*/
const getDom = (className, timeout = 1000, isDelay = false) => {
let DOMS = [], timer = undefined
let fn = (resolve) => {
return () => {
if (ciframe === undefined) {
ciframe = getCurIframe()
/** 若页面存在iframe, 则将成分标签相关样式css样式添加到iframe内 */
if (!init && ciframe !== undefined) {
init = true
const innerIframe_style = document.createElement('style')
innerIframe_style.innerHTML = css_tag
ciframe.contentWindow.document.querySelector('head').appendChild(innerIframe_style)
}
}
DOMS = (domain === 'live.bilibili.com' ?
ciframe !== undefined ?
ciframe.contentWindow.document
:
document
:
document).querySelectorAll(className)
if (DOMS.length > 0) {
clearInterval(timer)
resolve(DOMS)
}
}
}
return new Promise(resolve => {
!isDelay && fn(resolve)()
timer = setInterval(fn(resolve), timeout)
})
}
/** 获取反差强烈的另外一种颜色
* @param { string } oldColor 十六进制字符串
* @returns { string } 返回反差强烈的另外一种颜色的十六进制字符串
*/
const getContrastColor = (oldColor) => (('0x' + oldColor.slice(1, 3)) * 299 + ('0x' + oldColor.slice(3, 5)) * 587 + ('0x' + oldColor.slice(5, 7)) * 114) / 1000 >= 128 ? 'black' : 'white'
/** 获取待个人信息字符串并检测成分
* @param { string } uid 用户UID
* @param { Element | null } dom 用户评论元素
* @param { boolean } isdeep 是否需要深度收集, 默认为否
* @param { boolean } forceSync 是否强制刷新
* @returns { Promise<void> } 返回个人信息字符串
*/
const handleStr = async (uid, dom, isdeep = false, forceSync = false) => {
/** 并发进行网络请求及成分检测 */
await Promise.all(
[
/** 获取评论/弹幕文本 */
new Promise(resolve => {
if (DET_BASE['DET_BASE_COMMENT']) {
let textStr = ' '
switch (true) {
case domain === 'space.bilibili.com' || domain === 't.bilibili.com': textStr = (dom.closest('.text text-con') || dom.querySelector('.text-con') || dom.parentNode.querySelector('.text')).textContent
break
default:
textStr = (domain === 'live.bilibili.com' ?
dom : version ?
(dom.closest(domainMap[domain].classMap[dom.className][1]) || dom || dom.parentNode)
:
dom.parentNode.className !== 'reply-con' ?
dom.parentNode
:
dom.parentNode.parentNode)?.querySelector(dom.parentNode.className === 'reply-con' ?
'.text-con'
:
domainMap[domain].classMap[dom.className][2])?.textContent
break
}
if (textStr?.trim().length > 0) dealRes(uid, '『评论内容』' + textStr, '评论', forceSync, resolve); else resolve()
} else resolve()
}),
/** 获取用户全部粉丝牌列表 */
new Promise(async (resolve) => {
if (DET_BASE['DET_BASE_MedalWall'] && !API_OVERLOAD['DET_BASE_MedalWall'].state) {
let medalStr = ' '
const medalList = await GM_Request(getRequestParams('Medal', uid, 1))
if (medalList?.code === 0 && medalList?.data?.count > 0) {
medalList?.data?.list.forEach((v) => {
let
/** 粉丝牌正主名称 */
target_name = (v?.target_name ?? ''),
/** 粉丝牌名称 */
medal_name = (v?.medal_info.medal_name ?? ''),
/** 粉丝牌等级 */
level = (v?.medal_info.level ?? '') + ''
medalStr
+= (target_name.length > 0 ? `『${level.length > 0 ? `${level}级` : ''}粉丝牌->正主』` : '') + target_name + '∏'
+ (medal_name.length > 0 ? `『${level.length > 0 ? `${level}级` : ''}粉丝牌->名称』` : '') + medal_name + '∏'
userTags[uid]['MedalWall'] = medalList?.data?.list
})
} else if (medalList?.code === 412) banAPI('DET_BASE_MedalWall', status_MedalWall)
if (medalStr.trim().length > 0) dealRes(uid, medalStr, '粉丝牌', forceSync, resolve); else resolve()
} else resolve()
}),
/** 获取用户前(50n)个投稿列表 */
new Promise(async (resolve) => {
if (DET_BASE['DET_BASE_VIDEOS'] && !API_OVERLOAD['DET_BASE_VIDEOS'].state) {
let videoStr = ' '
for (let pn = 1, count = 0; pn <= (isdeep ? 2 : 1); pn++) {
const videoList = await GM_Request(getRequestParams('Video', uid, pn))
if (videoList?.code === 0 && videoList?.data?.list?.vlist.length > 0) {
videoList?.data?.list?.vlist.forEach((v) => {
let
/** 投稿标题 */
title = (v?.title ?? ''),
/** 投稿描述 */
description = (v?.description ?? '')
videoStr
+= (title.length > 0 ? '『投稿标题』' : '') + title + '∏'
+ (description.length > 0 ? '『投稿描述』' : '') + description + '∏'
})
count += 50
if (count >= videoList?.data?.page?.count) break
} else if (videoList?.code === 412) {
banAPI('DET_BASE_VIDEOS', status_VIDEOS)
break
}
}
if (videoStr.trim().length > 0) dealRes(uid, videoStr, '投稿列表', forceSync, resolve); else resolve()
} else resolve()
}),
/** 获取用户前(50n)个关注列表 */
new Promise(async (resolve) => {
if (DET_BASE['DET_BASE_FOLLOWINGS'] && !API_OVERLOAD['DET_BASE_FOLLOWINGS'].state) {
let subStr = ' '
for (let pn = 1, count = 0; pn <= (isdeep ? 5 : 1); pn++) {
const subList = await GM_Request(getRequestParams('SubList', uid, pn))
if (subList?.code === 0 && subList?.data?.list?.length > 0) {
subList?.data?.list.forEach((v) => {
/** 关注用户的名称 */
let uname = (v?.uname ?? '')
subStr += (uname.length > 0 ? '『关注用户』' : '') + uname + '∏'
})
count += 50
if (count >= subList?.data?.total) break
} else if (subList?.code === 412) {
banAPI('DET_BASE_FOLLOWINGS', status_FOLLOWINGS)
break
}
}
if (subStr.trim().length > 0) dealRes(uid, subStr, '关注列表', forceSync, resolve); else resolve()
} else resolve()
}),
/** 获取用户前n页动态列表 */
new Promise(async (resolve) => {
if (DET_BASE['DET_BASE_DYNAMIC'] && !API_OVERLOAD['DET_BASE_DYNAMIC'].state) {
let dynamicStr = '', offset = ''
for (let pn = 1; pn <= (isdeep ? 5 : 1); pn++) {
const dynamicList = await GM_Request(getRequestParams('Dynamic', uid, pn, offset))
if (dynamicList?.code === 0 && dynamicList?.data?.items?.length > 0) {
offset = dynamicList?.data?.offset
dynamicList?.data?.items.forEach((v) => {
let
/** 装扮名称 */
decorate_name = (v?.modules?.module_author?.decorate?.name ?? ''),
/** 头像装扮名称 */
pendant_name = (v?.modules?.module_author?.pendant?.name ?? ''),
/** 转发动态的作者名称 */
module_author = (v?.orig?.modules?.module_author?.name ?? ''),
/** 动态文本内容 */
text = (v?.modules?.module_dynamic?.desc?.text ?? '')
dynamicStr
+= (decorate_name.length > 0 ? '『装扮』' : '') + decorate_name + '∏'
+ (pendant_name.length > 0 ? '『头像装扮』' : '') + pendant_name + '∏'
+ (module_author.length > 0 ? '『转发作者』' : '') + module_author + '∏'
+ (text.length > 0 ? '『动态内容』' : '') + text + '∏'
})
if (!dynamicList?.data?.has_more) break
} else if (dynamicList?.code === 412) {
banAPI('DET_BASE_DYNAMIC', status_DYNAMIC)
break
}
}
if (dynamicStr.trim().length > 0) dealRes(uid, dynamicStr, '动态列表', forceSync, resolve); else resolve()
} else resolve()
}),
]
).then(() => {
if (DEL && userBlackList.has(uid)) block(dom)
if (DET && uidset.has(uid)) {
appendTags(uid, dom, userTags[uid], forceSync)
for (const tagname of Object.keys(userTags[uid])) userTagsHash[uid]['filter'].add(tagname)
}
})
}
/** 普通/深度检测
* @param { string } uid 用户UID
* @param { Element | null } dom 用户评论元素
* @param { boolean } isdeep 是否需要深度收集, 默认为否
* @returns { Promise<void> }
*/
const check = async (uid, dom, isdeep = false) => {
/** 获取旧标签容器 */
let container = (domain === 'live.bilibili.com' ?
dom : version ?
dom.closest(domainMap[domain].classMap[dom.className][1])
:
dom.parentNode.className !== 'reply-con' ?
dom.parentNode
:
dom.parentNode.parentNode).querySelector('.tag_container')
container.innerHTML = '<img src="" style="height:18px;width:18px;"></img>'
medalWall.className = 'xzsx_detect_tag-medalwall-hide'
medalWall.setAttribute('key', null)
if (userTagsHash[uid]['isdeepchecked']) {
container.querySelector('img').remove()
appendTags(uid, dom, userTags[uid])
return
}
if (isdeep) {
/** 添加已深度检测状态 */
dom.setAttribute('xzsx_isdeepchecked', true)
userTagsHash[uid]['isdeepchecked'] = true
}
/** 检测成分 */
await handleStr(uid, dom, isdeep, true)
container.querySelector('img').remove()
isdeep && container.querySelector('.icon-deepcheck').remove()
}
/** 数字转十六进制字符串
* @param { number } num 源字符串
* @returns { string }
*/
const intToHexString = (num) => {
let hexStr = num.toString(16)
for (let i = hexStr.length; i < 6; i++) hexStr = '0' + hexStr
return hexStr
}
/** 高亮关键词
* @param { string } str 源字符串
* @param { string } targetStr 目标字符串
* @returns { string }
*/
const highLight = (str, targetStr) => str.split(targetStr).join(`<span style="background-color:yellow;color:black;">${targetStr}</span>`)
/** 瀑布流
* @param { HTMLElement } container 瀑布流布局容器
* @param { HTMLElement[] } doms HTML元素集合
* @param { number } gap 元素间隙
* @param { 'Horizontal' | 'Vertical' } direction 方向
* @return { [ number, number ] } 最大高宽
*/
const waterfall = (container, doms, gap, direction) => {
const curAttr = { 'Horizontal': ['height', 'width', 'left', 'top'], 'Vertical': ['width', 'height', 'top', 'left'] }[direction], csize = parseInt(container.style[curAttr[0]])
if (typeof (csize) !== 'number' || csize <= 0) return 0
const size = parseInt(getComputedStyle(doms[0])[curAttr[0]]), num = parseInt(csize / (size + gap))
let pos = [], lowest = -1
for (let i = 0; i < doms.length; i++) {
let computedStyle = getComputedStyle(doms[i])
doms[i].style.position = 'absolute'
if (i < num) {
pos[i] = parseInt(computedStyle[curAttr[1]]) + gap
doms[i].style[curAttr[2]] = gap + 'px'
doms[i].style[curAttr[3]] = gap * (i % num + 1) + size * (i % num) + 'px'
} else {
lowest = pos.indexOf(Math.min(...pos))
doms[i].style[curAttr[2]] = pos[lowest] + gap + 'px'
doms[i].style[curAttr[3]] = gap * (lowest % num + 1) + size * (lowest % num) + 'px'
pos[lowest] = pos[lowest] + parseInt(computedStyle[curAttr[1]]) + gap
}
}
return [pos.length * (size + gap) + gap, pos[pos.indexOf(Math.max(...pos))] + gap]
}
/** 插入标签
* @param { string } uid 用户UID
* @param { Element | null } dom 用户评论元素
* @param { object } tags 用户成分名集合
* @param { boolean } forceSync 是否强制刷新, 默认为否
* @returns { void }
*/
const appendTags = (uid, dom, tags, forceSync = false) => {
let fansTags = [], antiTags = [], iframePos = ciframe?.getBoundingClientRect()
try {
Object.keys(tags).forEach(tagname => {
if (tagname !== 'MedalWall') {
tags[tagname].fan.isFans && fansTags.push(tagname)
tags[tagname].anti.isAnti && antiTags.push(tagname)
}
})
/** 生成标签DOM */
let Tags = [], length = 0, cur = (version && domain !== 'www.bilibili.com' && domain !== 'www.bilibili.com/bangumi/' && domain !== 'www.bilibili.com/read/'&& domain !== 'www.bilibili.com/blackboard/'&& domain !== 'www.bilibili.com/v/topic/') ? (dom.querySelector(domainMap[domain].classMap[dom.className][0]) || dom.closest(domainMap[domain].classMap[dom.className][0])) : dom, tag_container = cur.querySelector('.tag_container'), existTags = new Set(), container = tag_container !== null ? tag_container : document.createElement('div')
container.querySelectorAll('.xzsx_detect_tag').forEach(ele => existTags.add(ele.textContent))
fansTags.forEach((tagname) => {
if (!existTags.has(tagname)) {
length += tagname.replace(/\p{sc=Han}/gu, '**').length
let ele = document.createElement('div')
ele.className = 'xzsx_detect_tag'
ele.style.display = length <= TAG_MAX_LENGTH ? 'inline-flex' : 'none'
ele.style.color = getContrastColor(rulesApply.detect[tagname].color)
ele.style.borderColor = rulesApply.detect[tagname].color + '80'
ele.style.backgroundImage = `-o-linear-gradient(45deg, ${rulesApply.detect[tagname].color}, ${rulesApply.detect[tagname].color + '80'})`
ele.style.backgroundImage = `-moz-linear-gradient(45deg, ${rulesApply.detect[tagname].color}, ${rulesApply.detect[tagname].color + '80'})`
ele.style.backgroundImage = `-webkit-linear-gradient(45deg, ${rulesApply.detect[tagname].color}, ${rulesApply.detect[tagname].color + '80'})`
ele.style.backgroundImage = `linear-gradient(45deg, ${rulesApply.detect[tagname].color}, ${rulesApply.detect[tagname].color + '80'})`
ele.textContent = (tags[tagname].anti.isAnti ? '🐵' : '') + tagname
/** 点击成分标签展开溯源面板 */
ele.onclick = (e) => {
e.stopPropagation()
medalWall.setAttribute('key', null)
medalWall.className = 'xzsx_detect_tag-medalwall-hide'
trackPanel.className = 'xzsx_detect_tag-trackpanel-hide'
trackPanel.innerHTML =
`
<div title='${getUname(dom)} / ${tagname}' class='xzsx_detect_tag-trackpanel-title'>
${getUname(dom)} / ${tagname}
</div>
<div class='xzsx_detect_tag-trackpanel-content'>
${tags[tagname].anti.isAnti ?
`
<div class='xzsx_detect_tag-trackpanel-anti'>
<u class='xzsx_detect_tag-trackpanel-texttitle'>黑子关键词溯源</u>
${Object.keys(tags[tagname].anti.sources).map((src) => {
return tags[tagname].anti.sources[src].size > 0 ?
`
<div>
<div class="xzsx_detect_tag-trackpanel-source">${src}</div>
${Array.from(tags[tagname].anti.sources[src]).map((text) => {
let texts = text.split('Γ')
return `<div class="xzsx_detect_tag-trackpanel-text">${highLight(texts[1], texts[0])}</div>`
}).join('')}
</div>
` : ''}).join('')}
</div>
` : '' }
<div class='xzsx_detect_tag-trackpanel-fans'>
<u class='xzsx_detect_tag-trackpanel-texttitle'>粉丝关键词溯源</u>
${Object.keys(tags[tagname].fan.sources).map((src) => {
return tags[tagname].fan.sources[src].size > 0 ?
`
<div>
<div class="xzsx_detect_tag-trackpanel-source">${src}</div>
${Array.from(tags[tagname].fan.sources[src]).map((text) => {
let texts = text.split('Γ')
return `<div class="xzsx_detect_tag-trackpanel-text">${highLight(texts[1], texts[0])}</div>`
}).join('')}
</div>
` : ''}).join('')}
</div>
</div>
`
let title = trackPanel.querySelector('.xzsx_detect_tag-trackpanel-title')
title.onclick = e => { e.stopPropagation(); ele.scrollIntoView({ block: "center", inline: "center" }) }
title.onmouseenter = e => e.target.style.opacity = 0.8
title.onmouseleave = e => e.target.style.opacity = 1
trackPanel.style.display = 'flex'
let pos = e.target.getBoundingClientRect()
if (domain === 'live.bilibili.com') {
trackPanel.style.position = 'fixed'
trackPanel.style.right = '24px'
trackPanel.style.top = '24px'
} else {
trackPanel.style.position = 'absolute'
trackPanel.style.left = (iframePos?.left ?? 0) + (pos.left + 260 > document.documentElement.clientWidth ? document.documentElement.clientWidth - 300 : pos.left + 10) + 'px'
trackPanel.style.top = (iframePos?.top ?? 0) + document.scrollingElement.scrollTop + (pos.top + 220 > document.documentElement.clientHeight ? document.documentElement.clientHeight - 250 : pos.top + 20) + 'px'
}
trackPanel.className = 'xzsx_detect_tag-trackpanel-show'
}
Tags.push(ele)
}
})
let icon = document.createElement('div'), checkBtn = document.createElement('div'), exist_btn = container.querySelector('div.icon-expend') === null
if (exist_btn && length > TAG_MAX_LENGTH) {
icon.className = 'icon-expend'
icon.innerHTML =
`
<svg t="1636097794549" class="svg-expend" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5713" >
<path d="M586.624 234.624a74.624 74.624 0 1 1-149.184 0 74.624 74.624 0 0 1 149.12 0z m0 554.624a74.624 74.624 0 1 1-149.248 0 74.624 74.624 0 0 1 149.248 0zM512 586.624a74.624 74.624 0 1 0 0-149.248 74.624 74.624 0 0 0 0 149.248z" p-id="5714" fill="#9499a0"></path>
</svg>
`
icon.onclick = e => {
e.stopPropagation()
icon.style.display = 'none'
container.querySelectorAll('div.xzsx_detect_tag[style*="display: none;"]').forEach(tag => tag.style.display = 'inline-flex')
}
}
/** 构造标签容器 */
container.className = 'tag_container'
/** 构造主动检测按钮 */
checkBtn.className = 'icon-deepcheck'
checkBtn.innerHTML =
`
<div class="icon-deepcheck-hide" >
<svg class="svg-deepcehck" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
<path d="M16.32 14.9l5.39 5.4a1 1 0 0 1-1.42 1.4l-5.38-5.38a8 8 0 1 1 1.41-1.41zM10 16a6 6 0 1 0 0-12 6 6 0 0 0 0 12z"></path>
</svg>
</div>
<div style="position: absolute;">
<svg class='check-container'>
<circle class='check-circle' cx="9" cy="9" r="8" />
</svg>
</div>
`
checkBtn.onclick = () => check(uid, dom, false)
const [_ddeep, clearDeepTimer] = _debounce(() => { checkBtn.children[0].className = 'icon-deepcheck-hide'; check(uid, dom, true) }, 1000)
checkBtn.onmousedown = () => { checkBtn.children[0].style.transform = 'scale(0.55)'; checkBtn.children[1].children[0].children[0].style.strokeDashoffset = 0; _ddeep() }
checkBtn.onmouseup = () => { checkBtn.children[0].style.transform = 'scale(1)'; checkBtn.children[1].children[0].children[0].style.strokeDashoffset = 50.3; clearDeepTimer() }
const [_dshow, clearshowTimer] = _debounce(() => checkBtn.style.display = 'inline-flex', 500)
const [_dhide, clearhideTimer] = _debounce(() => checkBtn.style.display = 'none', 1000)
container.onmouseenter = () => { clearhideTimer(); _dshow() }
container.onmouseleave = () => { clearshowTimer(); _dhide() }
dom.onmouseenter = () => { clearhideTimer(); _dshow() }
dom.onmouseleave = () => { clearshowTimer(); _dhide() }
/** 勋章墙 */
let container2 = document.createElement('div'), medalBtn = null, medalList = []
if (tags['MedalWall'] !== undefined) {
tags['MedalWall'].sort((a, b) => b.medal_info.level - a.medal_info.level).forEach((info, i) => {
let medal = document.createElement('div')
medal.title = info.target_name
medal.onclick = () => window.open(info.link)
medal.style.cssText =
`
display: inline-flex;
justify-content: center;
align-items: center;
-webkit-box-sizing: content-box;
box-sizing: content-box;
width: fit-content;
height: 16px;
line-height: 16px;
color: #fff;
border: 1px solid transparent;
border-color: #${intToHexString(info.medal_info.medal_color_border)};
white-space: nowrap;
border-radius: 2px;
font-family: "Microsoft YaHei", "Microsoft Sans Serif", "Microsoft SanSerf", "微软雅黑";
font-size: 10px;
position: relative;
cursor: pointer;
z-index: 1;
`
let medal_name = info.medal_info.medal_name, target_name = info.target_name
Object.keys(rules.detect).forEach(tagname => { rules.detect[tagname].keywords.forEach(key => { medal_name = highLight(medal_name, key); target_name = highLight(target_name, key) }); rules.detect[tagname].antikeywords.forEach(antikey => medal_name = highLight(medal_name, antikey)) })
medal.innerHTML =
`
<div>
<span>${medal_name}</span>
</div>
<div>
<span style="margin-left:4px;margin-right:4px;">${target_name}</span>
</div>
<div style="color:#${intToHexString(info.medal_info.medal_color_start)};width: 16px; text-align: center; border-top-left-radius: 1px; border-bottom-right-radius: 1px;background-color: white;box-shadow: 0px 0px 4px #ddd;">${info.medal_info.level}</div>
`
medal.children[0].style.cssText =
`
display: -webkit-box;
display: -ms-flexbox;
display: flex;
-webkit-box-pack: center;
-ms-flex-pack: center;
justify-content: center;
-webkit-box-align: center;
-ms-flex-align: center;
align-items: center;
min-width: 12px;
text-align: center;
padding: 0 4px;
color: #fff;
border-top-left-radius: 1px;
border-bottom-left-radius: 1px;
background-image: linear-gradient(45deg, #${intToHexString(info.medal_info.medal_color_start)}, #${intToHexString(info.medal_info.medal_color_end)});
box-shadow: 0px 0px 4px #ddd;
`
medal.children[1].style.cssText =
`
color:black;
box-shadow: 2px 2px 0px #ddd inset;
`
medal.onmouseenter = e => e.target.style.opacity = 0.8
medal.onmouseleave = e => e.target.style.opacity = 1
medalList.push(medal)
if (i === 0) {
let clone = medal.cloneNode(true)
clone.children[0].children[0].textContent = info.medal_info.medal_name
clone.children[1].children[0].textContent = info.target_name
medalBtn = clone
}
})
}
let existMedal = cur.querySelector('.fans-medal-item')
if (tags.hasOwnProperty('MedalWall')) {
cur.querySelector('.xzsx_detect_tag-medalwall-medalBtn')?.remove()
container2.className = 'xzsx_detect_tag-medalwall-medalBtn'
container2.appendChild(medalBtn)
if (domain === 'live.bilibili.com' && existMedal !== null) {
container2 = existMedal
container2.style.cursor = 'pointer'
container2.onmouseenter = e => e.target.style.opacity = 0.8
container2.onmouseleave = e => e.target.style.opacity = 1
document.querySelector('#fans-medal-popover')?.remove()
}
container2.onclick = (e) => {
e.stopPropagation()
let ckey = medalWall.getAttribute('key'), key = getUname(dom)
if (ckey !== null && ckey === key) return
medalWall.setAttribute('key', key)
medalWall.innerHTML =
`
<div title="${getUname(dom)} / 勋章墙" class="xzsx_detect_tag-medalwall-title">${getUname(dom)} / 勋章墙</div>
<div class="xzsx_detect_tag-medalwall-container"></div>
`
let medalContainer = medalWall.querySelector('.xzsx_detect_tag-medalwall-container'), medalTitle = medalWall.querySelector('.xzsx_detect_tag-medalwall-title')
medalTitle.onclick = () => dom.scrollIntoView({ block: "center", inline: "center" })
medalTitle.onmouseenter = e => e.target.style.opacity = 0.8
medalTitle.onmouseleave = e => e.target.style.opacity = 1
medalContainer.style.height = medalList.length < 25 ? '120px' : '240px'
medalContainer.append(...medalList)
const [size1, size2] = waterfall(medalContainer, medalList, 8, 'Horizontal'), pos = e.target.getBoundingClientRect()
trackPanel.className = 'xzsx_detect_tag-trackpanel-hide'
medalWall.className = 'xzsx_detect_tag-medalwall-hide'
if (domain === 'live.bilibili.com') {
medalWall.style.position = 'fixed'
medalWall.style.right = '24px'
medalWall.style.top = '24px'
} else {
medalWall.style.position = 'absolute'
medalWall.style.left = (iframePos?.left ?? 0) + (pos.left + 260 > document.documentElement.clientWidth ? document.documentElement.clientWidth - 300 : pos.left + 10) + 'px'
medalWall.style.top = (iframePos?.top ?? 0) + document.scrollingElement.scrollTop + (pos.top + 220 > document.documentElement.clientHeight ? document.documentElement.clientHeight - 250 : pos.top + 20) + 'px'
}
medalWall.style.display = 'flex'
medalWall.style.height = size1 + (size2 > 400 ? 38 : 34) + 'px'
medalWall.style.width = size2 + 'px'
medalContainer.style.height = size1 + 'px'
medalContainer.style.width = size2 + 'px'
medalTitle.style.width = size2 - 8 + 'px'
medalWall.scrollLeft = 0
medalWall.className = 'xzsx_detect_tag-medalwall-show'
/** 滚轮控制横向滚动 */
if (size2 > 400) {
medalWall.addEventListener("wheel", e => {
e.preventDefault()
medalWall.scrollLeft += e.deltaY
})
}
}
}
container.append(domain === 'live.bilibili.com' ? '' : container2, ...Tags, exist_btn && length > TAG_MAX_LENGTH ? icon : '', forceSync ? checkBtn : (JSON.parse((dom.attributes?.xzsx_isdeepchecked?.value ?? false)) || userTagsHash[uid]['isdeepchecked']) ? '' : checkBtn)
tag_container === null && cur.append(container)
if (domain === 'live.bilibili.com' && existMedal == null) cur?.insertBefore(container2, cur.firstChild)
} catch (error) {
console.log(error);
}
}
/** 屏蔽操作
* @param { Element | null } user 用户评论元素
* @returns { void }
*/
const block = (user) => {
let closest = domain === 'live.bilibili.com' ?
user : version ?
user.closest(domainMap[domain].classMap[user.className][1])
:
user.parentNode.className !== 'reply-con' ?
user.parentNode
:
user.parentNode.parentNode
switch (DEL_STYLE) {
case '消息屏蔽':
closest = closest.querySelector(user.parentNode.className === 'reply-con' ?
'.text-con'
:
domainMap[domain].classMap[user.className][2])
closest.style.fontStyle = 'oblique'
closest.textContent = '[消息已删除]'
break
case '完全删除':
closest.style.display = 'none'
break
/** ... */
default:
break
}
}
/** 查重
* @param { string } uid 用户UID
* @param { Element | null } user 用户评论元素
* @returns { void }
*/
const filte = (uid, user) => {
if (DEL && userBlackList.has(uid)) block(user)
if (DET && (!DEL || DEL_STYLE !== '完全删除') && uidset.has(uid)) appendTags(uid, user, userTags[uid])
return uidset.has(uid) || userBlackList.has(uid)
}
/** 处理用户数据
* @param { string } uid 用户UID
* @param { string } resStr 用户信息字符串
* @param { string } source 字符串来源
* @param { boolean } forceSync 是否强制刷新
* @param { function } resolve 调用结束Promise状态
* @returns { void }
*/
const dealRes = (uid, resStr, source, forceSync, resolve) => {
try {
if (DEL) {
for (const key of Object.keys(rulesApply.blackList)) {
if (rulesApply.blackList[key] === '基于成分' && rulesApply.detect.hasOwnProperty(key) && rulesApply.detect[key].keywords.size > 0) {
rulesApply.detect[key].keywords.forEach(key => resStr.indexOf(key) !== -1 && userBlackList.add(uid))
}
}
}
if (DET) {
/** 结合黑子关键词atikeyword与粉丝关键词keyword查成分 */
for (const tagname of Object.keys(rulesApply.detect)) {
if ((forceSync || !userTagsHash[uid]['filter'].has(tagname)) && rulesApply.detect[tagname].keywords.size > 0) {
if (!userTags[uid].hasOwnProperty(tagname)) {
userTags[uid][tagname] = { fan: { isFans: false, sources: {} }, anti: { isAnti: false, sources: {} } }
}
rulesApply.detect[tagname].keywords.forEach((keyword) => {
let index = resStr.indexOf(keyword)
while (index !== -1) {
userTags[uid][tagname].fan.isFans = true
if (!userTags[uid][tagname].fan.sources.hasOwnProperty(source)) {
userTags[uid][tagname].fan.sources[source] = new Set()
}
/** 截取关键词前后十个字符 */
let begin = index, L = index, R = index + keyword.length, countL = 0, countR = 0
while (L > 0 && resStr[L] !== '』' && countL++ < 10) { --L }
while (R < resStr.length && resStr[R] !== '∏' && countR++ < 10) { ++R }
while (begin > 0 && resStr[begin] !== '『') { begin-- }
userTags[uid][tagname].fan.sources[source].add(keyword + 'Γ' + resStr.slice(begin, resStr.indexOf('』', begin) + 1) + resStr.slice(L + 1, R))
index = resStr.indexOf(keyword, index + 1)
}
})
if (rulesApply.detect[tagname].antikeywords.size > 0) {
rulesApply.detect[tagname].antikeywords.forEach((antikeyword) => {
let index = resStr.indexOf(antikeyword)
while (index !== -1) {
userTags[uid][tagname].anti.isAnti = true
if (!userTags[uid][tagname].anti.sources.hasOwnProperty(source)) {
userTags[uid][tagname].anti.sources[source] = new Set()
}
/** 截取关键词前后十个字符 */
let begin = index, L = index, R = index + antikeyword.length, countL = 0, countR = 0
while (L > 0 && resStr[L] !== '』' && countL++ < 10) { --L }
while (R < resStr.length && resStr[R] !== '∏' && countR++ < 10) { ++R }
while (begin > 0 && resStr[begin] !== '『') { begin-- }
userTags[uid][tagname].anti.sources[source].add(antikeyword + 'Γ' + resStr.slice(begin, resStr.indexOf('』', begin) + 1) + resStr.slice(L + 1, R))
index = resStr.indexOf(antikeyword, index + 1)
}
})
}
}
}
uidset.add(uid)
}
} catch (error) {
console.log(error)
} finally {
resolve()
}
}
/** 渲染行为
* @param { boolean } forceSync 是否强制刷新, 默认为否
* @param { boolean } isJump 是否插队
* @returns { Promise<void> }
*/
const render = async (forceSync = false, isJump = false) => {
if (!DET && !DEL) return
try {
/** 获取用户元素集合 */
let DOMS = Array.from(await getDom(...[domainMap[domain].userDomClass, ...domain === 'space.bilibili.com' ? [500, true] : []])), curDocument = domain === 'live.bilibili.com' ? ciframe !== undefined ? ciframe.contentWindow.document : document : document
/** 若需要插队则对数组排序进行特殊处理 */
if (isJump) DOMS = DOMS.slice(0, _T._limit - _T._count).concat(DOMS.slice(_T._limit - _T._count, DOMS.length).reverse())
for (const user of DOMS) {
/** 过滤不在视野内的元素 */
const pos = user.getBoundingClientRect()
if (pos.top + pos.height < 0 || pos.top > curDocument.documentElement.clientHeight) continue
/** 获取UID和昵称 */
const uid = getUID(user), uname = getUname(user)
/** 过滤未设置的Class 排除非评论元素 */
if (!domainMap[domain].classMap.hasOwnProperty(user.className) || !uid) continue
/** 过滤已查过成分的DOM元素 */
if (!forceSync && JSON.parse(user.attributes?.xzsx_ischecked?.value ?? false)) continue
else user.setAttribute('xzsx_ischecked', true)
/** 过滤已查过成分的用户 */
if (!forceSync && filte(uid, user)) continue
/** 基于昵称屏蔽 */
if (DEL && rules.blackList.hasOwnProperty(uname?.trim()) && rules.blackList[uname?.trim()] === '基于昵称') {
userBlackList.add(uid)
block(user)
}
/** 基于成分贴标签/屏蔽 */
if (DET) {
if (!userTagsHash.hasOwnProperty(uid)) {
userTagsHash[uid] = { isdeepchecked: false, filter: new Set() }
userTags[uid] = {}
}
/** 根据模式选择不同的处理方式 */
switch (MODE) {
case '自动':
_T.enqueue({ uid, user, isdeep: false, forceSync }, isJump)
break
case '静默':
appendTags(uid, user, [], true)
continue
default:
break
}
}
}
} catch (error) {
console.log(error)
}
}
/** 节流渲染次数
* @type { () => void }
*/
let _render = _throttle((forceSync, isJump) => render(forceSync, isJump), INTERVAL, { leading: false, trailing: true })[0]
/** <===================================================================================列表监听===========================================================================================> */
/** 监听器浏览器兼容
* @type { MutationObserver }
*/
const MutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver
/** 监听元素
* @param { boolean } params getDom 参数
* @param { boolean } isMark 是否开启优化, 默认为false
* @param { boolean } breakCondition 跳出情况, 默认为false
* @param { boolean } extra 额外需要执行的函数, 默认为空函数
* @returns { () => Promise<void> }
*/
const Listen = (params, isMark = false, breakCondition = false, extra = async () => { }) => {
if (!Array.isArray(params) || typeof (breakCondition) !== 'boolean' || typeof (isMark) !== 'boolean' || typeof (extra) !== 'function' || breakCondition) return async () => { }
return async () => {
const DOMS = await getDom(...params)
for (const dom of DOMS) {
if (isMark && dom.attributes?.xzsx_islistened?.value) continue
new MutationObserver(() => { _render(false, true); extra() }).observe(dom, { childList: true })
isMark && dom.setAttribute('xzsx_islistened', true)
}
}
}
/** 动态页面监听子列表
* @type { () => Promise<void> }
*/
const ListenSubSubReplyList = Listen(['.comment-list,.reply-box', ...domain === 'space.bilibii.com' ? [250, true] : []], true)
/** 监听子评论列表
* @type { () => Promise<void> }
*/
const ListenSubReplyList = Listen(
[versionMap[domain][version].subList],
true,
domain === 'live.bilibili.com',
async () => {
if (domain === 'space.bilibili.com') {
await Listen(['.comment-list', 250, true], true, false, async () => ListenSubSubReplyList())()
await ListenSubSubReplyList()
}
}
)
/** 监听主评论列表
* @type { () => Promise<void> }
*/
const ListenReplyList = Listen([versionMap[domain][version].mainList], true, false, async () => await ListenSubReplyList())
/** 滚动页面触发渲染&防抖 */
const _dscroll = _debounce(() => render(), 100, { leading: false, trailing: true })[0]
/** 监听直播页面的特殊滚动条
* @returns { Promise<void> }
*/
const ListenScrollBar = async () => {
const scroller = await getDom('#chat-history-list > div.ps__scrollbar-y-rail')
scroller.forEach(ele => new MutationObserver(() => _dscroll()).observe(ele, { attributeFilter: ['style'] }))
}
/** 批量监听
* @returns { Promise<void> }
*/
const batchListen = async () => {
/** 监听主列表 */
await ListenReplyList()
/** 监听子列表 */
await ListenSubReplyList()
/** 动态页面额外监听的子列表 */
domain === 'space.bilibili.com' && await ListenSubSubReplyList()
/** 直播页面额外监听列表特殊滚动条 */
domain === 'live.bilibili.com' && await ListenScrollBar()
}
/** 初始渲染 */
_render()
/** 初始监听 */
await batchListen()
/** 监听页面滚动并检测视野范围内的用户成分 */
window.onscroll = _dscroll
/** 单页面监听URL变化重新监听评论列表 */
history.replaceState = bindHistoryEvent('replaceState')
window.onhashchange = async () => { _T.resetqueue(); _T.reset(); await batchListen() }
window.onpopstate = async () => { _T.resetqueue(); _T.reset(); await batchListen() }
window.addEventListener('replaceState', async () => { _T.resetqueue(); _T.reset(); await batchListen() })
}
script()