哔哩哔哩视频笔记时间进度记录器 Bilibili Video Note Recorder

方便PC端观看B站系列视频时,时间进度记录(= ̄ω ̄=)

// ==UserScript==
// @name         哔哩哔哩视频笔记时间进度记录器 Bilibili Video Note Recorder
// @namespace    http://tampermonkey.net/
// @version      2.0
// @description  方便PC端观看B站系列视频时,时间进度记录(= ̄ω ̄=)
// @author       乃木流架
// @match        https://www.bilibili.com/*
// @match        https://member.bilibili.com/platform/upload-manager/*
// @icon         
// @grant        GM_addStyle
// @run-at       document-start
// @license      GPL-3.0 License
// ==/UserScript==

(() => {
  ("use strict");
  
  if (/member.bilibili.com/.test(window.location.href)) {
      GM_addStyle(`
      .ep-section-edit-video-list-item-title .title-text-content {
        display: contents !important;
      }
    `)
    throw new Error('🛑Bilibili Recorder脚本终止执行🛑')
  }
  


  const name = "Bilibili Recorder";
  const logPrefix = [
    "%c" + name,
    `background:#52c41a;border-radius: 0.5em;color: white;padding: 2px 0.5em`,
  ];
  function log(...args) {
    console.log(...logPrefix, ...args);
  }

  const iconUrl = "https://i.jpg.dog/037000721ae61d81533753de8af04d4d.png";
  const icon =
    "";

  var record = document.createElement("div");

  /**
   * @desc 属性改变监听,属性被set时出发watch的方法,类似vue的watch
   * @author Jason
   * @study https://www.jianshu.com/p/00502d10ea95
   * @data 2018-04-27
   * @constructor
   * @param {object} opts - 构造参数. @default {data:{},watch:{}};
   * @argument {object} data - 要绑定的属性
   * @argument {object} watch - 要监听的属性的回调
   * watch @callback (newVal,oldVal) - 新值与旧值
   */
  class watcher {
    constructor(opts) {
      this.$data = this.getBaseType(opts.data) === "Object" ? opts.data : {};
      this.$watch = this.getBaseType(opts.watch) === "Object" ? opts.watch : {};
      for (let key in opts.data) {
        this.setData(key);
      }
    }

    getBaseType(target) {
      const typeStr = Object.prototype.toString.apply(target);

      return typeStr.slice(8, -1);
    }

    setData(_key) {
      Object.defineProperty(this, _key, {
        get: function () {
          return this.$data[_key];
        },
        set: function (val) {
          const oldVal = this.$data[_key];
          if (oldVal === val) return val;
          this.$data[_key] = val;
          this.$watch[_key] &&
            typeof this.$watch[_key] === "function" &&
            this.$watch[_key].call(this, val, oldVal);
          return val;
        },
      });
    }
  }

  function add0(m) {
    return m < 10 ? "0" + m : m;
  }

  function format(timestamp) {
    //timestamp是整数,否则要parseInt转换
    var time = new Date(timestamp);
    var y = time.getFullYear();
    var m = time.getMonth() + 1;
    var d = time.getDate();
    var h = time.getHours();
    var mm = time.getMinutes();
    var s = time.getSeconds();
    return (
      y + "年" + m + "月" + d + "日" + add0(h) + ":" + add0(mm) + ":" + add0(s)
    );
  }

  function iss() {
    var is = document.getElementsByClassName("note-operation")[0];
    var operation = document.getElementsByClassName("note-header drag-el")[0];
    var back_note = document.getElementsByClassName("back-note-list")[0];
    var note_operation = document.getElementsByClassName(
      "list-note-operation"
    )[0];
    if (back_note) {
      back_note.addEventListener("click", function () {
        operation.insertBefore(record, operation.children[1].nextSibling);
      });
    }
    if (note_operation) {
      note_operation.addEventListener("click", function () {
        operation.insertBefore(record, operation.children[1].nextSibling);
      });
    }

    // operation.insertBefore(record, operation.children[1].nextSibling);
    if (is == undefined) {
      // operation.insertBefore(record, operation.children[1].nextSibling);
      record.style.display = "none";
    } else {
      // operation.insertBefore(record, operation.children[1].nextSibling);
      record.style.display = "flex";
    }
  }

  function init() {
    log("init button");

    var operation = document.getElementsByClassName("note-header drag-el")[0];
    var styleMap = {
      position: "relative",
      marginRight: "16px",
      width: "30px",
      height: "30px",
      display: "flex",
      alignItems: "center",
      justifyContent: "center",
      borderRadius: "6px",
      cursor: "pointer",
      fontSize: "12px",
      lineHeight: "20px",
      border: "1px solid #00aeec",
      color: "#00aeec",
      transition: "0.2s",
      background: "#ffffff",
    };

    record.setAttribute("class", "record");
    record.innerHTML =
      "<img src = " + icon + ' width="20" height="20" align="middle"></img>';

    for (let i in styleMap) {
      record.style[i] = styleMap[i];
    }

    // var noteop = document.getElementsByClassName('note-operation')[0];
    operation.insertBefore(record, operation.children[1].nextSibling);
    // operation.childNodes[1].appendChild(record);

    record.addEventListener("click", function () {
      log("start record");

      var editor = document.getElementsByClassName("ql-editor")[0];
      var now = +new Date();
      var number = 1;

      var lastHTML = editor.innerHTML;
      var lastContent = editor.textContent;
      log("lastHTML: " + lastHTML);
      log("lastContent: " + lastContent);

      var searchParams = new URLSearchParams(window.location.search);
      var end = document.getElementsByClassName("page-num").length;
      var list = document.getElementsByClassName("list-box")[0];

      number = searchParams.get("p");
      var numberP = "";

      if (list) {
        var on = list.getElementsByClassName("on")[0];
        numberP = on.getElementsByClassName("page-num")[0].innerHTML;
      } else {
        numberP = "P1";
      }

      log("numberP: " + numberP);
      log("list: " + list);

      if (number != end && lastContent == "") {
        editor.innerHTML = "<p>" + format(now) + "</p><p>" + numberP + "</p>";
      } else if (number != end && lastContent != "") {
        editor.innerHTML =
          lastHTML +
          "<p><br></p><p>" +
          format(now) +
          "</p><p>" +
          numberP +
          "</p>";
      } else if (number == end && lastContent == "") {
        editor.innerHTML = "<p>" + format(now) + "</p><p>完结撒花!!!</p>";
      } else if (number == end && lastContent != "") {
        editor.innerHTML =
          lastHTML +
          "<p><br></p><p>" +
          format(now) +
          "</p><p>完结撒花!!!</p>";
      }
    });
  }

  function start() {
    log("bilibiliRecorder userscript is running...");

    var noteButton = document.querySelector(".video-note-inner");
    var noteList = document.querySelector(".note-list");
    var noteOper = document.querySelector(".list-note-operation");
    var noteClose = document.querySelector(".close-note");

    log("noteButton: " + noteButton);

    noteButton.addEventListener("click", function () {
      log("noteButton pressed");

      let wm = new watcher({
        data: {
          ne: false,
        },
        watch: {
          ne(newVal) {
            log("noteEditor changed to: " + newVal);
            if (!newVal) {
              log("button no init");
              noteOper.addEventListener("click", function () {
                log("noteOper pressed");
                init();
              });
            } else {
              log("button init");
              init();
            }
          },
        },
      });

      var isss = setInterval(function () {
        iss();

        var noteEditor = document.querySelector(".note-editor") != null;
        wm.ne = noteEditor;
      }, 500);

      if (noteClose) {
        noteClose.onclick = () => {
          log("noteClose pressed");

          clearInterval(isss);

          log("Interval cleared");
        };
      }
    });
  }

  window.onload = function () {
    log("window.onload");
    setTimeout(() => {
      start();
    }, 5000);
  };

  // Your code here...
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址