// ==UserScript==
// @name 记住阅读进度
// @namespace http://tampermonkey.net/
// @version 4.3.2
// @description 记住页面阅读进度,即使对于单页面,也能很好的工作!
// @match *://*/*
// @exclude http://127.0.0.1*
// @exclude http://localhost*
// @exclude http://192.168.*
// @author zhuangjie
// @icon 
// @license MIT
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_deleteValue
// @grant GM_registerMenuCommand
// ==/UserScript==
(function() {
'use strict';
// Your code here...
// 编写脚本学习教程:http://www.ttlsa.com/docs/greasemonkey/#pattern.addcss
// 解决广工商学校选课网无法显示左边栏:原因是脚本中加了 window.onload 有关
// 上一个重要版本:https://cdn.jsdelivr.net/gh/18476305640/typora@master/images/2022/10/21/recoverHistorySchedule.js
// https://cdn.jsdelivr.net/gh/18476305640/typora@master/images/2022/10/26/f.txt
// 初始化事件容器-页面活跃与否事件(节省性能而使用)
(function () {
window.onblur = function () {
window.events.onblur.trigger();
}
window.onfocus = function () {
window.events.onfocus.trigger();
}
window.events = {
onblur: {
monitors: [],
add(fun) {
this.monitors.push(fun)
},
trigger() {
for(let i = 0; i < this.monitors.length; i++) {
this.monitors[i]();
}
}
},
onfocus: {
monitors: [],
add(fun) {
this.monitors.push(fun)
},
trigger() {
for(let i = 0; i < this.monitors.length; i++) {
this.monitors[i]();
}
}
}
}
})();
// 【url改变监听器】
function onUrlChange(fun) {
let initUrl = window.location.href.split("#")[0];
function urlChange() {
let currentUrl = window.location.href.split("#")[0];
if(initUrl != currentUrl) {
// 新的=>旧的
initUrl = currentUrl;
fun();
initUrl = currentUrl;
}
}
let si = setInterval(urlChange,460)
window.onblur = function() {
clearInterval(si);
}
window.onfocus = function() {
si = setInterval(urlChange,460)
}
}
// 全局url事件
window.urlChangeListener = {
events:[],
add(event) {
this.events.push(event)
},
trigger (){
for(let event of this.events) {
event()
}
}
}
onUrlChange(function(){window.urlChangeListener.trigger()})
// 防抖函数
function debounce(fn, wait) {
var timeout = null;
return function() {
if(timeout != null ) clearTimeout(timeout);
timeout= setTimeout (fn, wait);
}
}
// 节流函数
const throttle = (fn, Intervals, ...args) => {
let timeNo;
return (...params) => {
if(timeNo) return;
timeNo = setTimeout(() => {
fn(...args,...params)
clearTimeout(timeNo);
timeNo = null;
}, Intervals);
}
}
// 多iframe 屏障
function iframeParclose(name) {
if(name == null) {
name = Date.now();
}
return {
set(timeout = 2000) {
const value = cache.get(name);
if(value && parseInt(value) > Date.now()) return false;
cache.set(name,Date.now() + timeout);
return true;
},
remove() {
cache.remove(name);
}
}
}
async function iframeParcloseWrapper(name,fun) {
const parclose = iframeParclose(name);
const timeout = 1200;
if(!parclose.set(timeout)) return;
try {
await fun();
}finally {
setTimeout(()=>parclose.remove(),timeout)
}
}
// 数据缓存器
let cache = {
get(key) {
return GM_getValue(key);
},
set(key,value) {
GM_setValue(key,value);
},
remove(key) {
GM_deleteValue(key)
}
}
let rpcCache; rpcCache = getRPC();
function setRPC(config) {
if(config == null) return;
cache.set("ReadingProgressConfig",rpcCache = config)
}
function getRPC() {
let result = cache.get("ReadingProgressConfig")
if(result == null) {
// 默认数据
result = {
// URL规则(满足应用)
ruleList: [
"**",
"https://www.cnblogs.com/**/p/**",
"https://blog.csdn.net/**"
],
heightThreshold: 2000, // 高度阈值
isShowSchedule: true // 是否显示进度控制
}
// 保存初始配置
setRPC(result);
}
return result;
}
GM_registerMenuCommand("开/关进度显示",function() {
iframeParcloseWrapper("ProgressDisplayStatusChange",function() {
return new Promise((resolve,reject)=>{
const rpc = getRPC();
rpc.isShowSchedule = !rpc.isShowSchedule;
setRPC(rpc)
alert(`┌(`▽′)╭ 进度显示已${rpc.isShowSchedule?"开启":"关闭"}`)
resolve();
})
})
});
GM_registerMenuCommand("配置规则",function() {
showConfigView();
});
// 显示配置规则视图
function showConfigView() {
// 后面可能会修改配置,刷新配置缓存,防止其它页面的修改导致当前页面的数据不一致
rpcCache = getRPC();
// 显示视图
var configViewContainer = document.createElement("div");
configViewContainer.style=`
width:300px; background:pink;
position: fixed;right: 0px; top: 0px;
z-index:10000;
padding: 20px;
border-radius: 14px;
`
configViewContainer.innerHTML = `
<p id="rpc_close">X</p>
<p class="rpc_config_title">URL规则:</p>
<textarea id="rpc_urlTextarea" ></textarea>
<p class="rpc_config_title">高度阈值:</p>
<input id="rpc_heightInput" />
<div id="rpc_controller">
<button id="rpc_save" >保存</button>
<span id="rpc_tis" title="规则与高度阈值都满足脚本才会生效!">一┗|`O′|┛ 说明 ~ </span>
</div>
`;
// 设置样式
document.body.appendChild(configViewContainer)
document.getElementById("rpc_close").style="color: red;font-weight: bold;font-size: 14px;cursor: pointer; position: absolute;right: 10px; top: 10px;margin: 0;";
Array.from(document.getElementsByClassName("rpc_config_title")).forEach(item => {
item.style = "font-size:14px;margin:7px 0 5px;color: black;";
});
document.getElementById("rpc_urlTextarea").style="width:100%;height:150px;border: 4px solid rgb(245, 245, 245);box-sizing: border-box;";
document.getElementById("rpc_heightInput").style="width:100%;border: 2px solid rgb(245, 245, 245);box-sizing: border-box;";
document.getElementById("rpc_controller").style="width:100%; margin-top:20px;";
document.getElementById("rpc_save").style="width:30%; border:none;border-radius:3px;padding:3px;";
document.getElementById("rpc_tis").style="color:#f5f5f5;display:block;text-align: center;float: right;cursor: pointer;";
// 回显
document.getElementById("rpc_urlTextarea").value = rpcCache.ruleList.join("\n")
document.getElementById("rpc_heightInput").value = rpcCache.heightThreshold
// 保存
document.getElementById("rpc_save").onclick=function() {
// 保存到对象
rpcCache.ruleList = document.getElementById("rpc_urlTextarea").value.split("\n")
rpcCache.heightThreshold = document.getElementById("rpc_heightInput").value
// 持久化
setRPC(rpcCache)
// 清除视图
configViewContainer.remove();
alert("保存配置成功!")
}
// 关闭
document.getElementById("rpc_close").onclick = ()=> configViewContainer.remove();
}
// 检查量下满足开启脚本条件
function checkIsSatisfyEnableCondition() {
let heightThreshold = rpcCache.heightThreshold;
let ruleList = rpcCache.ruleList;
// 判断高度是否满足
let isSatisfyHeight = getDocumentHeight() >= heightThreshold;
// 判断是否满足规则
let isSatisfyURL = seeSatisfyURL();
return isSatisfyHeight && isSatisfyURL;
}
// 看下是否满足开启脚本条件-根据URL规则
function seeSatisfyURL () {
let currentUrl = window.location.href;
let ruleList = rpcCache.ruleList;
// 当规则为空时,直接返回false
if(ruleList == null || ruleList.length == 0) return false;
for(let rule of ruleList) {
rule = rule.trim();
if(rule.indexOf("**") < 0 ) {
if(currentUrl== rule) return true;
continue;
}
// 满足泛匹配
let isOk = (function(){
let ruleChilds = rule.split("**")
if(ruleChilds == null || ruleChilds.length == 0) return false;
for(let block of ruleChilds) {
if(currentUrl.indexOf(block) < 0) {
// 表示当前测试的这个规则不满足
return false;
}
}
return true;
})();
if(isOk) return true;
}
// 当规则不为空时,且不通过上面的匹配时,返回false
return false;
}
// 【何时开始脚本】
// 获取滚动历史高度
let item_content = localStorage.getItem(getCurrentUrl())
let history_high = item_content == null?0:parseFloat(item_content);
// 是否已经初始化
let isInit = false;
// 初始化程序
do {
if(history_high <= getDocumentHeight() || document.readyState == "complete") {
setTimeout(()=>{init()},50)
break;
}
}while(history_high <= getDocumentHeight() || document.readyState == "complete");
// 【主程序】
function init() {
// 判断当前页面是否满足开启阅读进度
if(!checkIsSatisfyEnableCondition() && !isInit) {
// 当页面不满足初始化时,添加再次初始化器,当页面url改变时,会再次尝试
window.urlChangeListener.add(function() {
setTimeout(()=>{init() },1500)
})
return;
};
// 标记为已初始化
isInit = true;
// 初始化还原器
recoverMonitor();
// 初始化记录
initRecorder();
// 初始化视图(显示高度)
initView();
}
//【函数库】
//有动画地滚动, 这里不用,因为要直接恢复,而不浪费滚动的时间
let st = null; //保证多次执行 ScrollTo 函数不会相互影响
function ScrollTo(scroll, top) {
if(st != null ) {
//关闭上一次未执行完成的滚动
clearInterval(st);
}
//每次移动的跨度
let span = 5;
// 最长滚动时间
let timeout = false;
let timeout_time = 5000;
let timer = setTimeout(()=>{timeout=true},timeout_time);
st = setInterval(function () {
let currentTop = getCurrentTop();
//当在跨度内时,直接到达, 如果不在指定时间内滚动到,那将直接到达
if ((currentTop >= top - span && currentTop <= top + span) || timeout ) {
clearTimeout(timer);
timeout = false;
setTop(top);
// $(scroll).scrollTop(top);
//让st为null,让关闭定时器
let tmp_st = st;
st = null;
//关闭定时器(下一次不会再执行,但本次还会执行下去),再return;
clearInterval(tmp_st);
// console.log("滚动完成",top+"<is>"+ getCurrentTop() )
return;
}
//如果不在跨度内时,根据当前的位置与目的位置进行上下移动指定跨度
if (currentTop < top) {
setTop(currentTop + span)
} else {
setTop(currentTop - span)
}
span++
}, 20)
}
// 获取url,url经过了处理
function getCurrentUrl() {
return window.location.href.split("#")[0]
}
// 获取存储标记,用于存储滚动“责任人”
function getCurrentPageWhoRoll() {
return getCurrentUrl()+"<and>WhoRoll";
}
// 获取当前滚动的高度
function getCurrentTop() {
return document.documentElement.scrollTop || document.body.scrollTop;
}
// 获取文档高度
function getDocumentHeight() {
// 获取最大高度
return (document.documentElement.scrollHeight > document.body.scrollHeight?document.documentElement.scrollHeight:document.body.scrollHeight)
}
// 到达指定高度
function setTop(h,isCheck = true) {
let whoRoll = localStorage.getItem(getCurrentPageWhoRoll())
if(!isCheck) {
document.documentElement.scrollTop = h;
document.body.scrollTop = h;
return;
}
if(whoRoll == "document") {
if(getDocumentHeight() >= h ) {
document.documentElement.scrollTop = h;
}
}else {
if(getDocumentHeight() >= h ) {
document.body.scrollTop = h;
}
}
}
// 判断是否在滚动
function onNotScrolling(callback,ScrollingCallback) {
// 如果不在滚动调用回调
let h1 = parseInt(getCurrentTop());
setTimeout(function() {
let h2 = parseInt(getCurrentTop());
if(h1 == h2) {
callback();
}else {
ScrollingCallback();
}
},50)
}
// 检查是否到达指定位置
function checkIsArriveHeight(time,expectHeight = 0,callback,flag = true) {
setTimeout(function() {
let top_down_scope = 200/2;
let currentHiehgt = getCurrentTop();
if(!(expectHeight >= currentHiehgt-top_down_scope && expectHeight <= currentHiehgt+top_down_scope)) {
// 不管当前是否在滚动,都进行失败回调
callback();
}else {
// 只有到达了且不在滚动了才算成功,否则调用失败回调
onNotScrolling(function() {
},callback)
}
},time)
return null;
}
//【初始化视图显示】
function initView () {
let scheduleBox=document.createElement("div");
scheduleBox.innerText = "helloworld";
scheduleBox.id = "schedule_box";
scheduleBox.style.cssText=`
display:none;
height: 35px;
line-height: 35px;
font-size: 15px;
position: fixed;
right: 20px;
top: 20px;
z-index: 10000;
padding: 0px 10px;
background: #333333;
color: #fff;
overflow: hidden;
`;
// 进度显示防抖关闭函数
function showScheduleDebounce(fn, wait) {
let timer = null;
return function() {
window.showSchedule();
// 关闭定时器的
if(timer != null ) clearTimeout(timer);
timer= setTimeout (fn,wait);
}
}
// 隐藏容器-处理函数
function hideSchedule() {
// 关闭容器
scheduleBox.style.display="none";
}
window.addEventListener('scroll',showScheduleDebounce(hideSchedule,460));
document.body.appendChild(scheduleBox);
// 显示容器
window.showSchedule = function(content) {
// 是否要显示/隐藏
scheduleBox.style.display= rpcCache.isShowSchedule?"block":"none";
if(!rpcCache.isShowSchedule) return;
// 将 当前进度/ 总进度 放在显示容器中
scheduleBox.innerHTML = content || parseInt(getCurrentTop())+" / "+parseInt(getDocumentHeight() - window.innerHeight);
let scheduleChild=document.createElement("div");
scheduleChild.style.cssText=`
height: 100%;
background: rgba(26, 173, 25,0.5);
position: absolute;
top: 0px;
left: 0px;
`;
scheduleBox.appendChild(scheduleChild);
scheduleChild.style.width = scheduleBox.clientWidth*((getCurrentTop())/parseInt(getDocumentHeight() - window.innerHeight))+"px";
// 防抖关闭显示视图容器 -- 在上面的闭包中的监听了滚动
}
}
// 【初始化记录器】
function initRecorder() {
// 位置保存
// 处理函数
function handle() {
// console.log(document.documentElement.scrollTop , document.body.scrollTop )
let current_top = getCurrentTop();
let current_url = getCurrentUrl()
if(document.documentElement.scrollTop > document.body.scrollTop ) {
localStorage.setItem(getCurrentPageWhoRoll(),"document")
}else {
localStorage.setItem(getCurrentPageWhoRoll(),"body")
}
if(current_top <= 10) return;
// console.log("[记住历史进度]拿着小本本记着:",current_url,current_top+"px");
// console.log(">>> 滚动责任:",localStorage.getItem(getCurrentPageWhoRoll()));
localStorage.setItem(current_url,""+current_top)
}
// 滚动事件
window.addEventListener('scroll',debounce(handle, 460));
}
// 【还原监听器】
function recoverMonitor() {
// 位置还原
function recover() {
let item_content = localStorage.getItem(getCurrentUrl())
// 有记录
if (item_content == null) return;
// 获取历史高度
let history_high = parseFloat(item_content);
// 现在文档的高度
let current_height = getDocumentHeight();
// 如果没有历史高度,且高度不大于10就不还原
if(history_high != null && history_high >= 10 ) {
// 直接还原到历史位置
setTop(history_high);
// 检查是否恢复成功
//let fun = null;
//checkIsArriveHeight(500,history_high, function() {
// 如果失败,重试
//setTop(history_high,false);
//});
}
}
recover(); // 进入页面时还原
window.urlChangeListener.add(recover)
}
})();