记住阅读进度

记住页面阅读进度,即使对于单页面,也能很好的工作!

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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
// ==/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();
        // 初始化进度显示视图
        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+"<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 () {
        const viewHtml = `
            <div id='progress_container'>
                <span class='progress_text'>helloworld</span>
                <div class='progress_view'></div>
            </div>
            <style>
               #progress_container {
                 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;
               }
               #progress_container .progress_text {
                 z-index: 0;
               }
               #progress_container .progress_view {
                 height: 100%;
                 background: rgba(26, 173, 25,0.5);
                 position: absolute;
                 left: 0px;
                 top: 0;
                 z-index: 1;
               }
            </style>
        `;
        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)
    }
})();