// ==UserScript== // @name bilibili直播净化 // @namespace https://github.com/lzghzr/GreasemonkeyJS // @version 4.2.39 // @author lzghzr // @description 屏蔽聊天室礼物以及关键字, 净化聊天室环境 // @icon  // @supportURL https://github.com/lzghzr/GreasemonkeyJS/issues // @match https://live.bilibili.com/* // @match https://www.bilibili.com/blackboard/* // @license MIT // @require https://github.com/lzghzr/TampermonkeyJS/raw/5036ad2ff5875d313df654b8feca13cdb69d0d00/xhook/xhook.js#sha256=7l4G7dFdCp6GZibUe8Z3vl6jb6Q1yCjUrfUxI8k2KFY= // @compatible chrome 基础功能需要 88 以上支持 :not() 伪类,高级功能需要 105 及以上支持 :has() 伪类 // @compatible edge 基础功能需要 88 以上支持 :not() 伪类,高级功能需要 105 及以上支持 :has() 伪类 // @compatible firefox 基础功能需要 84 以上支持 :not() 伪类,高级功能需要 121 及以上支持 :has() 伪类 // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @grant unsafeWindow // @run-at document-start // @downloadURL none // ==/UserScript== const W = typeof unsafeWindow === 'undefined' ? window : unsafeWindow; class NoVIP { elmStyleCSS; chatObserver; danmakuObserver; Start() { this.elmStyleCSS = GM_addStyle(''); this.AddCSS(); const chatMessage = new Map(); this.chatObserver = new MutationObserver(mutations => { mutations.forEach(mutation => { mutation.addedNodes.forEach(addedNode => { if (addedNode instanceof HTMLDivElement && addedNode.classList.contains('danmaku-item')) { const chatNode = addedNode.querySelector('.danmaku-item-right'); if (chatNode !== null) { const chatText = chatNode.innerText; const dateNow = Date.now(); if (chatMessage.has(chatText) && dateNow - chatMessage.get(chatText) < 10_000) { addedNode.classList.add('NoVIP_chat_hide'); } else { chatMessage.set(chatText, dateNow); } } } }); }); }); const elmDivChatList = document.querySelector('#chat-items'); if (elmDivChatList !== null) { this.chatObserver.observe(elmDivChatList, { childList: true }); } const danmakuMessage = new Map(); this.danmakuObserver = new MutationObserver(mutations => { mutations.forEach(mutation => { mutation.addedNodes.forEach(addedNode => { const danmakuNode = addedNode instanceof Text ? addedNode.parentElement : addedNode; if (danmakuNode?.classList?.contains('bili-dm')) { const danmakuText = danmakuNode.innerText.split(/ ?[x×]\d+$/); const dateNow = Date.now(); if (danmakuMessage.has(danmakuText[0]) && dateNow - danmakuMessage.get(danmakuText[0]) < 10_000) { danmakuNode.classList.add('NoVIP_danmaku_hide'); } else if (danmakuText[1] !== undefined) { danmakuNode.classList.add('NoVIP_danmaku_hide'); } else { danmakuMessage.set(danmakuText[0], dateNow); } } }); }); }); const elmDivDanmaku = document.querySelector('#live-player'); if (elmDivDanmaku !== null) { this.danmakuObserver.observe(elmDivDanmaku, { childList: true, subtree: true }); } setInterval(() => { const dateNow = Date.now(); chatMessage.forEach((value, key) => { if (dateNow - value > 60_000) { chatMessage.delete(key); } }); danmakuMessage.forEach((value, key) => { if (dateNow - value > 60_000) { danmakuMessage.delete(key); } }); }, 60_000); const docObserver = new MutationObserver(mutations => { mutations.forEach(mutation => { mutation.addedNodes.forEach(addedNode => { if (addedNode instanceof HTMLDivElement) { if (addedNode.classList.contains('dialog-ctnr')) { const blockEffectCtnr = addedNode.querySelector('.block-effect-ctnr'); if (blockEffectCtnr !== null) { this.AddUI(blockEffectCtnr); } } else if (addedNode.classList.contains('control-panel-ctnr')) { const blockEffectBtnSvg = addedNode.querySelector('.block-effect-btn svg'); if (blockEffectBtnSvg !== null) { blockEffectBtnSvg.innerHTML = ' '; console.info(...scriptName('脚本 icon 已加载')); } else { console.error(...scriptName('插入脚本 icon 失效')); } } } }); }); }); docObserver.observe(document, { childList: true, subtree: true }); const block = localStorage.getItem('LIVE_BLCOK_EFFECT_STATE'); if (block?.includes('2')) { localStorage.setItem('LIVE_BLCOK_EFFECT_STATE', '2'); } else { localStorage.setItem('LIVE_BLCOK_EFFECT_STATE', ''); } this.ChangeCSS(); } NORoomSkin() { if (config.menu.noRoomSkin.enable) { W.roomBuffService.__NORoomSkin = true; W.roomBuffService.unmount(); } else { W.roomBuffService.__NORoomSkin = false; W.roomBuffService.mount(W.roomBuffService.__NORoomSkin_skin); } } ChangeCSS() { let height = 62; let cssText = ` /* 统一用户名颜色 */ .chat-item .user-name { color: var(--brand_blue) !important; }`; if (config.menu.noGuardIcon.enable) { cssText += ` /* 聊天背景 */ .chat-item.chat-colorful-bubble { background-color: unset !important; border-image-source: unset !important; border-radius: unset !important; display: block !important; margin: unset !important; } /* 聊天背景 */ .chat-item.chat-colorful-bubble div:has(div[style*="border-image-source"]), /* 欢迎提示条 */ #welcome-area-bottom-vm:has(.sama-avatar-box), /* 粉丝勋章内标识 */ .chat-item .fans-medal-item-ctnr .medal-guard, /* 舰队指挥官标识 */ .chat-item .pilot-icon, .chat-item .pilot-icon ~ br, /* 订阅舰长 */ .chat-item.guard-buy { display: none !important; } /* 兼容chrome 105以下版本 */ @supports not selector(:has(a, b)) { .chat-item.chat-colorful-bubble div[style*="border-image-source"] #welcome-area-bottom-vm { display: none !important; } }`; } if (config.menu.noWealthMedalIcon.enable) { cssText += ` /* 聊天背景, 存疑 */ .chat-item.wealth-bubble { border-image-source: unset !important; } /* 聊天背景, 存疑 */ .chat-item.has-bubble { border-image-source: unset !important; border-image-slice: unset !important; border-image-width: unset !important; box-sizing: unset !important; display: block !important; margin: unset !important; } /* 欢迎提示条 */ #welcome-area-bottom-vm:has(.wealth-medal), /* 弹幕 */ .bili-dm > .bili-icon, /* 聊天 */ .chat-item .wealth-medal-ctnr { display: none !important; } /* 兼容chrome 105以下版本 */ @supports not selector(:has(a, b)) { #welcome-area-bottom-vm { display: none !important; } }`; } if (config.menu.noGiftMsg.enable) { height -= 32; cssText += ` /* 底部小礼物, 调整高度 */ .chat-history-list.with-penury-gift { height: 100% !important; } /* 热门流量推荐 */ .chat-item.hot-rank-msg, /* VIP标识 */ #activity-welcome-area-vm, .chat-item .vip-icon, .chat-item.welcome-msg, /* 高能标识 */ .chat-item.top3-notice, .chat-item .rank-icon, /* 分享直播间 */ .chat-item.important-prompt-item, /* 礼物栏 */ .gift-control-panel > *:not(.left-part-ctnr), #web-player__bottom-bar__container, #chat-gift-bubble-vm, #penury-gift-msg, #gift-screen-animation-vm, #my-dear-haruna-vm .super-gift-bubbles, .chat-item.gift-item, .chat-item.system-msg, .web-player-inject-wrap .announcement-wrapper, .bilibili-live-player-video-operable-container>div:first-child>div:last-child, .bilibili-live-player-video-gift, .bilibili-live-player-danmaku-gift { display: none !important; }`; } if (config.menu.noSystemMsg.enable) { height -= 30; cssText += ` .chat-history-list.with-brush-prompt { height: 100% !important; } /* 目前只看到冲榜提示 */ .chat-history-panel #all-guide-cards, /* 聊天下方滚动消息,进场、点赞之类的 */ #brush-prompt, /* 初始系统提示 */ .chat-item.convention-msg, /* 各种野生消息 */ .chat-item.common-danmuku-msg, /* 各种野生消息 x2 */ .chat-item.misc-msg, /* 各种野生消息 x3 (Toasts) */ .link-toast, /* pk */ .chat-item.new-video-pk-item-dm { display: none !important; }`; } if (config.menu.noSuperChat.enable) { cssText += ` /* 调整 SuperChat 聊天框 */ .chat-history-list { padding-top: 5px !important; } .chat-item.superChat-card-detail { margin-left: unset !important; margin-right: unset !important; min-height: unset !important; } .chat-item .card-item-middle-top { background-color: unset !important; background-image: unset !important; border: unset !important; display: inline !important; padding: unset !important; } .chat-item .card-item-middle-top-right { display: unset !important; } .chat-item .superChat-base { display: unset !important; height: unset !important; line-height: unset !important; vertical-align: unset !important; width: unset !important; } .chat-item .superChat-base .fans-medal-item-ctnr { margin-right: 4px !important; } .chat-item .name, .chat-item .card-item-name { display: unset !important; font-size: unset !important; font-weight: unset !important; height: unset !important; line-height: 20px !important; margin-left: unset !important; opacity: unset !important; overflow: unset !important; text-overflow: unset !important; vertical-align: unset !important; white-space: unset !important; width: unset !important; } .chat-item .card-item-name>span { color: var(--brand_blue) !important; } /* 为 SuperChat 用户名添加 : */ .chat-item.superChat-card-detail .name:after, .chat-item.superChat-card-detail .card-item-name>span:after { content: ' : '; } .chat-item .card-item-middle-bottom { background-color: unset !important; display: unset !important; padding: unset !important; } .chat-item .input-contain { display: unset !important; } .chat-item .text { color: var(--text2) !important; } /* SuperChat 提示条 */ #chat-msg-bubble-vm, /* SuperChat 保留条 */ #pay-note-panel-vm, .chat-item .bottom-background, /* SuperChat 聊天条 右上角电池 */ .chat-item .card-item-top-right, /* SuperChat 按钮 */ #chat-control-panel-vm .super-chat { display: none !important; }`; } if (config.menu.noEmoticons.enable) { cssText += ` #chat-control-panel-vm .emoticons-panel, .chat-item.chat-emoticon { display: none !important; }`; } if (config.menu.noEmotDanmaku.enable) { cssText += ` .bili-dm > img:not(.bili-icon) { display: none !important; }`; } if (config.menu.noLikeBtn.enable) { cssText += ` /* 点赞按钮 */ #chat-control-panel-vm .like-btn, /* 点赞消息 */ .chat-item[data-type="6"], /* 点赞数 */ #head-info-vm .icon-ctnr:has(.like-icon) { display: none !important; } /* 兼容chrome 105以下版本 */ @supports not selector(:has(a, b)) { #head-info-vm .like-icon, #head-info-vm .like-text { display: none !important; } }`; } if (config.menu.noGiftControl.enable) { cssText += ` /* 排行榜 */ .rank-list-section .gift-rank-cntr .top3-cntr .default, .rank-list-section .guard-rank-cntr:not(.open) .guard-empty { height: 42px !important; } .rank-list-section .guard-rank-cntr:not(.open) .guard-empty { background-size: contain !important; background-position: center !important; background-repeat: no-repeat !important; } .rank-list-section .gift-rank-cntr .top3-cntr .default-msg { bottom: -12px !important; } .rank-list-section, .rank-list-section.new .rank-list-ctnr[style*="height: 178px;"] { height: 98px !important; } .rank-list-section .tab-content, .rank-list-section .tab-content-pilot, .rank-list-section.new .guard-rank-cntr .rank-list-cntr { min-height: unset !important; } .rank-list-section .tab-content[style*="height: 9"], .rank-list-section .tab-content-pilot[style*="height: 9"], .rank-list-section .gift-rank-cntr .top3-cntr { height: 64px !important; } .rank-list-section .guard-rank-cntr .top3-cntr > span { height: 32px !important; } .rank-list-section.new .gift-rank-cntr .top3-cntr, .rank-list-section.new .guard-rank-cntr { height: unset !important; } .rank-list-section.new .gift-rank-cntr .top3-cntr { padding-top: 5px !important; } .rank-list-section.new .guard-rank-cntr .top3-cntr { top: 15px !important; } /* 调整聊天区 */ .rank-list-section~.chat-history-panel { height: calc(100% - 98px - 145px) !important; } /* 直播分区 */ .live-area { display: flex !important; } /* 排行榜 */ .rank-list-section.new .gift-rank-cntr .top3 > div ~ div, .rank-list-section.new .guard-rank-cntr .top3-cntr > span ~ span, .rank-list-section.new .pilot, /* 人气榜 */ #head-info-vm .popular-and-hot-rank, /* 礼物星球 */ #head-info-vm .gift-planet-entry, /* 活动榜 */ #head-info-vm .activity-entry, /* 粉丝团 */ #head-info-vm .follow-ctnr, /* 礼物按钮 */ #web-player-controller-wrap-el .web-live-player-gift-icon-wrap, /* 头像框 */ .blive-avatar-pendant, /* 主播城市 */ .anchor-location, /* 水印 */ .web-player-icon-roomStatus, .blur-edges-ctnr, /* 遮罩 */ #web-player-module-area-mask-panel { display: none !important; }`; } if (config.menu.noFansMedalIcon.enable) { cssText += ` /* 团体勋章 */ .chat-item .group-medal-ctnr, /* 团体勋章 底部提示条 */ #brush-prompt .group-medal-ctnr, /* 粉丝勋章 聊天 */ .chat-item .fans-medal-item-ctnr, /* 粉丝勋章 底部提示条 */ #brush-prompt .fans-medal-item-ctnr { display: none !important; }`; } if (config.menu.noLiveTitleIcon.enable) { cssText += ` .chat-item .title-label { display: none !important; }`; } if (config.menu.noRaffle.enable) { cssText += ` body:not(.player-full-win):has(#anchor-guest-box-id)[style*="overflow: hidden;"] { overflow-y: overlay !important; } #shop-popover-vm, #anchor-guest-box-id, #player-effect-vm, #chat-draw-area-vm, /* 天选之类的 */ .gift-control-panel .left-part-ctnr, .anchor-lottery-entry, .popular-main .lottery { display: none !important; } /* 兼容chrome 105以下版本 */ @supports not selector(:has(a, b)) { body:not(.player-full-win)[style*="overflow: hidden;"] { overflow-y: overlay !important; } }`; } if (config.menu.noDanmakuColor.enable) { cssText += ` .bili-dm { color: #ffffff !important; }`; } if (config.menu.noGameId.enable) { cssText += ` /* 总容器 */ .web-player-inject-wrap, /* PK */ /* #pk-vm, */ /* #awesome-pk-vm, */ /* #chaos-pk-vm, */ /* 多人连麦 */ /* #multi-voice-index, */ /* #multi-player, */ /* 互动游戏 */ #game-id, /* 连麦 */ #chat-control-panel-vm .voice-rtc, /* 帮玩 */ #chat-control-panel-vm .play-together-service-card-container, /* 一起玩 */ #chat-control-panel-vm .play-together-entry, /* 神秘人 */ .chat-item .common-nickname-medal { display: none !important; }`; } if (config.menu.noBBChat.enable) { cssText += ` /* 官方 */ #aside-area-vm #combo-card, #aside-area-vm #combo-danmaku-vm, #aside-area-vm .vote-card, /* 自定义 */ .chat-item.NoVIP_chat_hide { display: none !important; }`; } if (config.menu.noBBDanmaku.enable) { cssText += ` /* 官方 */ .danmaku-item-container .bilibili-combo-danmaku-container, .danmaku-item-container .combo { display: none !important; } /* 自定义 */ .bili-dm.NoVIP_danmaku_hide, /* 官方 */ .danmaku-item-container .mode-adv { color: transparent !important; text-shadow: unset !important; }`; } cssText += ` .chat-history-list.with-penury-gift.with-brush-prompt { height: calc(100% - ${height}px) !important; }`; this.NORoomSkin(); this.elmStyleCSS.innerHTML = cssText; } AddUI(addedNode) { const elmUList = addedNode.firstElementChild; elmUList.childNodes.forEach(child => { if (child instanceof Comment) { child.remove(); } }); const listLength = elmUList.childElementCount; if (listLength > 10) { return; } const changeListener = (itemHTML, x) => { const itemSpan = itemHTML.querySelector('span'); const itemInput = itemHTML.querySelector('input'); itemInput.checked = config.menu[x].enable; itemInput.checked ? selectedCheckBox(itemSpan) : defaultCheckBox(itemSpan); itemInput.addEventListener('change', ev => { const evt = ev.target; evt.checked ? selectedCheckBox(itemSpan) : defaultCheckBox(itemSpan); config.menu[x].enable = evt.checked; GM_setValue('blnvConfig', encodeURI(JSON.stringify(config))); this.ChangeCSS(); }); }; const selectedCheckBox = (spanClone) => { spanClone.classList.remove('checkbox-default'); spanClone.classList.add('checkbox-selected'); }; const defaultCheckBox = (spanClone) => { spanClone.classList.remove('checkbox-selected'); spanClone.classList.add('checkbox-default'); }; const replaceItem = (listNodes, x) => { for (const child of listNodes) { if (child.innerText === config.menu[x].replace) { return child; } } }; const itemHTML = elmUList.firstElementChild.cloneNode(true); const itemInput = itemHTML.querySelector('input'); const itemLabel = itemHTML.querySelector('label'); itemInput.id = itemInput.id.replace(/\d/, ''); itemLabel.htmlFor = itemLabel.htmlFor.replace(/\d/, ''); let i = listLength + 10; const listNodes = elmUList.childNodes; for (const x in config.menu) { const child = replaceItem(listNodes, x); if (child === undefined) { const itemHTMLClone = itemHTML.cloneNode(true); const itemInputClone = itemHTMLClone.querySelector('input'); const itemLabelClone = itemHTMLClone.querySelector('label'); itemInputClone.id += i; itemLabelClone.htmlFor += i; i++; itemLabelClone.innerText = config.menu[x].name; changeListener(itemHTMLClone, x); elmUList.appendChild(itemHTMLClone); } else { const itemHTMLClone = child.cloneNode(true); const itemLabelClone = itemHTMLClone.querySelector('label'); itemLabelClone.innerText = config.menu[x].name; changeListener(itemHTMLClone, x); child.remove(); elmUList.appendChild(itemHTMLClone); } } } AddCSS() { GM_addStyle(` /* 多行菜单 */ .border-box.dialog-ctnr.common-popup-wrap.top-left[style*="width: 200px;"] { width: 270px !important; } .block-effect-ctnr .item { float: left; } .block-effect-ctnr .item .cb-icon { left: unset !important; margin-left: -6px; } .block-effect-ctnr .item label { width: 84px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; } /* 隐藏网页全屏榜单 */ .player-full-win .rank-list-section { display: none !important; } .player-full-win .chat-history-panel:not([style]) { height: calc(100% - 135px) !important; } .player-full-win .chat-history-panel.new { height: calc(100% - 135px) !important; }`); } } const defaultConfig = { version: 1710342294818, menu: { noGiftMsg: { name: '屏蔽礼物相关', replace: '屏蔽全部礼物及广播', enable: false }, noSystemMsg: { name: '屏蔽系统消息', replace: '屏蔽进场信息', enable: false }, noSuperChat: { name: '屏蔽醒目留言', replace: '屏蔽醒目留言', enable: false }, noEmoticons: { name: '屏蔽表情聊天', replace: '屏蔽表情动画(右下角)', enable: false }, noEmotDanmaku: { name: '屏蔽表情弹幕', replace: '屏蔽表情弹幕', enable: false }, noLikeBtn: { name: '屏蔽点赞按钮', enable: false }, noGiftControl: { name: '屏蔽活动控件', enable: false }, noGuardIcon: { name: '屏蔽舰队标识', enable: false }, noWealthMedalIcon: { name: '屏蔽荣耀勋章', enable: false }, noFansMedalIcon: { name: '屏蔽粉丝勋章', enable: false }, noLiveTitleIcon: { name: '屏蔽成就头衔', enable: false }, noRaffle: { name: '屏蔽抽奖橱窗', enable: false }, noDanmakuColor: { name: '屏蔽弹幕颜色', enable: false }, noGameId: { name: '屏蔽互动游戏', enable: false }, noBBChat: { name: '屏蔽刷屏聊天', enable: false }, noBBDanmaku: { name: '屏蔽刷屏弹幕', enable: false }, noRoomSkin: { name: '屏蔽房间皮肤', enable: false }, noActivityPlat: { name: '屏蔽活动皮肤', enable: false }, noRoundPlay: { name: '屏蔽视频轮播', enable: false }, noSleep: { name: '屏蔽挂机检测', enable: false }, invisible: { name: '隐身入场', enable: false } } }; const userConfig = GM_getValue('blnvConfig', null) === null ? defaultConfig : JSON.parse(decodeURI(GM_getValue('blnvConfig'))); let config; if (userConfig.version === undefined || userConfig.version < defaultConfig.version) { for (const x in defaultConfig.menu) { try { defaultConfig.menu[x].enable = userConfig.menu[x].enable; } catch (error) { console.error(...scriptName('载入配置失效'), error); } } config = defaultConfig; } else { config = userConfig; } if (location.href.match(/^https:\/\/live\.bilibili\.com\/(?:blanc\/)?\d/)) { W.getComputedStyle = new Proxy(W.getComputedStyle, { apply: function (target, _this, args) { if (args !== undefined && args[0] instanceof HTMLElement) { let htmlEle = Reflect.apply(target, _this, args); htmlEle = new Proxy(htmlEle, { get: function (_target, propertyKey) { if (propertyKey === 'display' && _target[propertyKey] === 'none') { return 'block'; } return Reflect.get(_target, propertyKey); } }); return htmlEle; } return Reflect.apply(target, _this, args); } }); Object.defineProperty(W, '__NEPTUNE_IS_MY_WAIFU__', {}); let push = 1 << 4; W.webpackChunklive_room = W.webpackChunklive_room || []; W.webpackChunklive_room.push = new Proxy(W.webpackChunklive_room.push, { apply: function (target, _this, args) { for (const [name, fn] of Object.entries(args[0][1])) { let fnStr = fn.toString(); if (fnStr.includes('return this.chatList.children.length')) { const regexp = /(?return )this\.chatList\.children\.length/s; const match = fnStr.match(regexp); if (match !== null) { fnStr = fnStr.replace(regexp, '$this.chatList.querySelectorAll(".danmaku-item:not(.NoVIP_hide)").length'); console.info(...scriptName('增强聊天显示已加载')); } else { console.error(...scriptName('增强聊天显示失效'), fnStr); } push |= 1 << 0; } if (config.menu.noRoundPlay.enable) { if (fnStr.includes('case"PREPARING":')) { const regexp = /(?case"PREPARING":)(?\w+\((?\w+)\);break;)/s; const match = fnStr.match(regexp); if (match !== null) { fnStr = fnStr.replace(regexp, '$$.round=0;$'); console.info(...scriptName('屏蔽下播轮播已加载')); } else { console.error(...scriptName('屏蔽下播轮播失效'), fnStr); } push |= 1 << 1; } } else { push |= 1 << 1; } if (config.menu.noSleep.enable) { if (fnStr.includes('prototype.sleep=function(')) { const regexp = /(?prototype\.sleep=function\(\w*\){)/; const match = fnStr.match(regexp); if (match !== null) { fnStr = fnStr.replace(regexp, '$return;'); console.info(...scriptName('屏蔽挂机检测已加载')); } else { console.error(...scriptName('屏蔽挂机检测失效'), fnStr); } push |= 1 << 2; } } else { push |= 1 << 2; } if (config.menu.invisible.enable) { if (fnStr.includes('this.enterRoomTracker=new ')) { const regexp = /(?this\.enterRoomTracker=new \w+),/s; const match = fnStr.match(regexp); if (match !== null) { fnStr = fnStr.replace(regexp, '$,this.enterRoomTracker.report=()=>{},'); console.info(...scriptName('房间心跳隐身已加载')); } else { console.error(...scriptName('房间心跳隐身失效'), fnStr); } push |= 1 << 3; } } else { push |= 1 << 3; } if (fn.toString() !== fnStr) { args[0][1][name] = str2Fn(fnStr); } if (isAllBitsSet(push)) { W.webpackChunklive_room.push = target; break; } } return Reflect.apply(target, _this, args); } }); let add = 1 << 1; Set.prototype.add = new Proxy(Set.prototype.add, { apply: function (target, _this, args) { if (args[0] && args[0] instanceof Function) { let fnStr = args[0].toString(); if (config.menu.noRoundPlay.enable) { if (fnStr.includes('.Preparing:')) { const regexp = /(?Preparing:)(?.*?1===(?\w+)\.round)/s; const match = fnStr.match(regexp); if (match !== null) { fnStr = fnStr.replace(regexp, '$$.round=0;$'); console.info(...scriptName('屏蔽下播轮播已加载')); } else { console.error(...scriptName('屏蔽下播轮播失效'), fnStr); } add |= 1 << 0; } } else { add |= 1 << 0; } if (args[0].toString() !== fnStr) { args[0] = str2Fn(fnStr); } if (isAllBitsSet(add)) { Set.prototype.add = target; } } return Reflect.apply(target, _this, args); } }); xhook.before((request) => { if (config.menu.invisible.enable) { if (request.url.includes('/web-room/v1/index/getInfoByUser')) { request.url = request.url.replace('not_mock_enter_effect=0', 'not_mock_enter_effect=1'); console.info(...scriptName('隐身入场已拦截')); } } if (config.menu.noRoundPlay.enable) { if (request.url.includes('/live/getRoundPlayVideo')) { request.url = request.url.replace(/room_id=\d+/, 'room_id='); console.info(...scriptName('屏蔽视频轮播已拦截')); } } }); xhook.after((request, response) => { if (request.url.includes('/xlive/app-room/v2/guardTab/topList')) { response.text = response.text.replace(/"anchor_guard_achieve_level":\d+/, '"anchor_guard_achieve_level":0'); console.info(...scriptName('屏蔽大航海榜单背景图已拦截')); } if (config.menu.noRoundPlay.enable) { if (request.url.includes('/xlive/web-room/v2/index/getRoomPlayInfo')) { response.text = response.text.replace('"live_status":2', '"live_status":0'); console.info(...scriptName('屏蔽视频轮播已拦截')); } } }); if (config.menu.noActivityPlat.enable) { if (self === top) { if (location.pathname.startsWith('/blanc')) { history.replaceState(null, '', location.href.replace(`${location.origin}/blanc`, location.origin)); } else { location.href = location.href.replace(location.origin, `${location.origin}/blanc`); } } else { top?.postMessage(location.origin + location.pathname, 'https://live.bilibili.com'); top?.postMessage(location.origin + location.pathname, 'https://www.bilibili.com'); } } document.addEventListener('readystatechange', () => { if (document.readyState === 'interactive') { if (W.roomBuffService.mount !== undefined) { W.roomBuffService.mount = new Proxy(W.roomBuffService.mount, { apply: function (target, _this, args) { if (args[0] !== undefined) { _this.__NORoomSkin_skin = args[0]; if (args[0].id !== 0) { _this.__NORoomSkin_skin_id = args[0].id; } if (_this.__NORoomSkin) { args[0].id = 0; args[0] = {}; } else if (args[0].id === 0 && args[0].start_time !== 0) { args[0].id = _this.__NORoomSkin_skin_id || 0; } } return Reflect.apply(target, _this, args); } }); W.roomBuffService.unmount = new Proxy(W.roomBuffService.unmount, { apply: function (target, _this, args) { if (_this.__NORoomSkin_skin !== undefined) { _this.__NORoomSkin_skin.id = 0; } return Reflect.apply(target, _this, args); } }); } } if (document.readyState === 'complete') { new NoVIP().Start(); } }); } else if (location.href.includes('bilibili.com/blackboard/')) { if (config.menu.noActivityPlat.enable) { W.addEventListener("message", msg => { if (msg.origin === 'https://live.bilibili.com' && msg.data.startsWith('https://live.bilibili.com/blanc/')) { location.href = msg.data; } }); } } function str2Fn(str) { const fnReg = str.match(/([^\{]*)\{(.*)\}$/s); if (fnReg !== null) { const [, head, body] = fnReg; const args = head.replaceAll(/function[^\(]*|[\s()=>]/g, '').split(','); return new Function(...args, body); } } function isAllBitsSet(value) { if (value === 0) { return false; } return (value & (value + 1)) === 0; } function scriptName(name) { return [ `%c${GM_info.script.name}%c ${name}`, "font-weight: bold; color: white; background-color: #FF6699; padding: 1px 4px; border-radius: 4px;", "font-weight: bold; color: #FF6699;" ]; }