Greasy Fork

Greasy Fork is available in English.

记录视频刷新历史(bilibili)

记录每次刷新的视频, 可以方便的回溯之前错过的视频

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

// ==UserScript==
// @name          记录视频刷新历史(bilibili)
// @namespace     http://greasyfork.icu/zh-CN/users/1196880-ling2ling4
// @version       1.0.0
// @author        Ling2Ling4
// @description   记录每次刷新的视频, 可以方便的回溯之前错过的视频
// @license MIT
// @icon 
// @match         *://www.bilibili.com/
// @match         *://www.bilibili.com/?*
// @run-at        document-end
// @grant         GM_setValue
// @grant         GM_getValue
// @grant         GM_registerMenuCommand
// @compatible    chrome
// @compatible    edge
// @compatible    firefox
// ==/UserScript==

(function () {
  ("use strict");

  const info = {
    maxHistory: 5, // 历史视频的记录次数, 每次刷新出的视频算一次, 5
    base_maxHistory: 5,
    size: 20, // 按钮尺寸
    index: -1, // 当前历史视频的组的索引
    hoverColor: "#e3e5e7", // 按钮的hover颜色, #f5e3c1
    hasCurVideos: false, // 最大索引位的那组视频是否存在, 通过该属性判断在计算时是否需要给最后一组视频占位
    // waitTime: 2000, // 刷新后记录刷新视频的时间间隔
  };
  const txt = {
    storageMaxHist: "setting_maxHistory",
    lBtnTt: "返回上一组视频",
    rBtnTt: "切换下一组视频",
    maxHistory: `设置历史视频的记录次数, 默认: `,
    // settingSize: `设置按钮尺寸`,
  };
  const classList = {
    vBox: "container", // 视频区域 的容器元素
    video: "feed-card", // 换一换的视频 (一般是前十个)
    btnBox: "feed-roll-btn", // 按钮的父元素
    btn: "roll-btn", // 右侧换一换按钮
  };
  const doms = {};
  const vHistory = []; // 历史视频的记录, [[vdom1, vdom2, ...], [vdom1, ...], ...]
  function initValue() {
    info.maxHistory = GM_getValue(txt.storageMaxHist) || info.maxHistory;
  }
  function getDoms() {
    doms.vBox = document.querySelector("." + classList.vBox);
    // doms.btnBox = document.querySelector("." + classList.btnBox);
    // doms.btn = doms.btnBox && doms.btnBox.querySelector("." + classList.btn);
  }
  function getMaxIndex() {
    return vHistory.length - 1 + (info.hasCurVideos ? 0 : 1);
  }
  // 更新视频历史记录
  function updateHistory() {
    doms.curVideos = [].slice.call(
      doms.vBox.querySelectorAll("." + classList.video)
    );
    if (
      vHistory[vHistory.length - 1] &&
      doms.curVideos[0] === vHistory[vHistory.length - 1][0]
    ) {
      return;
    }
    vHistory.push(doms.curVideos);
    if (info.index === vHistory.length - 1) {
      info.hasCurVideos = true;
    }
    if (vHistory.length > info.maxHistory) {
      vHistory.splice(0, 1); // 过长删除第一项
      info.index--;
    }
  }
  // 删除当前这组视频
  function delCurVideos() {
    const vArr = [].slice.call(
      doms.vBox.querySelectorAll("." + classList.video)
    );
    // const delArr = [];
    vArr.forEach((ele) => {
      ele.remove();
      // delArr.push(doms.vBox.removeChild(ele));
    });
    // return delArr;
  }
  // 切换 上一组/下一组 视频
  function historyChange(f = "left") {
    if (!info.hasCurVideos && info.index === vHistory.length - 1 + 1) {
      updateHistory();
    }
    if (f === "right") {
      // 下一组
      info.index++;
      if (info.index > vHistory.length - 1) {
        info.index = vHistory.length - 1;
        return;
      }
    } else {
      // 上一组
      info.index--;
      if (info.index < 0) {
        info.index = 0;
        return;
      }
    }
    delCurVideos();
    const twoVideo = doms.vBox.children[1];
    // 将历史视频插入最前面
    vHistory[info.index].forEach((ele) => {
      doms.vBox.insertBefore(ele, twoVideo);
    });
  }
  // 增加历史视频切换按钮
  function createBtns() {
    const dom = document.createElement("div");
    dom.innerHTML = `<div id="vHistory-box" style="display:flex;width:100%;line-height:${
      info.size - 3
    }px;margin-top:10px">
  <style>.vHistoryBtn {width:${info.size}px;height:${
      info.size
    }px;text-align:center;border-radius:${info.size / 5}px;cursor:pointer;}
    .vHistoryBtn:hover {background:${info.hoverColor};}</style>
  <div class="left-historyBtn vHistoryBtn" title="${txt.lBtnTt}">
    <svg t="1698507568902" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"
      p-id="918" xmlns:xlink="http://www.w3.org/1999/xlink" width="${
        info.size - 6
      }" height="${info.size - 6}">
      <path d="M796.444444 113.777778c0 17.066667-5.688889 34.133333-17.066666 45.511111L409.6 472.177778c-5.688889 11.377778-11.377778 17.066667-11.377778 34.133333 0 5.688889 5.688889 22.755556 11.377778 28.444445l364.088889 329.955555c22.755556 22.755556 22.755556 56.888889 5.688889 79.644445-22.755556 22.755556-56.888889 22.755556-79.644445 5.688888l-364.088889-329.955555c-34.133333-28.444444-51.2-73.955556-51.2-119.46666699s22.755556-85.333333 56.888889-119.46666601l364.088889-312.888889c22.755556-22.755556 56.888889-17.066667 79.644445 5.688889 5.688889 11.377778 11.377778 28.444444 11.377777 39.822222z" fill="#999999" p-id="919"></path>
    </svg>
  </div>
  <div class="right-historyBtn vHistoryBtn" title="${txt.rBtnTt}">
    <svg t="1698507574371" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"
      p-id="1067" xmlns:xlink="http://www.w3.org/1999/xlink" width="${
        info.size - 6
      }" height="${info.size - 6}">
      <path d="M227.555556 910.222222c0-17.066667 5.688889-34.133333 17.066666-45.511111L614.4 551.822222c5.688889-11.377778 11.377778-17.066667 11.377778-34.133333 0-5.688889-5.688889-22.755556-11.377778-28.444445l-364.088889-329.955555c-22.755556-22.755556-22.755556-56.888889-5.688889-79.644445 22.755556-22.755556 56.888889-22.755556 79.644445-5.688888l364.088889 329.955555c34.133333 28.444444 51.2 73.955556 51.2 119.46666699s-22.755556 85.333333-56.888889 119.46666601l-364.088889 312.888889c-22.755556 22.755556-56.888889 17.066667-79.644445-5.688889-5.688889-11.377778-11.377778-28.444444-11.377777-39.822222z" fill="#999999" p-id="1068"></path>
    </svg>
  </div>
</div>`;
    doms.btnBox.appendChild(dom);
    doms.lBtn = doms.btnBox.querySelector(".left-historyBtn");
    doms.rBtn = doms.btnBox.querySelector(".right-historyBtn");
  }
  // 历史按钮的相关函数
  function historyBtns() {
    const timer = setInterval(() => {
      doms.btnBox = document.querySelector("." + classList.btnBox);
      doms.btn = doms.btnBox && doms.btnBox.querySelector("." + classList.btn);
      if (!doms.btnBox || !doms.btn) {
        return;
      }
      createBtns();
      if (!doms.lBtn || !doms.rBtn) {
        return;
      }
      bindEvents();
      clearInterval(timer);
    }, 1000);
  }
  function showSetting() {
    let newMaxHistory = +prompt(
      txt.maxHistory + info.base_maxHistory,
      GM_getValue(txt.storageMaxHist) || info.base_maxHistory
    );
    if (newMaxHistory < 0) {
      newMaxHistory = 5;
    } else if (!newMaxHistory) {
      newMaxHistory = info.base_maxHistory;
    }
    GM_setValue(txt.storageMaxHist, newMaxHistory);
    info.maxHistory = newMaxHistory;
  }
  function bindEvents() {
    doms.btn.addEventListener("click", () => {
      const maxIndex = getMaxIndex(); // 历史视频数组的最大索引
      if (info.index !== maxIndex) {
        vHistory.splice(info.index, maxIndex - info.index); // 不在最后则删除后面几组视频
        info.index = getMaxIndex();
        info.hasCurVideos = true;
      }
      if (info.index === getMaxIndex()) {
        info.index++;
        info.hasCurVideos = false;
      }
      updateHistory(); // 记录旧的一组视频
    });
    doms.lBtn.addEventListener("click", () => {
      historyChange("left");
    });
    doms.rBtn.addEventListener("click", () => {
      historyChange("right");
    });
    // window.addEventListener("click", (e) => {
    //   if (e.target.classList.contain(classList.btnBox)) {
    //     // 获取按钮
    //   }
    // })
  }
  function main() {
    initValue();
    getDoms();
    if (!doms.vBox) {
      return;
    }
    info.index++;
    // updateHistory();
    historyBtns();
    GM_registerMenuCommand("设置", () => {
      showSetting();
    });
  }
  main();
})();