// ==UserScript==
// @name 哔哩哔哩新版首页排版调整和去广告(bilibili)
// @namespace http://tampermonkey.net/
// @version 1.3.7
// @author Ling2Ling4
// @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
// ==/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_isAutoLayout = true; // 是否根据缩放自动布局 (是否根据像素宽度布局). 默认 true
const base_isLoadOne = false; // 是否视频全加载. 默认 false
const base_videoNumRule =
"0,1300,2; 1300,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 isAutoLayout = getValue("setting_isAutoLayout", base_isAutoLayout);
const isLoadOne = getValue("setting_isLoadOne", base_isLoadOne);
let videoNumRule = getValue("setting_videoNumRule", base_videoNumRule);
let 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: `是否删除广告, 若不删除则会将所有广告移至视频列表的最后
默认: ${base_isClearAd ? "是 (确定)" : "否 (取消)"}
当前: `,
isTrueEnd: `是否将广告移至预加载视频的后面, 关闭后广告将放置在预加载视频的前面 一般视频的后面. 开启的效率更高
默认: ${base_isTrueEnd ? "是 (确定)" : "否 (取消)"}
当前: `,
isAutoLayout: `是否根据缩放自动调整视频布局
默认: ${base_isAutoLayout ? "是 (确定)" : "否 (取消)"}
当前: `,
isLoadOne: `是否在进入网站时加载视口区域的全部视频, 开启时视频将会全部加载, 但会闪一下
默认: ${base_isLoadOne ? "是 (确定)" : "否 (取消)"}
当前: `,
delClassArr: `屏蔽设置, 可根据需要自行修改, 可自定义, 每项用 ; 分隔
----默认:
${base_delClassArr}
----可选: ('特殊'包含了直播 番剧 课堂...)
广告、推广、特殊、直播、番剧、综艺、课堂、漫画、国创、电影、纪录片、电视剧
----自定义:
1. 标题=xxx, 可屏蔽标题含xxx的视频, xxx部分支持&&运算符, 如: 标题=A&&B, 表示屏蔽标题同时含有A B内容的视频
2. 作者=xxx, 可屏蔽作者名和发布日期中含xxx的视频`,
videoNumRule: `视频排列规则, 每条规则用 ; 分隔. 其他尺寸按照初始方式排列
示例: 1450,2400,4 表示浏览器宽度在1450~2400像素时每行显示4个视频(前两行)
默认:
${base_videoNumRule}`,
};
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; // 第四行及以上视频的上边距
const zoom = window.devicePixelRatio; // 获取浏览器缩放 (包括显示器缩放的影响)
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 && 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("基础设置", () => {
basicSetting(settingText);
});
GM_registerMenuCommand("屏蔽设置", () => {
delSetting(settingText);
});
GM_registerMenuCommand("排列规则", () => {
layoutSetting(settingText);
});
GM_registerMenuCommand("重置设置", () => {
resetSettings();
});
// 获取视口宽度
function getW() {
let width =
window.innerWidth ||
document.documentElement.clientWidth ||
document.body.clientWidth;
// console.log("显示器缩放:", zoom);
console.log("浏览器实际宽度:", width * window.devicePixelRatio);
console.log("浏览器像素宽度:", width * zoom);
!isAutoLayout && (width *= window.devicePixelRatio);
return width;
}
// 缩放页面 至消除横向滚动条
function zoomPage() {
if (document.getBoxObjectFor) {
return; // 火狐不支持zoom, 而用scale会有bug
}
const rootDom = document.documentElement;
let rate = rootDom.scrollWidth / getMainW();
!document.body.style.overflow &&
(document.body.style.overflow = "hidden auto");
if (rate > 1) {
// 存在横向滚动条
pageZoom *= 1 / rate;
rootDom.style.zoom = pageZoom;
} else {
pageZoom = 1;
rootDom.style.zoom = 1;
rate = rootDom.scrollWidth / getMainW();
if (rate > 1) {
pageZoom *= 1 / rate;
rootDom.style.zoom = 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] / (isAutoLayout ? zoom : 1);
const max = +vRule[1] / (isAutoLayout ? zoom : 1);
const num = +vRule[2];
// console.log(min, max, num, ">", w);
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;
// console.log("每行 " + num + " 个视频");
}
if (!isChange) {
cssText = ""; // 默认排列方式
showVideoNum = 3;
}
}
// 调整每行显示个数
function setStyle() {
isChange = false; // 每行视频数是否需要变化
videoNumRule.forEach((item) => {
setVideoNum(item);
});
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 getDelClassArr(value) {
value = value.replaceAll("\n", "").replaceAll(" ", "");
return value.split(/;|;|,|,/);
}
function getVideoNumRule(value) {
value = value.split(/;|;/);
return value.map((item) => item.split(/,|,/));
}
// 基础设置
function basicSetting(txt) {
GM_setValue(
"setting_isAutoLayout",
confirm(
txt.isAutoLayout +
(GM_getValue("setting_isAutoLayout") ? "是 (确定)" : "否 (取消)")
)
);
GM_setValue(
"setting_isLoadOne",
confirm(
txt.isLoadOne +
(GM_getValue("setting_isLoadOne") ? "是 (确定)" : "否 (取消)")
)
);
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);
}
history.go(0); // 刷新页面
}
// 屏蔽设置
function delSetting(txt) {
const value = GM_getValue("setting_delClassArr");
const newValue = prompt(
txt.delClassArr,
value === undefined || value === null ? base_delClassArr : value
);
GM_setValue("setting_delClassArr", newValue || value);
delClassArr = getDelClassArr(newValue || value);
let adArr = getAd(queryNum, delClassArr, newVideoNum, 1);
delAd(adArr, vDom);
}
// 视频排列规则的设置
function layoutSetting(txt) {
const value = GM_getValue("setting_videoNumRule");
const newValue = prompt(
txt.videoNumRule,
value === undefined || value === null ? base_videoNumRule : value
);
GM_setValue("setting_videoNumRule", newValue || value);
videoNumRule = getVideoNumRule(newValue || value);
zoomPage(); // 缩放页面
setStyle(); // 调整视频排列
}
// 重置设置
function resetSettings() {
GM_setValue("setting_isClearAd", base_isClearAd);
GM_setValue("setting_isTrueEnd", base_isTrueEnd);
GM_setValue("setting_isAutoLayout", base_isAutoLayout);
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); // 刷新页面
}
// 错误处理
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相关的数据");
}
}
})();