Greasy Fork

Greasy Fork is available in English.

哔哩哔哩新版首页排版调整和去广告(bilibili)

对新版B站首页的每行显示的视频数量进行调整, 同时删除所有广告 (大尺寸屏幕每行将显示更多的视频)

当前为 2023-10-26 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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
// ==/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); // 屏蔽的类名列表, 子元素包含某类名也可屏蔽
  const oldZoom =
    GM_getValue("setting_oldZoom") || document.documentElement.style.zoom;

  // console.log(isClearAd, isTrueEnd, isLoadOne, isZoom, 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 bannerW = banner ? banner.scrollWidth : 0;
      let navW = nav ? nav.scrollWidth : 0;
      return navW > bannerW ? bannerW : 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); // 刷新页面
  }
})();