// ==UserScript== // @name 哔哩哔哩新版首页排版调整和去广告(bilibili) // @namespace http://tampermonkey.net/ // @version 1.3.5 // @author LingLing // @description 对新版B站首页的每行显示的视频数量进行调整, 同时删除所有广告 (大尺寸屏幕每行将显示更多的视频) // @license MIT // @icon  // @match *://www.bilibili.com/* // @exclude *://www.bilibili.com/all* // @exclude *://www.bilibili.com/video* // @exclude *://www.bilibili.com/anime* // @exclude *://www.bilibili.com/pgc* // @exclude *://www.bilibili.com/live* // @exclude *://www.bilibili.com/article* // @exclude *://www.bilibili.com/upuser* // @exclude *://www.bilibili.com/match* // @exclude *://www.bilibili.com/platform* // @exclude *://www.bilibili.com/bangumi* // @exclude *://www.bilibili.com/cheese* // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @compatible chrome // @compatible edge // @compatible firefox // @downloadURL none // ==/UserScript== (function () { ("use strict"); // >>>>> 请在浏览器右上角的油猴插件的设置面板中设置该插件的部分功能 <<<<< // 类名表 (或选择器) const classMap = { 标题: "h3 a", 作者: "bili-video-card__info--owner", // 含日期. 仅作者: bili-video-card__info--author 分类: "floor-title", vDom: "container", // 视频区域 的容器元素 nav: "bili-header__bar", // 导航栏的元素 banner: "bili-header__banner", // 横幅背景的元素 btn: "roll-btn", // 右侧换一换按钮 btn2: "flexible-roll-btn", // 新版右下角换一换按钮 }; const vDom = document.querySelector("." + classMap.vDom); const nav = document.querySelector("." + classMap.nav); const banner = document.querySelector("." + classMap.banner); if (!vDom) { return; } // 默认值 const base_isClearAd = true; // 是否删除'广告'(屏蔽视频). 默认 true const base_isTrueEnd = false; // 是否将广告移至预加载视频的后面. 默认 false const base_isLoadOne = false; // 是否视频全加载. 默认 false const base_videoNumRule = "0,1500,2; 1500,1800,3; 1800,3000,4; 3000,3700,5; 3700,6300,6"; const base_delClassArr = "广告, 推广"; // 获取存储的数据 const isClearAd = getValue("setting_isClearAd", base_isClearAd); const isTrueEnd = getValue("setting_isTrueEnd", base_isTrueEnd); const isLoadOne = getValue("setting_isLoadOne", base_isLoadOne); const videoNumRule = getValue("setting_videoNumRule", base_videoNumRule); // 视频排列规则, 其他尺寸按照初始方式排列 const delClassArr = getValue("setting_delClassArr", base_delClassArr); // 屏蔽的类名列表, 子元素包含某类名也可屏蔽 // console.log(isClearAd, isTrueEnd, isLoadOne, videoNumRule, delClassArr); // 屏蔽的类名表 const delClassMap = { 广告: "bili-video-card__info--ad", 推广: "bili-video-card__info--creative-ad", 特殊: "floor-single-card", 直播: "living", // 分类=直播 番剧: "分类=番剧", 综艺: "分类=综艺", 课堂: "分类=课堂", 漫画: "分类=漫画", 国创: "分类=国创", 电影: "分类=电影", 纪录片: "分类=纪录片", 电视剧: "分类=电视剧", }; // 设置的文本 const settingText = { isClearAd: `是否删除广告, 若不删除则会将所有广告移至视频列表的最后 默认: 是 (确定) 当前: `, isTrueEnd: `是否将广告移至预加载视频的后面, 关闭后广告将放置在预加载视频的前面 一般视频的后面. 开启的效率更高 默认: 否 (取消) 当前: `, isLoadOne: `是否在进入网站时加载视口区域的全部视频, 开启时视频将会全部加载, 但会闪一下 默认: 否 (取消) 当前: `, delClassArr: `屏蔽设置, 可根据需要自行修改, 可自定义, 每项用 ; 分隔 ----默认: 广告; 推广 ----可选: ('特殊'包含了直播 番剧 课堂...) 广告、推广、特殊、直播、番剧、综艺、课堂、漫画、国创、电影、纪录片、电视剧 ----自定义: 1. 标题=xxx, 可屏蔽标题含xxx的视频, xxx部分支持&&运算符, 如: 标题=A&&B, 表示屏蔽标题同时含有A B内容的视频 2. 作者=xxx, 可屏蔽作者名和发布日期中含xxx的视频`, videoNumRule: `视频排列规则, 每条规则用 ; 分隔. 其他尺寸按照初始方式排列 示例: 1450,2400,4 表示浏览器宽度在1450~2400像素时每行显示4个视频(前两行) 默认: 0,1500,2; 1500,1800,3; 1800,3000,4; 3000,3700,5; 3700,6300,6`, }; const errKeyArr = ["", "_2"]; const errKeyInfo = { disNum: "setting_err_disNum", errNum: "setting_err_num", errTime: "setting_err_time", isTip: "setting_err_isTip", }; const disErrTipNum = 3; // 每小段报错弹窗提醒次数 (短时间内的提醒次数) const errTipNum = 5 * disErrTipNum; // 报错弹窗的总提醒次数 const errTipInterval = 2; // 每段报错弹窗提醒时间间隔(小时) const errNumReset = 5; // 报错次数重置的天数 const queryNum = 0; // 处理的视频数量, 对前 queryNum 个视频中的广告进行处理(删除或置后), 0表示对全部视频进行处理. 默认 0 const marginTop1 = 40; // 第三行视频的上边距 const marginTop2 = 24; // 第四行及以上视频的上边距 let cssDom; let cssText; let oldCssText; let isChange = false; // 每行视频数是否需要变化 let showVideoNum = 3; // 当前每行显示的视频数 (以第一行为准), 网站默认值为3 let videoNum = 0; // 视频总数 let newVideoNum = 0; // 新获取的视频总数 let firstAdIndex = 0; // 第一个广告的索引 let pageZoom = 1; // 页面缩放 let w = getW(); // 浏览器视口宽度 videoNum = getVideoNum(vDom); // 计算当前视频总数 let adArr = getAd(queryNum, delClassArr, newVideoNum, 1); delAd(adArr, vDom); // 将所有广告放置在最后 或 删除 setTimeout(() => { delAdFn(); loadTopVideo(); }, 1000); zoomPage(); // 缩放页面 setStyle(); // 调整视频排列 resetErrInfo(); // 重置err相关的数据 let rollBtn; let btnSvg; let rollBtn2; // 刷新视频 window.addEventListener("click", () => { if (!rollBtn) { adArr = getAd(showVideoNum * 3 + 2, delClassArr, newVideoNum, 1); delAd(adArr, vDom); rollBtn = document.querySelector("button." + classMap.btn); // 换一换按钮 btnSvg = rollBtn && rollBtn.querySelector("svg"); // 换一换按钮的旋转图标 // 点击按钮后对新视频中的广告进行处理 if (btnSvg) { btnSvg.addEventListener("transitionend", () => { // console.log("视频刷新成功"); adArr = getAd(showVideoNum * 3 + 2 + 3, delClassArr, newVideoNum, 1); !isTrueEnd && adArr.forEach((item) => { item.forEach((adItem) => { adItem.style.display = "block"; }); }); delAd(adArr, vDom); }); } else { rollBtn && rollBtn.addEventListener("click", () => { setTimeout(() => { adArr = getAd( showVideoNum * 3 + 2 + 3, delClassArr, newVideoNum, 1 ); delAd(adArr, vDom); }, 500); }); } } if (!rollBtn2) { adArr = getAd(queryNum, delClassArr, newVideoNum, 1); delAd(adArr, vDom); rollBtn2 = document.querySelector("." + classMap.btn2); // 新版右下角的换一换按钮 // 点击按钮后对新视频中的广告进行处理 rollBtn2.addEventListener("click", () => { setTimeout(() => { videoNum = getVideoNum(vDom); // 计算当前视频总数 firstAdIndex = 0; adArr = getAd(queryNum, delClassArr, 0, 1, isTrueEnd ? true : false); delAd(adArr, vDom, true); // 强制删除广告 loadTopVideo(); }, 800); }); } }); // 窗口调整后重新计算视频的行数量 let timer; window.addEventListener("resize", () => { timer && clearTimeout(timer); timer = setTimeout(() => { console.log("窗口改变"); const newW = getW(); if (newW > w) { delAdFn(); // 若新增广告则删除 } w = newW; zoomPage(); setStyle(); }, 400); }); // 加载的新视频去除广告 let timer2, timer3; window.addEventListener("wheel", () => { timer2 && clearTimeout(timer2); timer3 && clearTimeout(timer3); timer2 = setTimeout(() => { delAdFn(timer3); }, 600); timer3 = setTimeout(() => { delAdFn(); }, 1500); }); GM_registerMenuCommand("重置设置", () => { resetSettings(); }); GM_registerMenuCommand("设置", () => { showSetting(settingText); }); // 获取视口宽度 function getW() { let width = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth; const zoom = window.devicePixelRatio; width *= zoom; // console.log("显示器缩放:", zoom); console.log("浏览器宽度:", width); return width; } // 缩放页面 至消除横向滚动条 function zoomPage() { const rootDom = document.documentElement; let rate = rootDom.scrollWidth / getMainW(); if (!rootDom.style.transformOrigin) { rootDom.style.transformOrigin = "top left"; document.body.style.overflow = "hidden auto"; } if (rate > 1) { // 存在横向滚动条 pageZoom *= 1 / rate; rootDom.style.transform = "scale(" + pageZoom + ")"; } else { pageZoom = 1; rootDom.style.transform = "scale(1)"; rate = rootDom.scrollWidth / getMainW(); if (rate > 1) { pageZoom *= 1 / rate; rootDom.style.transform = "scale(" + pageZoom + ")"; } } // 主区域的宽度, 部分时候总宽(导航栏)会大于主区域的宽度 function getMainW() { let navW = nav ? nav.scrollWidth : 0; let bannerW = banner ? banner.scrollWidth : 0; return navW > bannerW ? navW : rootDom.clientWidth; } } /** * 获取所有的 推广 和 广告 的元素的列表 * @param {*} queryNum 需要检索的视频数量 * @param {Array} delClassArr 需要删除的类名列表 * @param {*} vNum 视频总数 * @param {*} startIndex 检索的视频的起始索引位 * @param {Boolean} isAll 是否检索预加载视频以及后面的视频 * @returns {Array} 含各类广告列表的列表 [[...],[...],...] */ function getAd(queryNum, delClassArr, vNum, startIndex = 1, isAll = false) { const arr = []; delClassArr.forEach(() => { arr.push([]); }); const vList = [].slice.call(vDom.children); let len = vNum || vList.length; len = len > vList.length ? vList.length : len; queryNum = queryNum || len; // 0则全检索 queryNum += startIndex; if (queryNum > len) { queryNum = len; } // console.log("queryNum, vNum, startIndex, len\n",queryNum,vNum,startIndex,len); for (let i = startIndex; i < queryNum; i++) { const vItem = vList[i]; // console.log(i, item); if (!isAll && !vItem.querySelector("a")) { break; // 如果是预加载的视频 } for (let j = 0; j < delClassArr.length; j++) { if (isChecked(vItem, delClassArr[j])) { arr[j].push(vItem); break; } } } // console.log("广告列表:", arr); return arr; } // 删除广告 或 放置在最后, 返回减少的数量 function delAd(adArr, dom = vDom, isDel = false) { for (let i = adArr.length - 1; i >= 0; i--) { delInArr(adArr[i]); } function delInArr(arr) { arr.forEach((item) => { if (isClearAd || isDel) { item.remove(); newVideoNum--; videoNum--; } else { if (isTrueEnd) { dom.appendChild(item); // 放在最后 (预加载视频后) } else { dom.insertBefore(item, dom.children[newVideoNum]); // 放在预加载视频前 } } }); } } // 设置浏览器宽度在某个范围时[左闭右开], 每行显示的视频数 function setVideoNum(vRule) { const min = +vRule[0]; const max = +vRule[1]; const num = +vRule[2]; if ((min !== 0 && !min) || !max || !num) { errHandle({ errTxt: `插件设置的视频排列规则设置中 '${vRule.join("")}' 格式书写错误`, key: errKeyArr[1], // 2 }); return; } if (w >= min && w < max) { cssText = ` .container {grid-template-columns: repeat(${num + 2},1fr) !important} .container>div:nth-child(n){margin-top:${marginTop2}px !important} .container>div:nth-child(-n+${ num * 3 + 2 + 1 }){margin-top:${marginTop1}px !important;display:block !important} .container>div:nth-child(-n+${ (num + 1) * 2 - 1 }){margin-top:0px !important}`; isChange = true; showVideoNum = num; } if (!isChange) { cssText = ""; // 默认排列方式 showVideoNum = 3; } } // 调整每行显示个数 function setStyle() { isChange = false; // 每行视频数是否需要变化 videoNumRule.forEach((item) => { setVideoNum(item); // 视口宽度在 1450~2400 px 时则每行显示 4 个视频(前两行) }); if (isChange) { let isCssDom = !!cssDom; // 是否已添加style if (!isCssDom) { cssDom = document.createElement("style"); cssDom.setAttribute("type", "text/css"); } oldCssText !== cssText && (cssDom.innerHTML = cssText); oldCssText = cssText; !isCssDom && vDom.parentElement.insertBefore(cssDom, vDom); } else { // 尺寸缩小时触发 if (!isChange && cssDom) { oldCssText = ""; cssDom.innerHTML = ""; } } } // 获取视频总数 function getVideoNum(dom) { const arr = [].slice.call(dom.children); const len = arr.length; let i; let isGetAdIndex = false; for (i = 1; i < len; i++) { const item = arr[i]; // 获取第一个广告的索引 if (!isTrueEnd && !isGetAdIndex) { const vItem = dom.children[i]; for (let j = 0; j < delClassArr.length; j++) { if (isChecked(vItem, delClassArr[j])) { isGetAdIndex = true; firstAdIndex = i; break; } } } // 如果是预加载视频 if (!item.querySelector("a")) { newVideoNum = i; return i; } } newVideoNum = i; return i; } // 判断是否是查找的目标 function isChecked(vEle, delStr) { let flag = false; const map = classMap; delStr = delClassMap[delStr] || delStr; // 自定义的屏蔽内容 function custom(txt, type, selector) { const dom = vEle.querySelector(selector); if (!dom) { return; } const domTxt = dom.innerText; const txtArr = txt.replace(type, "").split("&&"); if (!txtArr[0]) { return; } let f = false; txtArr.forEach((item) => { f = f || domTxt.includes(item, ""); }); flag = flag || f; } if (delStr.includes("标题=")) { custom(delStr, "标题=", map.标题); } else if (delStr.includes("作者=")) { custom(delStr, "作者=", "." + map.作者); } else if (delStr.includes("分类=")) { custom(delStr, "分类=", "." + map.分类); } else { flag = flag || vEle.classList.contains(delStr); try { flag = flag || vEle.querySelector("." + delStr); } catch (e) { errHandle({ errTxt: `插件设置的屏蔽设置中 '${delStr}' 格式书写错误应以 '标题=' 或 '作者=' 开头`, e, }); } } return flag; } // 根据视频总数是否变化删除广告 function delAdFn(timer = null) { getVideoNum(vDom); if (newVideoNum > videoNum) { console.log("加载新视频"); adArr = getAd( queryNum, delClassArr, newVideoNum, isTrueEnd ? videoNum : firstAdIndex ); delAd(adArr, vDom); videoNum = newVideoNum; timer && clearTimeout(timer); } } // 加载顶部位置的接下来的一组视频 function loadTopVideo() { isLoadOne && document.documentElement.scrollTo(0, 400); isLoadOne && setTimeout(() => { document.documentElement.scrollTo(0, 0); setTimeout(() => { delAdFn(); }, 800); }, 20); } // 获取存储的值, 并解析成对应数据类型 function getValue(key, defa) { let value = GM_getValue(key); if (key === "setting_videoNumRule") { if (value !== undefined && value !== null) { value = getVideoNumRule(value); } else { defa = getVideoNumRule(defa); } } else if (key === "setting_delClassArr") { if (value !== undefined && value !== null) { value = getDelClassArr(value); } else { defa = getDelClassArr(defa); } } return value === undefined || value === null ? defa : value; } // 解析数据字符串为对应数据类型 function getVideoNumRule(value) { value = value.split(/;|;/); return value.map((item) => item.split(/,|,/)); } function getDelClassArr(value) { value = value.replaceAll("\n", "").replaceAll(" ", ""); return value.split(/;|;|,|,/); } // 设置 function showSetting(txt) { const isClearAd = confirm( txt.isClearAd + (GM_getValue("setting_isClearAd") ? "是 (确定)" : "否 (取消)") ); GM_setValue("setting_isClearAd", isClearAd); if (!isClearAd) { const value = GM_getValue("setting_isTrueEnd"); GM_setValue( "setting_isTrueEnd", confirm( txt.isTrueEnd + ((value === undefined ? base_isTrueEnd : value) ? "是 (确定)" : "否 (取消)") ) ); } else { GM_setValue("setting_isTrueEnd", base_isTrueEnd); } GM_setValue( "setting_isLoadOne", confirm( txt.isLoadOne + (GM_getValue("setting_isLoadOne") ? "是 (确定)" : "否 (取消)") ) ); if (confirm("是否进入更多设置")) { const value1 = GM_getValue("setting_videoNumRule"); const value2 = GM_getValue("setting_delClassArr"); const txt2 = prompt( txt.delClassArr, value2 === undefined || value2 === null ? base_delClassArr : value2 ); const txt1 = prompt( txt.videoNumRule, value1 === undefined || value1 === null ? base_videoNumRule : value1 ); GM_setValue("setting_videoNumRule", txt1 || value1); GM_setValue("setting_delClassArr", txt2 || value2); } history.go(0); // 刷新页面 } // 错误处理 function errHandle({ e = null, errTxt = "", logTxt = "", key = "" } = {}) { let errNum = GM_getValue(errKeyInfo.errNum) || 0; if (errNum >= errTipNum) { return; } let disErrNum = GM_getValue(errKeyInfo.disNum + key) || 0; const curTime = Date.now(); const errTime = GM_getValue(errKeyInfo.errTime + key) || curTime; let disS = (curTime - errTime) / 1000; disS = disS === 0 ? 5 : disS; if (disS < 5) { return; } let flag = GM_getValue(errKeyInfo.isTip + key); // 是否能弹窗提示 flag = flag === undefined ? true : flag; e && console.log(e); console.log(logTxt || errTxt); if (disS >= errTipInterval * 60 * 60) { // 每errTipInterval小时允许提醒disErrTipNum次 flag = true; GM_setValue(errKeyInfo.isTip + key, true); GM_setValue(errKeyInfo.disNum + key, 0); } if ( flag && disErrNum <= disErrTipNum && disS < (errTipInterval / 10) * 60 * 60 ) { // 在errTipInterval/10小时内允许disErrTipNum次提示 errNum++; disErrNum++; GM_setValue(errKeyInfo.errNum, errNum); GM_setValue(errKeyInfo.disNum + key, disErrNum); GM_setValue(errKeyInfo.errTime + key, curTime); alert(errTxt); disErrNum === disErrTipNum && GM_setValue(errKeyInfo.isTip + key, false); } } // 重置err相关的数据 function resetErrInfo() { const curTime = Date.now(); const errTime = errKeyArr.reduce((a, b) => { const t = +GM_getValue(errKeyInfo.errTime + b); return t < a ? t : a; }, curTime); if ((curTime - errTime) / 1000 >= errNumReset * 24 * 60 * 60) { GM_setValue(errKeyInfo.errNum, 0); // 重置 errKeyArr.forEach((key) => { GM_setValue(errKeyInfo.disNum + key, 0); // 重置 }); console.log("重置err相关的数据"); } } // 重置设置 function resetSettings() { GM_setValue("setting_isClearAd", base_isClearAd); GM_setValue("setting_isTrueEnd", base_isTrueEnd); GM_setValue("setting_isLoadOne", base_isLoadOne); GM_setValue("setting_videoNumRule", base_videoNumRule); GM_setValue("setting_delClassArr", base_delClassArr); GM_setValue(errKeyInfo.errNum, 0); errKeyArr.forEach((key) => { GM_setValue(errKeyInfo.disNum + key, 0); // 重置 }); history.go(0); // 刷新页面 } })();