// ==UserScript== // @name 哔哩哔哩新版首页排版调整和去广告(bilibili) // @namespace http://tampermonkey.net/ // @version 1.1.2 // @description 对新版B站首页的每行显示的视频数量进行调整, 同时删除所有广告 (大尺寸屏幕每行将显示更多的视频) // @author LingLing // @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* // @compatible chrome // @compatible firefox // @license MIT // @grant none // @downloadURL none // ==/UserScript== (function () { ("use strict"); const isClearAd = true; // 是否删除广告, 若不删除请改为 false, 此时会将所有广告移至视频列表的最后. 默认 true const queryNum = 0; // 处理的视频数量, 对前 queryNum 个视频中的广告进行处理(删除或置后), 0表示对全部视频进行处理. 默认 0 // 视频排列规则, 其他尺寸按照初始方式排列 const videoNumRule = [ [1450, 2400, 4], // 浏览器宽度在 1450~2400 px 时每行显示 4 个视频(前两行), 第三行开始每行显示 6 个视频 [2400, 3000, 5], // 浏览器宽度在 2400~5000 px 时每行显示 5 个视频, ... [3000, 5000, 6], ]; // 屏蔽的类名列表, 可根据需要自行添加其他需要屏蔽的内容, 子元素包含某类名也可屏蔽 const delClassArr = [ "bili-video-card__info--ad", // 广告元素的类名 "bili-video-card__info--creative-ad", // 推广元素的类名, // "floor-single-card", // 特殊视频的类名 [直播,番剧,综艺,课堂...], 如需屏蔽取消注释即可 ]; const marginTop1 = 40; // 第三行视频的上边距 const marginTop2 = 24; // 第四行及以上视频的上边距 const rollBtn_class = "roll-btn"; // 右侧换一换按钮的类名 const rollBtn2_class = "flexible-roll-btn"; // 新版右下角换一换按钮的类名 let w = getW(); // 浏览器视口宽度 const vDom = document.querySelector(".container"); // 视频区域 的容器元素 if (!vDom) { return; } let cssDom; let cssText; let oldCssText; let isChange = false; // 每行视频数是否需要变化 let showVideoNum = 3; // 当前每行显示的视频数 (以第一行为准), 网站默认值为3 let videoNum = 0; // 视频总数 videoNum = getVideoNum(vDom); // 计算当前视频总数 let adArr = getAD(queryNum, delClassArr, videoNum, 0); AdMoveEnd(adArr, vDom); // 将所有广告放置在最后 或 删除 setStyle(); // 调整视频排列 let rollBtn; let btnSvg; let rollBtn2; // 刷新视频 window.addEventListener("click", () => { if (!rollBtn) { adArr = getAD(showVideoNum * 3 + 2, delClassArr, videoNum, 0); AdMoveEnd(adArr, vDom); rollBtn = document.querySelector("button." + rollBtn_class); // 换一换按钮 btnSvg = rollBtn && rollBtn.querySelector("svg"); // 换一换按钮的旋转图标 // 点击按钮后对新视频中的广告进行处理 if (btnSvg) { btnSvg.addEventListener("transitionend", () => { // console.log("视频刷新成功"); adArr = getAD(showVideoNum * 3 + 2 + 3, delClassArr, videoNum, 0); AdMoveEnd(adArr, vDom); }); } else { rollBtn && rollBtn.addEventListener("click", () => { setTimeout(() => { adArr = getAD(showVideoNum * 3 + 2 + 3, delClassArr, videoNum, 0); AdMoveEnd(adArr, vDom); }, 500); }); } } if (!rollBtn2) { adArr = getAD(queryNum, delClassArr, videoNum, 0); AdMoveEnd(adArr, vDom); rollBtn2 = document.querySelector("." + rollBtn2_class); // 新版右下角的换一换按钮 // 点击按钮后对新视频中的广告进行处理 rollBtn2.addEventListener("click", () => { setTimeout(() => { videoNum = getVideoNum(vDom); // 计算当前视频总数 adArr = getAD(queryNum, delClassArr, videoNum, 0); AdMoveEnd(adArr, vDom); }, 800); }); } }); // 窗口调整后重新计算视频的行数量 let timer; window.addEventListener("resize", () => { timer && clearTimeout(timer); timer = setTimeout(() => { console.log("窗口改变"); w = getW(); setStyle(); }, 400); }); // 加载的新视频去除广告 window.addEventListener("wheel", () => { timer && clearTimeout(timer); timer = setTimeout(() => { const curVideoNum = getVideoNum(vDom); if (curVideoNum > videoNum) { console.log("加载新视频"); adArr = getAD(queryNum, delClassArr, curVideoNum, videoNum); AdMoveEnd(adArr, vDom); videoNum = curVideoNum; } }, 600); }); // 获取视口宽度 function getW() { const width = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth; console.log("浏览器宽度:", width); return width; } /** * 获取所有的 推广 和 广告 的元素的列表 * @param {*} queryNum 需要检索的视频数量 * @param {Array} delClassArr 需要删除的类名列表 * @param {*} vNum 视频总数 * @param {*} startIndex 检索的视频的起始索引位 * @returns {Array} 广告列表 */ function getAD(queryNum, delClassArr, vNum, startIndex = 0) { const arr = []; 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 item = vList[i]; if (!item.querySelector("a")) { break; // 如果是预加载的视频 } let isDel = false; // 是否应该删除 for (let j = 0; j < delClassArr.length; j++) { const delItem = delClassArr[j]; isDel = isDel || item.classList.contains(delItem); isDel = isDel || item.querySelector("." + delItem); if (isDel) { break; } } isDel && arr.push(item); } // console.log("广告列表:", arr); return arr; } // 将所有广告放置在最后 或 删除 function AdMoveEnd(adArr, dom = vDom) { adArr.forEach((item) => { if (isClearAd) { item.remove(); videoNum--; } else { dom.appendChild(item); } }); } // 设置浏览器宽度在某个范围时[左闭右开], 每行显示的视频数 function setVideoNum(vRule) { const min = vRule[0]; const max = vRule[1]; const num = vRule[2]; if (w >= min && w < max) { cssText = ` .container {grid-template-columns: repeat(${num + 2},1fr) !important} .container>div:nth-child(n){margin-top:0px !important} .container>div:nth-child(n+${ (num + 1) * 2 }){margin-top:${marginTop1}px !important} .container>div:nth-child(n+${ num * 3 + 2 + 2 }){margin-top:${marginTop2}px !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; for (i = 0; i < len; i++) { const item = arr[i]; if (!item.querySelector("a")) { return i; // 如果是预加载视频 } } return i; } })();