// ==UserScript== // @name b站首页黑名单 屏蔽首页视频 // @description 屏蔽b站首页推荐中的指定up // @namespace https://github.com/kuzen // @version 1.8.0 // @author kuzen // @icon https://www.google.com/s2/favicons?domain=bilibili.com // @run-at document-start // @include *://www.bilibili.com/ // @include *://www.bilibili.com/?* // @grant GM_setValue // @grant GM_getValue // @grant GM_deleteValue // @grant GM_addStyle // @grant GM_log // @grant GM_addElement // @license MIT // ==/UserScript== /* ==UserConfig== blockList: uid: title: uid黑名单 description: uid黑名单,注意若格式填写有问题则会影响脚本运行!格式为 ["xxx", "xxx"] default: s[] ==/UserConfig== */ (function() { 'use strict'; class Setting { constructor(blockList) { this.blockList = blockList; const deleteIcon = ``; this.css = { settingsPanel: `#brlb-settings { font-size: 12px; color: #6d757a } #brlb-settings h1 { color: #161a1e } #brlb-settings a { color: #00a1d6 } #brlb-settings a:hover { color: #f25d8e } #brlb-settings input { margin-left: 3px; margin-right: 3px } #brlb-settings label { width: 100%; display: inline-block; cursor: pointer } #brlb-settings label:after { content: ""; width: 0; height: 1px; background: #4285f4; transition: width .3s; display: block } #brlb-settings label:hover:after { width: 100% } form { margin: 0 } #brlb-settings input[type=radio] { -webkit-appearance: radio; -moz-appearance: radio; appearance: radio } #brlb-settings input[type=checkbox] { -webkit-appearance: checkbox; -moz-appearance: checkbox; appearance: checkbox } .brlb-block-line-delete { background:url(${deleteIcon}); width: 16px; background-repeat: no-repeat; background-position: center }`, bui: `.bui, .bui-tabs .bui-tabs-header { display: -webkit-box; display: -ms-flexbox; display: flex } .bui { vertical-align: middle; -webkit-box-align: center; -ms-flex-align: center; align-items: center; -webkit-box-pack: center; -ms-flex-pack: center; justify-content: center } .bui-tabs { -webkit-box-pack: start; -ms-flex-pack: start; justify-content: flex-start } .bui-tabs .bui-tabs-header { margin-bottom: 8px } .bui-tabs .bui-tabs-header .bui-tabs-header-item { text-align: center; margin-right: 20px; font-size: 12px; color: #212121; cursor: pointer } .bui-tabs .bui-tabs-header .bui-tabs-header-item.bui-tabs-header-item-active { color: #00a1d6; border-bottom: 1px solid #00a1d6 } .bui-tabs .bui-tabs-body .bui-tabs-body-item { display: none } .bui-tabs .bui-tabs-body .bui-tabs-body-item.bui-tabs-body-item-active { display: block } .bui-button { display: -webkit-inline-box; display: -ms-inline-flexbox; display: inline-flex; min-width: 68px; height: 24px; font-size: 12px; border-radius: 2px; -webkit-box-sizing: border-box; box-sizing: border-box; -webkit-transition: .2s; -o-transition: .2s; transition: .2s; -webkit-transform: translateZ(0); transform: translateZ(0); background: 0 0; padding: 0; outline: 0; color: inherit; text-align: inherit; line-height: inherit } .bui-button.bui-button-border, .bui-button.bui-button-transparent { color: #fff; border: 1px solid rgba(255, 255, 255, .2) } .bui-button.bui-button-border:hover, .bui-button.bui-button-transparent:hover { color: #00a1d6; border-color: #00a1d6 } .bui-button, .bui-button.bui-button-border { cursor: pointer } .bui-button.bui-button-border.bui-button-disabled { background: 0 0; color: rgba(255, 255, 255, .2); border: 1px solid rgba(255, 255, 255, .1) } .bui-button.bui-button-border.bui-button-disabled:hover { background: 0 0; color: rgba(255, 255, 255, .2) } .bui-button.bui-button-white { color: #757575; border: 1px solid silver; background-color: #fff } .bui-button.bui-button-white:hover { color: #00a1d6; border-color: #00a1d6 } .bui-button.bui-button-gray { background-color: #e5e9ef; color: #212121 } .bui-button.bui-button-gray:hover { background-color: #00a1d6; color: #fff } .bui-button.bui-button-gray2 { color: #505050; background-color: #f4f4f4 } .bui-button.bui-button-gray2:hover { background-color: #f4f4f4; color: #222 } .bui-button.bui-button-gray2.bui-button-disabled, .bui-button.bui-button-gray2.bui-button-disabled:hover { background-color: #f4f4f4; color: #ccd0d7 } .bui-button.bui-button-gray3 { color: #999 } .bui-button.bui-button-blue, .bui-button.bui-button-gray3:hover { background-color: #00a1d6; color: #fff } .bui-button.bui-button-blue:hover { background-color: #00b5e5 } .bui-button.bui-button-blue2 { color: #00a1d6; background-color: #fff; border: 1px solid #00a1d6 } .bui-button.bui-button-blue2:hover { background-color: #00a1d6; color: #fff } .bui-button.bui-button-yellow { background-color: #f5b23d; color: #fff } .bui-button.bui-button-yellow:hover { background-color: #ffc154 } .bui-button.bui-button-text { color: #00a1d6 } .bui-button.bui-button-text:hover { color: #00b5e5 } .bui-button.bui-button-disabled { cursor: default; background: #f5f7fa; color: silver; border: 0 } .bui-button.bui-button-disabled:hover { background: #f5f7fa; color: silver }`, brlbBlockList: `.brlb-block-setting { padding-bottom: 24px } .brlb-block-label { font-weight: 700; font-size: 12px; color: #18191c; vertical-align: middle } .brlb-block-tabpanel-row { zoom: 1; line-height: 20px; margin-bottom: 4px; font-size: 0 } .input-row { display: -webkit-box; display: -ms-flexbox; display: flex } .brlb-block-tabpanel { position: relative; height: auto; -webkit-transition: height .3s; transition: height .3s; pointer-events: auto } .brlb-block-tabpanel.no-bottom { padding-bottom: 0; border-bottom: 0 } .brlb-block-tablist { margin: 0 16px; transition-timing-function: cubic-bezier(.165, .84, .44, 1); transition-duration: 0s; transform: translateX(0) translateY(0) translateZ(1px); transition-property: transform } .brlb-block-wrap { width: 320px; flex: none; border-bottom: 1px solid #e3e5e7; touch-action: pan-x; user-select: none; -webkit-user-drag: none; -webkit-tap-highlight-color: transparent; height: 377px } .brlb-block-string { -webkit-box-sizing: border-box; box-sizing: border-box; width: 75%; margin-right: 6px; padding: 1px 10px 1px 5px; border-radius: 2px; vertical-align: middle; background-color: #fff; color: #18191c; font-size: 12px; border: 1px solid #e3e5e7; height: 20px; line-height: 20px; display: inline-block } .bui-button-gray { background-color: #f1f2f3; color: #18191c; min-width: -webkit-fit-content; min-width: -moz-fit-content; min-width: fit-content; -webkit-box-flex: 1; flex: 1 } .brlb-block-empty, .brlb-block-list-function { text-align: center; color: #9499a0; color: var(--text3, #9499a0) } .brlb-block-list-function { display: -webkit-box; display: -ms-flexbox; display: flex; -webkit-box-pack: justify; -ms-flex-pack: justify; justify-content: space-between; padding: 0 6px; font-size: 12px; line-height: 24px } .brlb-block-empty { display: none; width: 100%; height: 100%; line-height: 100px } .brlb-block-line { display: -webkit-box; display: -ms-flexbox; display: flex; -webkit-box-pack: justify; -ms-flex-pack: justify; justify-content: space-between; padding-left: 5px; height: 24px; color: #18191c; color: var(--text1, #18191c); background: #fff; background: var(--bg1, #fff); position: relative; font-size: 100% } .brlb-block-line>div { font-size: 12px; line-height: 24px; white-space: nowrap; height: 24px; text-overflow: ellipsis } .brlb-block-line-content { text-align: left; display: inline-block; width: 150px; padding-left: 4px } .icon-general-del { -webkit-touch-callout: none; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; font-family: bilibili-new-iconfont !important; font-size: 16px; font-style: normal; line-height: inherit; vertical-align: top; -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale; -webkit-transition: color .3s; transition: color .3s } .brlb-block-list-function-delete { padding-right: 16px } .brlb-block-line-delete { padding-right: 36px }`, }; this.listWrap = null; const btnWarpCallback = (mutationsList, _observer) => { setTimeout(() => { this.addSettingBtn(); }, 100); this.btnWarpObserver.disconnect(); }; this.btnWarpObserver = new MutationObserver(btnWarpCallback); const targetNode = document.getElementById('i_cecream'); const config = {attributes: false, childList: true, subtree: false}; this.btnWarpObserver.observe(targetNode, config); } // TODO refreshList(key) { if (this.listWrap) { this.listWrap.innerHTML = ''; this.addItems(key); } } addItems(key, text) { const itemDom = createElement('div', { className: 'brlb-block-line', }, [createElement('div', { className: 'brlb-block-line-content', }), createElement('span', { className: 'brlb-block-line-delete', })]); if (this.listWrap) { if (text == null) { for (const uid of this.blockList.list[key]) { const item = itemDom.cloneNode(true); item.getElementsByClassName('brlb-block-line-content')[0].innerText = uid; this.listWrap.appendChild(item); } } else { const item = itemDom.cloneNode(true); item.getElementsByClassName('brlb-block-line-content')[0].innerText = text; this.listWrap.appendChild(item); } } } addSettingBtn() { const addBtnClick = (event) => { // TODO const uid = event.currentTarget.parentElement.getElementsByClassName('brlb-block-string')[0].value; if (uid.length > 0) { GM_log(uid); this.blockList.add('uid', uid); this.addItems('uid', uid); } }; const brlbBlockListWrap = createElement('div', { className: 'brlb-block-setting', }, [ createElement('div', { className: 'brlb-block-label', }, '屏蔽列表'), createElement('div', { className: 'brlb-block brlb-block-wrap', }, [ createElement('div', { className: 'brlb-block-tablist', }, [ createElement('div', { className: 'brlb-block-tabpanel', role: 'list', }, [ createElement('div', { className: 'brlb-block-tabpanel-row input-row', }, [ createElement('input', { type: 'text', className: 'brlb-block-string', placeholder: '添加屏蔽词,正则以"/"开头"/"结尾', }), createElement('div', { className: 'brlb-block-string-btn bui bui-button bui-button-gray', role: 'button', event: { click: addBtnClick, }, }, [createElement('span', {}, '添加')]), ]), ]), createElement('div', { className: '

黑名单内容

`; cardView.replaceWith(newCardView); newCardView.parentElement.dataset.blocked = '1'; newCardView.parentElement.dataset.brlbUid = uid.toString(); return newCardView; } unblockCardView(cardView, id) { // 再次点击取消屏蔽 const cv = this.history[cardView.parentElement.parentElement.className][id]; cardView.replaceWith(cv); cv.parentElement.dataset.blocked = '0'; return cv; } // TODO: 合并广告链接检测 getUid(cardView) { const owner = cardView.getElementsByClassName('bili-video-card__info--owner')[0].href; if (owner.length > 0) { const uid = owner.substr(owner.lastIndexOf('/') + 1); return uid; } else { return cardView.parentElement.dataset.brlbUid; } } isAd(cardView) { return cardView.getElementsByClassName('bili-video-card__info--ad-img').length > 0; } // 换一换 rollObserver(recommendContainer) { const rollCallback = (mutationsList, observer) => { const recommendList = recommendContainer.getElementsByClassName('bili-video-card__wrap'); this.history[recommendContainer.className] = Array.from(recommendList); this.run(recommendList); }; const rollObse = new MutationObserver(rollCallback); const config = {attributes: false, childList: true, subtree: false}; rollObse.observe(recommendContainer, config); } register(container) { const cardViewList = container.getElementsByClassName('bili-video-card__wrap'); this.history[container.className] = Array.from(cardViewList); this.run(cardViewList); this.setBlockBtnEvent(container); } run(cardViewList) { let index = 0; for (let cardView of cardViewList) { if (this.isAd(cardView)) { // 广告 cardView = this.blockCardView(cardView, 0); this.addBlockBtn(cardView); this.setCardViewEvent(cardView); } else { // 普通视频 const uid = this.getUid(cardView); if (uid != null && this.blockList.isContained('uid', uid) === true) { cardView = this.blockCardView(cardView, uid); } this.addBlockBtn(cardView); this.setCardViewEvent(cardView); } cardView.parentElement.dataset.brlbId = index.toString(); index++; } } } class BlockList { constructor() { // 处理历史遗留问题(逃 this.list = JSON.parse(GM_getValue('blockList') || null); if (this.list != null) { if (this.list instanceof Array) { this.list = {'uid': this.list, 'username': [], 'title': []}; } Object.entries(this.list).forEach(([key, value]) => { this.list[key] = this.list[key].sort(); this.removeDuplicates(key); GM_setValue(`blockList.${key}`, JSON.stringify(this.list[key])); }); GM_deleteValue(`blockList`); } // 新版本读取 this.list = {'uid': [], 'username': [], 'title': []}; Object.entries(this.list).forEach(([key, value]) => { this.list[key] = JSON.parse(GM_getValue(`blockList.${key}`) || '[]'); this.list[key] = this.list[key].sort(); this.removeDuplicates(key); }); GM_log(`黑名单列表:${JSON.stringify(this.list)}`); } length(key) { return this.list[key].length; } isContained(key, item) { return (this.list[key][this.search(key, item)] === item); } add(key, item) { const index = this.search(key, item); if (this.list[key][index] !== item) { this.list[key].splice(index, 0, item); GM_setValue(`blockList.${key}`, JSON.stringify(this.list[key])); return true; } return false; } remove(key, item) { const index = this.search(key, item); if (this.list[key][index] === item) { this.list[key].splice(index, 1); GM_setValue(`blockList.${key}`, JSON.stringify(this.list[key])); return true; } return false; } clr() { GM_log(`清空黑名单`); GM_setValue('blockList.uid', '[]'); GM_setValue('blockList.username', '[]'); GM_setValue('blockList.title', '[]'); this.list = {'uid': [], 'username': [], 'title': []}; } search(key, target) { const n = this.list[key].length; let left = 0; let right = n - 1; let ans = n; while (left <= right) { const mid = ((right - left) >> 1) + left; if (target <= this.list[key][mid]) { ans = mid; right = mid - 1; } else { left = mid + 1; } } return ans; } removeDuplicates(key) { const n = this.list[key].length; if (n === 0) { return 0; } let r = 1; let l = 1; while (r < n) { if (this.list[key][r] !== this.list[key][r - 1]) { this.list[key][l] = this.list[key][r]; ++l; } ++r; } return l; } } // eslint-disable-next-line valid-jsdoc /** * refer to: https://github.com/ipcjs/bilibili-helper */ function createElement(type, props, children) { let elem = null; if (type === 'text') { return document.createTextNode(props); } else { elem = document.createElement(type); } for (const n in props) { if (n === 'style') { // eslint-disable-next-line guard-for-in for (const x in props.style) { elem.style[x] = props.style[x]; } } else if (n === 'className') { elem.className = props[n]; } else if (n === 'event') { // eslint-disable-next-line guard-for-in for (const x in props.event) { elem.addEventListener(x, props.event[x]); } } else { props[n] !== undefined && elem.setAttribute(n, props[n]); } } if (children) { if (typeof children === 'string') { elem.innerHTML = children; } else { for (let i = 0; i < children.length; i++) { if (children[i] != null) { elem.appendChild(children[i]); } } } } return elem; } window.addEventListener('DOMContentLoaded', () => { const blockList = new BlockList(); const biliBlocker = new BiliBlocker(blockList, true); const recommendContainer = document.querySelectorAll('div[class^="recommend-container__"]')[0]; if (recommendContainer != null) { const evaContainer = document.querySelectorAll('div[class^="eva-extension-body"]')[0]; biliBlocker.register(recommendContainer); biliBlocker.register(evaContainer); // 延迟一会,避免重复处理 setTimeout(() => { biliBlocker.rollObserver(recommendContainer); }, 100); } }, false); })();