Greasy Fork

Greasy Fork is available in English.

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

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

目前为 2023-10-16 提交的版本。查看 最新版本

// ==UserScript==
// @name         哔哩哔哩新版首页排版调整和去广告(bilibili)
// @namespace    http://tampermonkey.net/
// @version      1.1.3
// @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(); // 浏览器视口宽度
  let 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);
  delAd(adArr, vDom); // 将所有广告放置在最后 或 删除
  setTimeout(() => {
    delAdFn();
  }, 1000);
  setStyle(); // 调整视频排列

  let rollBtn;
  let btnSvg;
  let rollBtn2;
  // 刷新视频
  window.addEventListener("click", () => {
    if (!rollBtn) {
      adArr = getAd(showVideoNum * 3 + 2, delClassArr, videoNum, 0);
      delAd(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);
          delAd(adArr, vDom);
        });
      } else {
        rollBtn &&
          rollBtn.addEventListener("click", () => {
            setTimeout(() => {
              adArr = getAd(showVideoNum * 3 + 2 + 3, delClassArr, videoNum, 0);
              delAd(adArr, vDom);
            }, 500);
          });
      }
    }
    if (!rollBtn2) {
      adArr = getAd(queryNum, delClassArr, videoNum, 0);
      delAd(adArr, vDom);
      rollBtn2 = document.querySelector("." + rollBtn2_class); // 新版右下角的换一换按钮
      // 点击按钮后对新视频中的广告进行处理
      rollBtn2.addEventListener("click", () => {
        setTimeout(() => {
          videoNum = getVideoNum(vDom); // 计算当前视频总数
          adArr = getAd(queryNum, delClassArr, videoNum, 0);
          delAd(adArr, vDom);
        }, 800);
      });
    }
  });

  // 窗口调整后重新计算视频的行数量
  let timer;
  window.addEventListener("resize", () => {
    timer && clearTimeout(timer);
    timer = setTimeout(() => {
      console.log("窗口改变");
      w = getW();
      setStyle();
    }, 400);
  });

  // 加载的新视频去除广告
  let timer2, timer3;
  window.addEventListener("wheel", () => {
    timer2 && clearTimeout(timer2);
    timer3 && clearTimeout(timer3);
    timer2 = setTimeout(() => {
      console.log("1. videoNum:", videoNum);
      delAdFn(timer3);
    }, 600);
    timer3 = setTimeout(() => {
      delAdFn();
    }, 1500);
  });

  // 获取视口宽度
  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];
      // console.log(i, item);
      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 delAd(adArr, dom = vDom) {
    let dis = 0;
    adArr.forEach((item) => {
      if (isClearAd) {
        item.remove();
        videoNum--;
        dis++
      } else {
        dom.appendChild(item);
      }
    });
    return dis;
  }

  // 设置浏览器宽度在某个范围时[左闭右开], 每行显示的视频数
  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;
  }

  // 根据情况删除广告
  function delAdFn(timer = null) {
    const newVideoNum = getVideoNum(vDom);
    if (newVideoNum > videoNum) {
      console.log("加载新视频");
      adArr = getAd(queryNum, delClassArr, newVideoNum, videoNum);
      const dis = delAd(adArr, vDom);
      videoNum = newVideoNum - dis;
      timer && clearTimeout(timer);
    }
  }
})();