// ==UserScript== // @name 记住阅读进度 // @namespace http://tampermonkey.net/ // @version 4.4.0 // @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 // @downloadURL https://update.greasyfork.icu/scripts/452142/%E8%AE%B0%E4%BD%8F%E9%98%85%E8%AF%BB%E8%BF%9B%E5%BA%A6.user.js // @updateURL https://update.greasyfork.icu/scripts/452142/%E8%AE%B0%E4%BD%8F%E9%98%85%E8%AF%BB%E8%BF%9B%E5%BA%A6.meta.js // ==/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 = `

X

URL规则:

高度阈值:

一┗|`O′|┛ 说明 ~
`; // 设置样式 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(); // 初始化进度显示视图 initView(); // 初始化记录器 initRecorder(); } //【函数库】 //有动画地滚动, 这里不用,因为要直接恢复,而不浪费滚动的时间 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+""+ 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()+"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 () { const viewHtml = `
helloworld
`; document.body.insertAdjacentHTML('beforeend', viewHtml); let container, progressText, progressView; requestAnimationFrame(() => { container = document.querySelector('#progress_container'); progressText = document.querySelector('#progress_container .progress_text'); progressView = document.querySelector('#progress_container .progress_view'); }); // 显示容器 function showSchedule() { if(!rpcCache.isShowSchedule || container == null) return; // 是否要显示/隐藏 container.style.display= rpcCache.isShowSchedule?"block":"none"; // 将 当前进度/ 总进度 放在显示容器中 const curr = parseInt(getCurrentTop()), sum = parseInt(getDocumentHeight() - window.innerHeight); progressText.innerHTML = `${curr} / ${sum}`; progressView.style.width = container.clientWidth * (curr/sum) + "px"; // 防抖关闭显示视图容器 -- 在上面的闭包中的监听了滚动 } // 进度显示防抖关闭函数 function showScheduleDebounce(fn, wait) { let timer = null; return function() { showSchedule(); // 关闭定时器的 if(timer != null ) clearTimeout(timer); timer= setTimeout (fn,wait); } } // 隐藏容器-处理函数 function hideSchedule() { // 关闭容器 container.style.display="none"; } window.addEventListener('scroll', showScheduleDebounce(hideSchedule,460)); } // 【初始化记录器】 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) } })();