学习强国梨酱小帮手

页面内登录/搜索+视频/音频/电子书一键批量下载+拦截Log请求+电子书去水印

// ==UserScript==
// @name         学习强国梨酱小帮手
// @namespace    https://qinlili.bid/
// @version      1.1.8
// @description  页面内登录/搜索+视频/音频/电子书一键批量下载+拦截Log请求+电子书去水印
// @author       琴梨梨
// @match        *://www.xuexi.cn/*
// @match        *://boot-source.xuexi.cn/newmoocdown?*
// @match        *://boot-source.xuexi.cn/audiodown?*
// @match        *://preview-pdf.xuexi.cn/*
// @match        *://article.xuexi.cn/*
// @match        https://login.xuexi.cn/login/xuexiWeb?*
// @match        https://static.xuexi.cn/search/*
// @icon         https://www.xuexi.cn/favicon.ico
// @homepage     https://github.com/qinlili23333/XXQG-DL
// @supportURL   https://github.com/qinlili23333/XXQG-DL
// @grant        none
// @run-at       document-end
// @require      https://lib.baomitu.com/jspdf/2.5.1/jspdf.umd.min.js
// @license      Anti996License
// ==/UserScript==


(async function () {
    'use strict';
    //既然连喜欢的人都没能力留住,那还是把更多时间投入到写代码吧--记于与悦悦子分手的7天之后(2022.1.11)
    //这些内容给本项目开发提供了帮助,感谢
    //https://stackoverflow.com/a/60644673
    //https://stackoverflow.com/a/55165133
    //也感谢每一位相信琴梨梨的用户

    //真的有人会看琴梨梨写的注释吗?在看的话MUA你一下~

    //是否开启跨源服务器
    //开启跨源服务器可以下载部分本来会出错的电子书
    //请访问此地址获取跨源服务器https://github.com/Rob--W/cors-anywhere
    //Clone到本地后运行npm install,然后运行node server.js,即可运行在默认地址和端口上
    var corsServer = "http://localhost:8080/";
    if (document.location.host == "preview-pdf.xuexi.cn" && (document.location.search.indexOf("boot-video.xuexi.cn") > 1) && (window.self === window.top) && confirm("该地址可能需要跨源服务器下载。启用跨源服务器吗?请在确认跨源服务器已启动之后点击确定。\n不知道跨源服务器是什么的话打开脚本源码看注释")) {
        var valueProp = Object.getOwnPropertyDescriptor(Image.prototype, 'src');
        Object.defineProperty(Image.prototype, 'src', {
            set: function(newimgValue){
                if (!newimgValue.startsWith("data:")) {
                    newimgValue = corsServer + newimgValue;
                }
                this.crossOrigin = "anonymous"
                valueProp.set.call(this, newimgValue);
            }
        });
    }
    //iframe页面处理
    if (!(window.self === window.top)) {
        var transparentStyle = "background:none transparent !important;";
        document.documentElement.style = transparentStyle;
        document.body.style = transparentStyle;
        if (document.location.href.indexOf("login.xuexi.cn/login/xuexiWeb?") > 1) {
            document.getElementsByClassName("login_content")[0].style.background = "none"
        };
        if (document.location.href.indexOf("static.xuexi.cn/search/online/index.html") > 1) {
            document.getElementById("root").style.background = transparentStyle;
            if ((window.self.innerWidth > window.self.innerHeight) && window.self.innerWidth > 1000) {
                document.getElementsByClassName("search-content")[0].style = "padding-left:20px;padding-right:20px;"
            }
        };
    }
    //干掉日志
    (open=> {
        XMLHttpRequest.prototype.open = function (method, url, async, user, pass) {
            if (!(async === false)) {
                async = true;
            };
            if (url.startsWith("https://iflow-api.xuexi.cn/logflow/api/v1/pclog") || url.startsWith("https://arms-retcode.aliyuncs.com/r.png")) {
                console.log("Rejected Log XHR! " + url + " -Qinlili")
                url = "data:text,null"
            }
            open.call(this, method, url, async, user, pass);
        };
    })(XMLHttpRequest.prototype.open);
    var originFetch = fetch;
    window.fetch = (url, options) => {
        if (url.startsWith("https://iflow-api.xuexi.cn/logflow/api/v1/pclog") || url.startsWith("https://arms-retcode.aliyuncs.com/r.png")) {
            console.log("Rejected Log Fetch! " + url + " -Qinlili")
            url = "data:text,null"
        }
        return originFetch(url, options);
    }
    //干掉PDF水印
    if (document.location.host == "preview-pdf.xuexi.cn") {
        CanvasRenderingContext2D.prototype.fillText = () => { }
    }
    //共享库
    const SakiProgress = {
        isLoaded: false,
        progres: false,
        pgDiv: false,
        textSpan: false,
        first: false,
        alertMode: false,
        init: function (color) {
            if (!this.isLoaded) {
                this.isLoaded = true;
                console.info("SakiProgress Initializing!\nVersion:1.0.3\nQinlili Tech:Github@qinlili23333");
                this.pgDiv = document.createElement("div");
                this.pgDiv.id = "pgdiv";
                this.pgDiv.style = "z-index:9999;position:fixed;background-color:white;min-height:32px;width:auto;height:32px;left:0px;right:0px;top:0px;box-shadow:0px 2px 2px 1px rgba(0, 0, 0, 0.5);transition:opacity 0.5s;display:none;";
                this.pgDiv.style.opacity = 0;
                this.first = document.body.firstElementChild;
                document.body.insertBefore(this.pgDiv, this.first);
                this.first.style.transition = "margin-top 0.5s"
                this.progress = document.createElement("div");
                this.progress.id = "dlprogress"
                this.progress.style = "position: absolute;top: 0;bottom: 0;left: 0;background-color: #F17C67;z-index: -1;width:0%;transition: width 0.25s ease-in-out,opacity 0.25s,background-color 1s;"
                if (color) {
                    this.setColor(color);
                }
                this.pgDiv.appendChild(this.progress);
                this.textSpan = document.createElement("span");
                this.textSpan.style = "padding-left:4px;font-size:24px;";
                this.textSpan.style.display = "inline-block"
                this.pgDiv.appendChild(this.textSpan);
                var css = ".barBtn:hover{ background-color: #cccccc }.barBtn:active{ background-color: #999999 }";
                var style = document.createElement('style');
                if (style.styleSheet) {
                    style.styleSheet.cssText = css;
                } else {
                    style.appendChild(document.createTextNode(css));
                }
                document.getElementsByTagName('head')[0].appendChild(style);
                console.info("SakiProgress Initialized!");
            } else {
                console.error("Multi Instance Error-SakiProgress Already Loaded!");
            }
        },
        destroy: function () {
            if (this.pgDiv) {
                document.body.removeChild(this.pgDiv);
                this.isLoaded = false;
                this.progres = false;
                this.pgDiv = false;
                this.textSpan = false;
                this.first = false;
                console.info("SakiProgress Destroyed!You Can Reload Later!");
            }
        },
        setPercent: function (percent) {
            if (this.progress) {
                this.progress.style.width = percent + "%";
            } else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        },
        clearProgress: function () {
            if (this.progress) {
                this.progress.style.opacity = 0;
                setTimeout(function () { SakiProgress.progress.style.width = "0%"; }, 500);
                setTimeout(function () { SakiProgress.progress.style.opacity = 1; }, 750);
            } else {
                console.error("Not Initialized Error-Please Call `init` First!")
            }
        },
        hideDiv: function () {
            if (this.pgDiv) {
                if (this.alertMode) {
                    setTimeout(function () {
                        SakiProgress.pgDiv.style.opacity = 0;
                        SakiProgress.first.style.marginTop = "";
                        setTimeout(function () {
                            SakiProgress.pgDiv.style.display = "none";
                        }, 500);
                    }, 3000);
                } else {
                    this.pgDiv.style.opacity = 0;
                    this.first.style.marginTop = "";
                    setTimeout(function () {
                        SakiProgress.pgDiv.style.display = "none";
                    }, 500);
                }
            }
            else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        },
        showDiv: function () {
            if (this.pgDiv) {
                this.pgDiv.style.display = "";
                setTimeout(function () { SakiProgress.pgDiv.style.opacity = 1; }, 10);
                this.first.style.marginTop = (this.pgDiv.clientHeight + 8) + "px";
            }
            else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        },
        setText: function (text) {
            if (this.textSpan) {
                if (this.alertMode) {
                    setTimeout(function () {
                        if (!SakiProgress.alertMode) {
                            SakiProgress.textSpan.innerText = text;
                        }
                    }, 3000);
                } else {
                    this.textSpan.innerText = text;
                }
            }
            else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        },
        setTextAlert: function (text) {
            if (this.textSpan) {
                this.textSpan.innerText = text;
                this.alertMode = true;
                setTimeout(function () { this.alertMode = false; }, 3000);
            }
            else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        },
        setColor: function (color) {
            if (this.progress) {
                this.progress.style.backgroundColor = color;
            }
            else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        },
        addBtn: function (img) {
            if (this.pgDiv) {
                var btn = document.createElement("img");
                btn.style = "display: inline-block;right:0px;float:right;height:32px;width:32px;transition:background-color 0.2s;"
                btn.className = "barBtn"
                btn.src = img;
                this.pgDiv.appendChild(btn);
                return btn;
            }
            else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        },
        removeBtn: function (btn) {
            if (this.pgDiv) {
                if (btn) {
                    this.pgDiv.removeChild(btn);
                }
            }
            else {
                console.error("Not Initialized Error-Please Call `init` First!");
            }
        }
    }
    const XHRDL = {
        isLoaded: false,
        dlList: [],
        listBtn: false,
        listDiv: false,
        listBar: false,
        clsBtn: false,
        init: function () {
            if (!this.isLoaded) {
                console.info("WebXHRDL Initializing!\nVersion:Preview0.1.0\nQinlili Tech:Github@qinlili23333")
                try {
                    SakiProgress.init();
                } catch {
                    console.error("Initialize Failed!Is SakiProgress Loaded?")
                    return false;
                }
                this.isLoaded = true;
                //this.listBtn = SakiProgress.addBtn("");
                //this.listBtn.onclick = XHRDL.showList;
                SakiProgress.showDiv();
                SakiProgress.setText("初始化下载器...");
                SakiProgress.setPercent(20);
                this.listDiv = document.createElement("div");
                this.listDiv.style = "z-index:9999;position:fixed;background-color:white;width:auto;margin-top:32px;height:100%;left:0px;right:0px;top:0px;transition:opacity 0.5s;display:none;";
                this.listDiv.style.opacity = 0;
                this.listBar = document.createElement("div");
                this.listBar.style = "z-index:10000;position:fixed;background-color:white;min-height:32px;margin-top:32px;width:auto;height:32px;left:0px;right:0px;top:0px;box-shadow:0px 2px 2px 1px rgba(0, 0, 0, 0.5);";
                this.listDiv.appendChild(this.listBar);
                document.body.appendChild(this.listDiv);
                var btn = document.createElement("img");
                btn.style = "display: inline-block;right:0px;float:right;height:32px;width:32px;transition:background-color 0.2s;"
                btn.className = "barBtn"
                btn.src = "";
                this.listBar.appendChild(btn);
                btn.onclick = function () {
                    XHRDL.hideList();
                }
                this.clsBtn = btn;
                SakiProgress.setPercent(100);
                SakiProgress.setText("下载器已加载!");
                setTimeout(function () { SakiProgress.clearProgress(); SakiProgress.hideDiv(); }, 1000);
                console.info("WebXHRDL Initialized!");
            } else {
                console.error("Multi Instance Error-WebXHRDL Already Loaded!")
            }
        },
        destroy: function (saki) {
            if (this.isLoaded) {
                if (saki) {
                    SakiProgress.destroy();
                }
                this.isLoaded = false;
                this.dlList = [];
                this.listBtn = false;
                this.listDiv = false;
                this.listBar = false;
                this.clsBtn = false;
                console.info("WebXHRDL Destroyed!You Can Reload Later!");
            }
        },
        showList: function () {
            if (XHRDL.isLoaded) {
                XHRDL.listDiv.style.display = "";
                setTimeout(function () { XHRDL.listDiv.style.opacity = 1; }, 10);
            } else {
                console.error("Not Initialized Error-Please Call `init` First!")
            }
        },
        hideList: function () {
            if (XHRDL.isLoaded) {
                XHRDL.listDiv.style.opacity = 0;
                setTimeout(function () { XHRDL.listDiv.style.display = "none"; }, 500);
            } else {
                console.error("Not Initialized Error-Please Call `init` First!")
            }
        },
        saveTaskList: function () {
            if (XHRDL.isLoaded) {
                var storage = window.localStorage;
                storage.setItem("XHRDL_List", JSON.stringify(this.dlList));
            } else {
                console.error("Not Initialized Error-Please Call `init` First!")
            }
        },
        loadTaskList: function () {
            if (XHRDL.isLoaded) {
                var storage = window.localStorage;
                this.dlList = JSON.parse(storage.getItem("XHRDL_List"));
            } else {
                console.error("Not Initialized Error-Please Call `init` First!")
            }
        },
        newTask: function (url, name) {
            if (this.isLoaded) {
                var list = this.dlList;
                list[list.length] = {
                    taskUrl: url,
                    fileName: name
                }
                SakiProgress.showDiv();
                SakiProgress.setText("已添加新任务:" + name);
                if (!this.DLEngine.isWorking) {
                    this.DLEngine.start();
                }
            } else {
                console.error("Not Initialized Error-Please Call `init` First!")
            }
        },
        DLEngine: {
            isWorking: false,
            start: function () {
                if (!this.isWorking) {
                    console.info("Start WebXHRDL Engine...\nChecking Tasks...");
                    this.isWorking = true;
                    SakiProgress.showDiv();
                    this.dlFirstFile();
                } else {
                    console.error("WebXHRDL Engine Already Started!");
                }
            },
            stop: function () {
                this.isWorking = false;
                SakiProgress.hideDiv();
                SakiProgress.setText("");
                if (XHRDL.dlList[0]) {
                    console.info("All Tasks Done!WebXHRDL Engine Stopped!");
                } else {
                    console.info("WebXHRDL Engine Stopped!Tasks Paused!");
                }
            },
            dlFirstFile: function () {
                var taskInfo = XHRDL.dlList[0];
                SakiProgress.showDiv();
                SakiProgress.setPercent(0);
                SakiProgress.setText("正在下载" + taskInfo.fileName);
                var xhr = new XMLHttpRequest();
                xhr.responseType = "blob";
                xhr.onprogress = event => {
                    if (event.loaded && event.total) {
                        var percent = String(Number(event.loaded) / Number(event.total) * 100).substring(0, 4);
                        SakiProgress.setText(taskInfo.fileName + "已下载" + percent + "%");
                        SakiProgress.setPercent(percent)
                    }
                };
                xhr.onload = event => {
                    if (xhr.readyState === 4) {
                        if (xhr.status === 200) {
                            var bloburl = URL.createObjectURL(xhr.response);
                            SakiProgress.setText("正在写出" + taskInfo.fileName);
                            var a = document.createElement('a');
                            var filename = taskInfo.fileName;
                            a.href = bloburl;
                            a.download = filename;
                            a.click();
                            window.URL.revokeObjectURL(bloburl);
                            SakiProgress.clearProgress();
                            XHRDL.dlList.splice(0, 1);
                            XHRDL.DLEngine.checkNext();
                        } else {
                            //TODO:支持更多特殊状态处理
                            SakiProgress.setTextAlert(taskInfo.fileName + "暂不支持下载,跳过");
                            XHRDL.dlList.splice(0, 1);
                            XHRDL.DLEngine.checkNext();
                        }
                    }
                }
                xhr.onerror = function (e) {
                    //TODO:支持处理不同类别出错
                    if (!taskInfo.errorRetry) {
                        SakiProgress.setTextAlert(taskInfo.fileName + "下载失败,置入列尾等待重试");
                        taskInfo.errorRetry = true;
                        var list = XHRDL.dlList;
                        list[list.length] = taskInfo;
                    } else {
                        SakiProgress.setTextAlert(taskInfo.fileName + "下载又失败了,放弃");
                    }
                    XHRDL.dlList.splice(0, 1);
                    XHRDL.DLEngine.checkNext();
                }
                xhr.open('GET', taskInfo.taskUrl, true)
                xhr.send()
            },
            checkNext: function () {
                if (XHRDL.dlList[0]) {
                    this.dlFirstFile();
                } else {
                    this.stop();
                }
            }
        }
    }


    const sleep = delay => new Promise(resolve => setTimeout(resolve, delay));
    const openFrame=url=>{
        var searchFrame = document.createElement("iframe");
        searchFrame.frameBorder = 0;
        searchFrame.style = "padding:100%;z-index:9999;position:fixed;backdrop-filter: blur(10px) brightness(100%);background-color: rgba(255, 255, 255, .6);width:100%;margin-top:0px;height:100%;left:0px;right:0px;top:0px;";
        document.body.appendChild(searchFrame);
        searchFrame.src = url;
        var clsBtn = document.createElement("img");
        clsBtn.style = "z-index:10000;position:fixed;display: inline-block;right:0px;top:0px;float:right;height:32px;width:32px;transition:background-color 0.2s;"
        clsBtn.className = "barBtn"
        clsBtn.src = "";
        document.body.appendChild(clsBtn);
        searchFrame.addEventListener("load", async () => { await sleep(150); searchFrame.style.padding = "0px"; });
        clsBtn.onclick = () => {
            document.body.removeChild(searchFrame);
            document.body.removeChild(clsBtn);
        }
    };
    //主站检测
    if (document.location.host == "www.xuexi.cn" || document.location.host == "preview-pdf.xuexi.cn") {
        console.log("JS Loaded,Sleep 3 Sec-Qinlili");
        await sleep(3000)
        if (window.self === window.top) {
            //初始化下载工具条
            XHRDL.init();
            var dlPannel = document.createElement("div");
            var downloadBtn = document.createElement("button");
            downloadBtn.innerText = "下载本页内容";
            downloadBtn.style.display = "inline-block";
            dlPannel.appendChild(downloadBtn);
            var dlText = document.createElement("p");
            dlText.style.display = "inline-block";
            dlText.innerText = "等待检测页面类型";
            dlPannel.appendChild(dlText);
            var first = document.body.firstChild;
            document.body.insertBefore(dlPannel, first);
            //接管搜索
            if (document.getElementsByClassName("icon search-icon")[0]) {
                var scBtn = document.getElementsByClassName("icon search-icon")[0];
                var scPrt = scBtn.parentElement;
                scPrt.removeChild(scBtn);
                scBtn = document.createElement("a");
                scBtn.className = "icon search-icon";
                scPrt.appendChild(scBtn);
                scBtn.addEventListener("click", async e => {
                    var searchFrame = document.createElement("iframe");
                    searchFrame.frameBorder = 0;
                    searchFrame.style = "padding:100%;z-index:9999;position:fixed;backdrop-filter: blur(10px) brightness(100%);background-color: rgba(255, 255, 255, .6);width:100%;margin-top:0px;height:100%;left:0px;right:0px;top:0px;";
                    document.body.appendChild(searchFrame);
                    searchFrame.src = "https://static.xuexi.cn/search/online/index.html";
                    var clsBtn = document.createElement("img");
                    clsBtn.style = "z-index:10000;position:fixed;display: inline-block;right:0px;top:0px;float:right;height:32px;width:32px;transition:background-color 0.2s;"
                    clsBtn.className = "barBtn"
                    clsBtn.src = "";
                    document.body.appendChild(clsBtn);
                    searchFrame.addEventListener("load", async () => { await sleep(150); searchFrame.style.padding = "0px"; });
                    clsBtn.onclick = () => {
                        document.body.removeChild(searchFrame);
                        document.body.removeChild(clsBtn);
                        window.removeEventListener("message", msg, false);
                    }
                    function msg(e) {
                        //抄的官方JS改出来的,我对于这种非要传值到上层窗口的做法完全无法理解,但既然能跑,管他呢
                        console.log('e:', e)
                        console.log('e.data:', e.data)
                        try {
                            var params = JSON.parse(e.data);
                            if (params.type) {
                                console.log('params.type:', params.type);
                                console.log('params.data:', params.data);
                                switch (params.type) {
                                    case 'search':
                                        var useQuestionMark = false;
                                        var targetUrl = 'https://static.xuexi.cn/search/online/index.html'
                                        for (var key in params.data) {
                                            var value = params.data[key];
                                            var op = '&'
                                            if (!useQuestionMark) {
                                                op = '?';
                                                useQuestionMark = true;
                                            }
                                            targetUrl += op + key + '=' + value;
                                        }
                                        searchFrame.style.padding = "100%";
                                        searchFrame.src = targetUrl;
                                        break;
                                    default:
                                        break;
                                }
                            }
                        } catch (error) {
                            console.log(error)
                        }
                    }
                    window.addEventListener("message", msg, false);
                })
            }
            //接管登录
            if (document.getElementsByClassName("icon login-icon")[0]) {
                var dlBtn = document.getElementsByClassName("icon login-icon")[0];
                var dlPrt = dlBtn.parentElement;
                dlPrt.removeChild(dlBtn);
                dlBtn = document.createElement("a");
                dlBtn.className = "icon login-icon";
                dlPrt.appendChild(dlBtn);
                dlBtn.addEventListener("click", async e => {
                    e.preventDefault();
                    SakiProgress.showDiv();
                    SakiProgress.setPercent(5);
                    SakiProgress.setText("正在准备登录...")
                    console.log("Login Hooked!-Qinlili");
                    var closeBtn = SakiProgress.addBtn("")
                    var loginFrame = document.createElement("iframe");
                    loginFrame.frameBorder = 0;
                    loginFrame.scrolling = "no";
                    loginFrame.style = "padding:100%;z-index:9999;position:fixed;backdrop-filter: blur(10px) brightness(100%);background-color: rgba(255, 255, 255, .6);width:100%;margin-top:32px;height:100%;left:0px;right:0px;top:0px;";
                    document.body.appendChild(loginFrame);
                    loginFrame.addEventListener("load", async function () { await sleep(250); loginFrame.style.padding = "0px"; });
                    var refreshBtn = SakiProgress.addBtn("")
                    refreshBtn.onclick = () => {
                        scanLogin();
                    }
                    closeBtn.onclick = () => {
                        document.body.removeChild(loginFrame);
                        SakiProgress.removeBtn(closeBtn);
                        SakiProgress.removeBtn(refreshBtn);
                        closeBtn = false;
                        SakiProgress.clearProgress();
                        SakiProgress.hideDiv();
                    }
                    async function scanLogin() {
                        await sleep(100);
                        SakiProgress.setPercent(10);
                        SakiProgress.setText("正在获取登录口令...");
                        let token;
                        let tokenText;
                        try {
                            token = await fetch("https://pc-api.xuexi.cn/open/api/sns/sign")
                            tokenText = JSON.parse(await token.text())
                        } catch (e) {
                            SakiProgress.setPercent(100);
                            SakiProgress.setText("出错了:网络连接中断! " + e.message);
                            return;
                        }
                        if (tokenText.code = "200") {
                            token = tokenText.data.sign;
                            SakiProgress.setPercent(40);
                            SakiProgress.setText("口令获取成功,加载登录页面...");
                            loginFrame.onload = () => {
                                SakiProgress.setPercent(65);
                                SakiProgress.setText("等待扫码...");
                            }
                            loginFrame.style.padding = "100%";
                            loginFrame.src = "https://login.xuexi.cn/login/xuexiWeb?appid=dingoankubyrfkttorhpou&goto=https%3A%2F%2Foa.xuexi.cn&type=1&state=" + token + "&check_login=https%3A%2F%2Fpc-api.xuexi.cn"
                            window.addEventListener("message", event => {
                                event.preventDefault();
                                console.log(event);
                                if (event.data.success == true) {
                                    SakiProgress.setPercent(100);
                                    SakiProgress.setText("登录成功,正在刷新页面...");
                                    document.location.reload();
                                } else {
                                    SakiProgress.setPercent(100);
                                    SakiProgress.setText("出错了:扫码登录失败,错误码为" + event.data.errorCode);
                                }
                            }, false);
                        } else {
                            SakiProgress.setPercent(100);
                            SakiProgress.setText("出错了:获取口令失败!");
                        }
                    }
                    scanLogin();
                })
            }
            //检测爬取页面类型
            console.log("Detecting Page " + document.location.pathname + "-Qinlili");
            var detected = false;
            //旧慕课列表
            if ((document.body.innerText.indexOf("课程介绍") >= 1) || (document.body.innerText.indexOf("课程详情") >= 1)) {
                console.log("Old Mooc List Detected " + document.location.pathname + "-Qinlili");
                detected = true;
                dlText.innerText = "页面类型:旧慕课列表,支持全部批量下载,请开启网站自动下载权限";
                downloadBtn.onclick = () => {
                    OldMoocListDL();
                }
            }
            //页面有播放器
            if (window.Aliplayer) {
                //旧慕课、电视剧播放单页
                if (document.getElementsByClassName("radio-inline")[0]) {
                    console.log("Old Video Player Detected " + document.location.pathname + "-Qinlili");
                    detected = true;
                    dlText.innerText = "页面类型:旧视频播放单页,支持全部批量下载,请开启网站自动下载权限";
                    downloadBtn.onclick = () => {
                        OldMoocVideoDL();
                    }
                }
                //新慕课、影视总页
                if (document.getElementsByClassName("video-article-content")[0] || document.getElementsByClassName("videoSet-article-wrap")[0]) {
                    console.log("New Video Player Detected " + document.location.pathname + "-Qinlili");
                    detected = true;
                    dlText.innerText = "页面类型:新视频总,支持全部批量下载最高清晰度,需要打开新标签页下载";
                    downloadBtn.onclick = () => {
                        NewMoocPageDL();
                    }
                }
            }
            //音频专题
            if (document.getElementsByClassName("album-play-btn")[0]) {
                console.log("Audio Detected " + document.location.pathname + "-Qinlili");
                detected = true;
                dlText.innerText = "页面类型:音频,支持全部批量下载,需要打开新标签页下载";
                downloadBtn.onclick = () => {
                    AudioDL();
                }
            }
            //页面上就一个音频
            //解锁音频播放器下载按钮
            if (document.getElementsByTagName("audio").length) {
                detected = true;
                dlText.innerText = "页面类型:单个音频,已经解锁播放器下载能力,点击播放器右侧菜单下载";
                downloadBtn.style.display = "none";
                for (var la = 0; document.getElementsByTagName("audio")[la]; la++) {
                    document.getElementsByTagName("audio")[la].removeAttribute("controlslist");
                }
            }
            //电子书下载
            if (document.location.host == "preview-pdf.xuexi.cn") {
                detected = true;
                dlText.innerText = "页面类型:电子书,支持打包下载";
                downloadBtn.onclick = () => {
                    PDFDL();
                }
            }
            if (!detected) {
                console.log("Unsupported Page " + document.location.pathname + "-Qinlili");
                dlText.innerText = "本页面不支持下载";
                downloadBtn.innerText = "暂不支持";
            }

            //下载器部分
            //旧慕课列表
            function OldMoocListDL() {
                //读取全部视频列表
                var videoList = globalCache[Object.keys(globalCache)[0]];
                console.log("Found " + videoList.length + " Videos-Qinlili")
                for (var i = 0; videoList[i]; i++) {
                    console.log("Try Analysis " + i + " Video-Qinlili")
                    getInfoAndDL(pagetoinfourl(videoList[i].static_page_url));
                }

            }
            //旧慕课播放单页
            function OldMoocVideoDL() {
                console.log("Analysis Page Info-Qinlili")
                getInfoAndDL(pagetoinfourl(document.location.href))
            }
            function NewMoocPageDL() {
                console.log("Open DL Page-Qinlili");
                var searchParams = new URLSearchParams(document.location.search);
                var dlurl = "https://boot-source.xuexi.cn/newmoocdown?id=" + searchParams.get("id");
                openFrame(dlurl);
            }
            function AudioDL() {
                console.log("Open DL Page-Qinlili");
                var searchParams = new URLSearchParams(document.location.search);
                var dlurl = "https://boot-source.xuexi.cn/audiodown?id=" + searchParams.get("id");
                openFrame(dlurl);
            }
            async function PDFDL() {
                //webp压缩用处和顶碗人一样大,所以换成灰度压缩
                let enableGreyCompress = confirm("是否启用灰度压缩?\n适合保存以黑白文本内容的书籍或用于Kindle等墨水屏阅读,可大幅削减文件体积,需额外消耗压缩时间。\n根据琴梨梨自己的测试可削减大约44%大小,可用于解决Chrome无法爬512M以上书的问题。")
                let lowQuality = confirm("是否启用低质量模式?\n轻微降低画质换取显著的文件缩小");
                SakiProgress.showDiv();
                SakiProgress.setText("正在加载依赖...");
                await sleep(100)
                console.log("Preparing jsPDF Library...-Qinlili");
                var jsPDF = jspdf.jsPDF;
                try {
                    console.log(jsPDF)
                    console.log("jsPDF Ready!")
                } catch {
                    console.error("jsPDF Not Ready!")
                    alert("jsPDF加载失败,请检查网络并尝试重新安装脚本!")
                }
                SakiProgress.setText("正在调整尺寸...");
                SakiProgress.setPercent(2);
                await sleep(100)
                if(confirm("是否开启高清模式?请观察原书清晰度,若原书清晰度较高建议开启。\n对超过300页的书开启可能导致无法生成文件。")){
                    //按钮循环点击得稍微延迟一点否则可能卡死
                    //放大到最大保障清晰度
                    for (; document.getElementsByClassName("ctrl-icon")[0].className.animVal.indexOf("disabled") < 0;) {
                        document.getElementsByClassName("ctrl-icon")[0].parentElement.click()
                        await sleep(50)
                    }
                }
                SakiProgress.setText("正在回到第一页...");
                SakiProgress.setPercent(4);
                await sleep(100)
                //回到第一页
                for (; document.getElementsByClassName("ctrl-icon")[2].className.animVal.indexOf("disabled") < 0;) {
                    document.getElementsByClassName("ctrl-icon")[2].parentElement.click()
                    await sleep(50)
                }
                //创建文件
                SakiProgress.setText("正在创建文件...");
                SakiProgress.setPercent(6);
                await sleep(100)
                var samplePage = document.getElementsByTagName("canvas")[0]
                var ori;
                let wP = samplePage.width;
                let hP = samplePage.height;
                if (wP > hP) { ori = "l" } else { ori = "p" }
                var PDFfile = new jsPDF({
                    orientation: ori,
                    unit: 'px',
                    format: [wP, hP],
                    putOnlyUsedFonts: true,
                });
                console.log("Preparing PDF File...-Qinlili");
                console.log(PDFfile);
                //监听函数
                SakiProgress.setText("正在设置监听函数...");
                SakiProgress.setPercent(8);
                await sleep(100)
                var onPageChange = function () { };
                var val = document.getElementsByTagName("input")[0];
                function waitPageChange() {
                    return new Promise(resolve => {
                        onPageChange = () => {
                            resolve();
                        }
                    });
                }
                //加载完成后页码显示才会变化,监听页码显示来等待加载
                Object.defineProperty(val, 'value', {
                    set: newValue => {
                        var valueProp = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value');
                        valueProp.set.call(val, newValue);
                        onPageChange();
                    }
                });
                //循环保存
                var page = 1;
                var totalPage = parseInt(document.getElementsByClassName("total")[0].innerText.substr(1));
                const saveCurrent=()=>{
                    //不管有几页,把当前全部canvas保存再说
                    for (var i = 0; document.getElementsByTagName("canvas")[i]; i++) {
                        if (enableGreyCompress) {
                            //灰度压缩
                            let cnv = document.getElementsByTagName("canvas")[i];
                            let cnx = cnv.getContext('2d');
                            let width = cnv.width;
                            let height = cnv.height;
                            var imgPixels = cnx.getImageData(0, 0, width, height);
                            for (var y = 0; y < height; y++) {
                                for (var x = 0; x < width; x++) {
                                    let i = (y * 4) * width + x * 4;
                                    var avg = (imgPixels.data[i] + imgPixels.data[i + 1] + imgPixels.data[i + 2]) / 3;
                                    imgPixels.data[i] = avg;
                                    imgPixels.data[i + 1] = avg;
                                    imgPixels.data[i + 2] = avg;
                                }
                            }
                            cnx.putImageData(imgPixels, 0, 0, 0, 0, imgPixels.width, imgPixels.height);

                        }
                        if(lowQuality){
                            PDFfile.addImage(document.getElementsByTagName("canvas")[i].toDataURL("image/webp",0.75),"WEBP",0,0,wP,hP,null,"SLOW");
                        }else{
                            PDFfile.addImage(document.getElementsByTagName("canvas")[i], "WEBP", 0, 0, wP, hP, null, "SLOW");
                        }
                        PDFfile.addPage();
                        page++
                        console.log("Saved One Page!-Qinlili");
                    }
                }
                for (; document.getElementsByClassName("ctrl-icon")[3].className.animVal.indexOf("disabled") < 0;) {
                    SakiProgress.setText("正在保存第" + page + "页...");
                    SakiProgress.setPercent(10 + 80 * (page / totalPage));
                    console.log("Work Current Page:" + page + "...-Qinlili");
                    await sleep(100)
                    saveCurrent();
                    //虽然不知道为什么加了延迟半秒就不会卡住,但既然能用管他为什么呢
                    setTimeout(() => { document.getElementsByClassName("ctrl-icon")[3].parentElement.click(); }, 500)
                    //显示的正在加载的页面可能比实际加载页面小一页,但估计1919810个用户里也没一个意识到,不影响保存效果这种细节就不管了,问就是爷懒的写
                    SakiProgress.setText("正在等待加载第" + (page + 1) + "页...");
                    console.log("Waiting For Loading...-Qinlili");
                    await waitPageChange();
                };
                saveCurrent();
                PDFfile.setFontSize(40);
                PDFfile.text('Powered By Qinlili',35, 65);
                PDFfile.textWithLink('https://greasyfork.org/zh-CN/scripts/429991',35, 25,{align: 'center', url: 'https://greasyfork.org/zh-CN/scripts/429991'});
                //生成文件导出
                SakiProgress.setText("正在导出文件...");
                SakiProgress.setPercent(90);
                PDFfile.save("学习强国电子书导出.pdf", { returnPromise: true }).then(finish => {
                    SakiProgress.clearProgress;
                    SakiProgress.hideDiv();
                });
            }
        } else {
            console.log("Iframe Page " + document.location.pathname + "\nSkip Detect-Qinlili");
        }
    }

    //全局共享函数
    //读取视频信息并下载
    function getInfoAndDL(infourl) {
        console.log("Get Video Info:" + infourl + "\n-Qinlili")
        var xhr = new XMLHttpRequest();
        xhr.onload = event => {
            console.log("Success Get Video Info:" + infourl + "\n-Qinlili")
            if (xhr.readyState === 4 && xhr.status === 200) {
                var videoInfo = JSON.parse(xhr.response.replace("globalCache = ", "").replace(";", ""));
                stringToObject(videoInfo);
                //判断慕课
                if (videoInfo[Object.keys(videoInfo)[0]].info) {
                    videoInfo = videoInfo[Object.keys(videoInfo)[0]].info;
                    for (var vi = 0; videoInfo.ossUrl[vi]; vi++) {
                        console.log("Video Name:" + videoInfo.frst_name + "-" + (vi + 1) + "\nChapter Name:" + videoInfo.mooc_class + "\nMooc Name:" + videoInfo.mooc + "\nVideo Url:" + videoInfo.ossUrl[vi] + "\n-Qinlili");
                        var filename = videoInfo.frst_name + "-" + (vi + 1) + "-" + videoInfo.mooc_class + "-" + videoInfo.mooc + ".mp4"
                        console.log("File Name:" + filename + "\nPrepare Download-Qinlili");
                        downloadFile(videoInfo.ossUrl[vi], filename);
                    }
                }
                //判断电视剧
                if (videoInfo[Object.keys(videoInfo)[0]].list) {
                    videoInfo = videoInfo[Object.keys(videoInfo)[0]].list;
                    for (var vii = 0; videoInfo[vii]; vii++) {
                        console.log("Video Name:" + videoInfo[vii].frst_name + "-" + (vi + 1) + "\nList Name:" + videoInfo[vii].title + "\nVideo Url:" + videoInfo[vii].ossUrl + "\n-Qinlili");
                        var filename2 = videoInfo[vii].frst_name + "-" + (vi + 1) + "-" + videoInfo[vii].title + ".mp4"
                        console.log("File Name:" + filename2 + "\nPrepare Download-Qinlili");
                        downloadFile(videoInfo[vii].ossUrl, filename2);
                    }
                }
            }
            xhr.onerror = () => {
                console.log("Fail Get Video Info:" + infourl + "\n-Qinlili")
            }
        }
        xhr.open('GET', infourl, false);
        xhr.send();

        //平整化Array工具,从学习强国本身的js里抄过来的,大概原理就是尝试把值作为json解析,解析成功就把解析结果替换回去,总之我大受震撼
        function stringToObject(params) {
            for (var key in params) {
                var value = params[key];
                if (isString(value)) {
                    try {
                        if (typeof JSON.parse(value) == 'object') {
                            params[key] = JSON.parse(value);
                        }
                    } catch (e) { }
                } else if (isArray(value)) {
                    try {
                        for (var index = 0; index < value.length; index++) {
                            stringToObject(value[index]);
                        }
                    } catch (e) { }
                } else if (isObject(value)) {
                    try {
                        stringToObject(params[key])
                    } catch (e) { }
                }
            }
        }

        function isArray(o) {
            return Object.prototype.toString.call(o) === '[object Array]';
        }

        function isString(o) {
            return Object.prototype.toString.call(o) === '[object String]';
        }

        function isObject(o) {
            return Object.prototype.toString.call(o) === '[object Object]';
        }
    }

    //跳转避免跨域问题
    if (document.location.host == "article.xuexi.cn") {
        //检测是不是PDF页面,移动端分享文章也是这个域名,之前没发现
        var obj = document.getElementsByTagName("link");
        var isPDF = false;
        for (var pdfJS = 0; obj[pdfJS]; pdfJS++) {
            if (obj[pdfJS].href.indexOf("js/pdf") > 1) {
                isPDF = true;
            }
        }
        if (isPDF) {
            console.log("JS Loaded,Sleep 5 Sec-Qinlili");
            var tip = document.createElement("H1");
            tip.innerText = "即将跳转页面,请等待五秒";
            tip.style = "z-index:9999;position:fixed;backdrop-filter: blur(10px) brightness(100%);background-color: rgba(255, 255, 255, .6);width:100%;margin-top:0px;height:100%;left:0px;right:0px;top:0px;";
            document.body.appendChild(tip);
            await sleep(5000)
            document.location.href = document.getElementsByClassName("pdf-iframe")[0].src;
        }
    }


    //404注入页面避免cors
    if (document.location.host == "boot-source.xuexi.cn") {
        console.log("JS Domain Detected, Prepare Inject-Qinlili");
        document.querySelector("body").innerHTML = "<H2>本页面仅用于CORS注入,分享网址没有用,请允许下载多个文件</H2><H4 id=\"logcat\"></H4>"
        XHRDL.init();
        logcat("Initializing Downloader...");
        await sleep(3000)
        var searchParams = new URLSearchParams(document.location.search);
        var vid = searchParams.get("id");
        document.title = "下载:" + vid;
        logcat("Found ID:" + vid);
        logcat("Get Video Info:" + vid)
        var xhr = new XMLHttpRequest();
        xhr.onload = event => {
            if (xhr.readyState === 4 && xhr.status === 200) {
                logcat("Success Get JSON Info:" + vid)
                var videoInfo = JSON.parse(xhr.response.replace("callback(", "").replace("})", "}"));
                console.log(videoInfo);
                logcat("List Name:" + videoInfo.normalized_title)
                logcat("List Origin:" + videoInfo.show_source)
                //文件名后缀
                var filenamesource = "-" + videoInfo.normalized_title + "-" + videoInfo.show_source;
                //视频下载模式
                if (document.location.pathname == "/newmoocdown") {
                    //检测是否为多视频
                    if (videoInfo.sub_items) {
                        logcat("Found " + videoInfo.sub_items.length + " Videos");
                        //循环解析并下载视频
                        for (var vi = 0; videoInfo.sub_items[vi]; vi++) {
                            logcat("Analysis Video " + (vi + 1));
                            //currentVideo,缩写为cV看起来清爽点
                            var cV = videoInfo.sub_items[vi];
                            var vName = cV.title;
                            logcat("Video Name:" + vName);
                            //检测多个清晰度
                            var vurl = getHighest(cV.videos[0].video_storage_info);
                            logcat("Video Url:" + vurl);
                            var fName = vName + filenamesource + ".mp4";
                            logcat("File Name:" + fName);
                            logcat("Call Downloader, Downloader Log Output In F12");
                            downloadFile(vurl, fName);
                        }
                    } else {
                        //单个视频
                        var singlevurl = getHighest(videoInfo.videos[0].video_storage_info);
                        logcat("Video Url:" + singlevurl);
                        vName = videoInfo.title;
                        var sfName = vName + filenamesource + ".mp4";
                        logcat("File Name:" + sfName);
                        logcat("Call Downloader, Downloader Log Output In F12");
                        downloadFile(singlevurl, sfName);
                    }
                }
                //音频下载模式
                if (document.location.pathname == "/audiodown") {
                    if (videoInfo.sub_items) {
                        logcat("Found " + videoInfo.sub_items.length + " Audios");
                        //循环解析并下载音频
                        for (var ai = 0; videoInfo.sub_items[ai]; ai++) {
                            logcat("Analysis Video " + (ai + 1));
                            //currentAudio,缩写为cA看起来清爽点
                            var cA = videoInfo.sub_items[ai];
                            var aName = cA.title;
                            logcat("Audio Name:" + aName);
                            //音频不区分清晰度
                            var aurl = cA.audios[0].audio_storage_info[0].url;
                            logcat("Audio Url:" + aurl);
                            var afName = aName + filenamesource + ".mp3";
                            logcat("File Name:" + fName);
                            logcat("Call Downloader, Downloader Log Output In F12");
                            downloadFile(aurl, afName);
                        }
                    }
                }
            }
        }
        xhr.onerror = () => {
            logcat("Fail Get Json Info:" + vid)
        }
        xhr.open('GET', "https://boot-source.xuexi.cn/data/app/" + vid + ".js?callback=callback&_st=" + Date.now());
        xhr.send();
        //打印日志方法,空页面就不用console.log了
        function logcat(text) {
            //获取时间参考https://www.jianshu.com/p/067469a4eed8,稍微整合了一下
            document.getElementById("logcat").innerText = new Date().toTimeString().substring(0, 8) + "  " + text + "\n" + document.getElementById("logcat").innerText;
        }
        //分析最高清晰度
        function getHighest(vObj) {
            var maxHeight = 1;
            var maxId = 0;
            for (var vii = 0; vObj[vii]; vii++) {
                if (!(vObj[vii].format == "m3u8")) {
                    if (vObj[vii].height > maxHeight) {
                        maxHeight = vObj[vii].height;
                        maxId = vii;
                    }
                }
            }
            logcat("Max Vide Height:" + maxHeight);
            return vObj[maxId].normal
        }
    }


    //地址转换函数
    function pagetoinfourl(pageurl) {
        var tempurl = pageurl.replace(".html", ".js");
        tempurl = insertStr(tempurl, tempurl.indexOf("/", 21) + 1, "data");
        return tempurl;
    }
    //插入
    function insertStr(soure, start, newStr) {
        return soure.slice(0, start) + newStr + soure.slice(start);
    }
    //下载
    function downloadFile(url, name) {
        XHRDL.newTask(url, name);
    }
}
)();