// ==UserScript==
// @name 记录视频刷新历史(bilibili)
// @namespace http://greasyfork.icu/zh-CN/users/1196880-ling2ling4
// @version 1.0.4
// @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: 10, // 历史视频的记录次数, 每次刷新出的视频算一次, 10
isLoadVideo: true, // 是否加载上次关闭页面的视频历史记录, true
base_isLoadVideo: true,
base_maxHistory: 10,
size: 20, // 按钮尺寸
index: 0, // 当前历史视频的组的索引
hoverColor: "#e3e5e7", // 按钮的hover颜色, #f5e3c1
hasCurVideos: false, // 最大索引位的那组视频是否存在, 通过该属性判断在计算时是否需要给最后一组视频占位
// waitTime: 2000, // 刷新页面后记录刷新视频的时间间隔
storageMaxHist: "setting_maxHistory",
storageHistVideos: "setting_histVideos",
storageIsLoadVideo: "setting_isLoadVideo",
};
const txt = {
lBtnTt: "返回上一组视频",
rBtnTt: "切换下一组视频",
maxHistory: `设置历史视频的记录次数, 默认: `,
isLoadVideo: `每次打开B站时是否恢复上次关闭页面的视频历史记录
默认: ${info.base_isLoadVideo ? "是 (确定)" : "否 (取消)"}
当前: `,
// settingSize: `设置按钮尺寸`,
};
const classList = {
vBox: "container", // 视频区域 的容器元素
video: "feed-card", // 换一换的视频 (一般是前十个)
btnBox: "feed-roll-btn", // 按钮的父元素
btn: "roll-btn", // 右侧换一换按钮
};
const doms = {};
let vHistory = []; // 历史视频的记录, [[vdom1, vdom2, ...], [vdom1, ...], ...]
function initValue() {
info.maxHistory = GM_getValue(info.storageMaxHist) || info.base_maxHistory;
info.isLoadVideo = GM_getValue(info.storageIsLoadVideo);
info.isLoadVideo === undefined &&
(info.isLoadVideo = info.base_isLoadVideo);
if (info.isLoadVideo) {
loadVideos();
} else {
GM_setValue(info.storageHistVideos, "[]");
}
}
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--;
}
info.isLoadVideo && saveVideos();
}
// 删除当前这组视频
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);
}
// 将dom转为string
function domToString(dom) {
let tmpDom = document.createElement("div");
tmpDom.appendChild(dom.cloneNode(true));
const str = tmpDom.innerHTML;
tmpDom = null;
return str;
}
// 将string转为dom对象
function strToDom(str) {
let tmpDom = document.createElement("div");
tmpDom.innerHTML = str;
const dom = tmpDom.children[0];
tmpDom = null;
return dom;
}
// 保存历史视频的数据
function saveVideos() {
const arr = [];
vHistory.forEach((vArr) => {
const curArr = [];
vArr.forEach((item) => {
curArr.push(domToString(item));
});
arr.push(curArr);
});
GM_setValue(info.storageHistVideos, JSON.stringify(arr));
}
// 加载历史视频的数据
function loadVideos() {
const value = GM_getValue(info.storageHistVideos);
if (!value) {
return;
}
vHistory = JSON.parse(value) || [];
vHistory.forEach((vArr) => {
vArr.forEach((item, i, arr) => {
arr[i] = strToDom(item);
});
});
info.index = vHistory.length;
info.hasCurVideos = false;
}
function showSetting() {
let newMaxHistory = +prompt(
txt.maxHistory + info.base_maxHistory,
info.maxHistory || info.base_maxHistory
);
if (newMaxHistory < 0) {
newMaxHistory = 5;
} else if (!newMaxHistory) {
newMaxHistory = info.maxHistory;
}
GM_setValue(info.storageMaxHist, newMaxHistory);
info.maxHistory = newMaxHistory;
info.isLoadVideo = confirm(
txt.isLoadVideo + (info.isLoadVideo ? "是 (确定)" : "否 (取消)")
);
GM_setValue(info.storageIsLoadVideo, info.isLoadVideo);
}
function bindEvents() {
doms.btn.addEventListener("click", () => {
if (info.index !== getMaxIndex()) {
vHistory.splice(info.index + 1, vHistory.length - 1 - 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");
});
}
function main() {
// GM_setValue(info.storageHistVideos, "[]");
initValue();
getDoms();
if (!doms.vBox) {
return;
}
// updateHistory();
historyBtns();
GM_registerMenuCommand("设置", () => {
showSetting();
});
}
main();
})();