- // ==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://gf.qytechs.cn/zh-CN/scripts/429991',35, 25,{align: 'center', url: 'https://gf.qytechs.cn/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);
- }
- }
- )();