// ==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);
}
}
)();