北航论文平台下载工具,请勿传播下载的文件,否则后果自负。
当前为
// ==UserScript==
// @name BUAA-Thesis-Download
// @namespace https://github.com/xiaotianxt
// @version 0.1.0
// @description 北航论文平台下载工具,请勿传播下载的文件,否则后果自负。
// @author xiaotianxt
// @match https://d.buaa.edu.cn/*/pdfindex.jsp?fid=*
// @match https://copyright.lib.buaa.edu.cn/*/pdfindex.jsp?fid=*
// @require https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js
// @require https://cdnjs.cloudflare.com/ajax/libs/notify/0.4.2/notify.min.js
// @license GNU GPLv3
// ==/UserScript==
(function () {
"use strict";
const RESOLUTION = "buaa_thesis_download.resolution";
const fid = $("#fid").val();
const totalPage = parseInt($("#totalPages").html().replace(/ \/ /, ""));
const baseUrl = `https://copyright.lib.buaa.edu.cn/jumpServlet?fid=${fid}`;
// const baseUrl = `https://d.buaa.edu.cn/https/77726476706e69737468656265737421f3f8518535396f586a4685a59a1b2120b46fb5d23026dc57e5/jumpServlet?fid=${fid}`;
const msgBox = initUI();
initMonitor();
function initUI() {
// 下载按钮
const downloadButton = document.querySelector("#thumbtab").cloneNode(true);
downloadButton.innerHTML = `
<div class="panel-bg" style="background: url("") center center no-repeat;"></div>
<span class="panel-name">下载</span>
`;
document.querySelector("#btnList").appendChild(downloadButton);
downloadButton.addEventListener("click", download);
// 清晰度
const resolution = localStorage.getItem(RESOLUTION) || "2f";
const resolutionRadioGroup = document
.querySelector("#thumbtab")
.cloneNode(true);
resolutionRadioGroup.innerHTML = `
<div resolution>
<input type="radio" name="resolution" id="standard" value="2f"> <label for="standard">标清</label>
<input type="radio" name="resolution" id="high" value="3f"> <label for="high">超清</label>
<input type="radio" name="resolution" id="super" value="5f"> <label for="super">巨清</label>
</div>
`;
document.querySelector("#btnList").appendChild(resolutionRadioGroup);
$("input[name='resolution'][value='" + resolution + "']").prop(
"checked",
true
);
$("input[name='resolution']").on("click", (e) => {
localStorage.setItem(RESOLUTION, e.target.value);
$("#jspPane_scroll img").each((i, elem) => {
elem.src = elem.src.replace(/2f|3f|5f/, e.target.value);
});
$.notify("清晰度已调整", "success");
});
$("input[name='resolution'][value='5f']").on("click", (e) => {
$("[resolution]").notify("图片尺寸过大,加载速度会比较缓慢。", "warn");
});
// msgBox
const msgBox = downloadButton.querySelector("span");
return msgBox;
}
function initMonitor() {
const targetNode = document.getElementById("jspPane_scroll");
const config = { childList: true, subtree: true };
const callback = (mutationList, observer) => {
const resolution = document.querySelector(
'input[name="resolution"]:checked'
).value;
for (const mutation of mutationList) {
if (mutation.type === "childList") {
const target = mutation.target.querySelector("img");
if (target) target.src = target.src.replace(/2f$/, resolution);
}
}
};
// Create an observer instance linked to the callback function
const observer = new MutationObserver(callback);
// Start observing the target node for configured mutations
observer.observe(targetNode, config);
}
async function download(e) {
e.preventDefault();
e.target.disabled = true;
await solveSrc().then(solveImg).then(solvePDF);
e.target.disabled = false;
}
/**
* 解析pdf图片链接
*/
async function solveSrc() {
async function downloadSrcInfo(url) {
return fetch(url)
.then((res) => res.json())
.then((json) => {
finished++;
msgBox.innerHTML = finished + "/" + page;
return json.list;
});
}
let urlPromise = [];
let page = 0;
let finished = 0;
for (; page < totalPage; page++) {
const url = baseUrl + "&page=" + page;
urlPromise.push(downloadSrcInfo(url));
msgBox.innerHTML = finished + "/" + page;
}
return Promise.all(urlPromise);
}
/**
* 下载图片
*/
async function solveImg(urls) {
async function downloadPdf(url, i) {
return fetch(url)
.then((res) => res.blob())
.then((blob) => {
const reader = new FileReader();
reader.readAsDataURL(blob);
return new Promise((resolve) => {
reader.onloadend = () => {
resolve(reader.result);
numFinished++;
msgBox.innerHTML = numFinished + "/" + numTotal;
};
});
});
}
// remove duplicated
const map = new Map();
const resolution = localStorage.getItem(RESOLUTION) || "2f";
urls.forEach((triple) => {
triple.forEach((item) => {
map.set(item.id, item.src.replace(/2f$/, resolution));
});
});
// sort and clear index
urls = [...map.entries()]
.sort((a, b) => a[0] - b[0])
.map((item) => item[1]);
// download images
const base64Promise = [];
let numFinished = 0;
let numTotal = 0;
urls.forEach((url) => {
base64Promise.push(downloadPdf(url));
numTotal++;
msgBox.innerHTML = numFinished + "/" + numTotal;
});
return Promise.all(base64Promise);
}
/**
* 拼接为pdf
* @param {*} base64s
*/
async function solvePDF(base64s) {
msgBox.innerHTML = "拼接中";
const doc = new jspdf.jsPDF();
base64s.forEach((base64, index) => {
doc.addImage(base64, "JPEG", 0, 0, 210, 297);
index + 1 == base64s.length || doc.addPage();
});
msgBox.innerHTML = "保存中";
doc.save(document.title + ".pdf");
msgBox.innerHTML = "完成!";
}
})();