Greasy Fork

Greasy Fork is available in English.

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         哔哩哔哩新版首页排版调整和去广告(bilibili)
// @namespace    http://tampermonkey.net/
// @version      1.1.1
// @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
// ==/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 curVideoNum = 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(curVideoNum * 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(curVideoNum * 3 + 2 + 3, delClassArr, videoNum, 0);
          AdMoveEnd(adArr, vDom);
        });
      } else {
        rollBtn &&
          rollBtn.addEventListener("click", () => {
            setTimeout(() => {
              adArr = getAD(curVideoNum * 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(() => {
          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;
      curVideoNum = num;
    }
    if (!isChange) {
      cssText = ""; // 默认排列方式
      curVideoNum = 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;
  }
})();