您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
北航论文平台下载工具,请勿传播下载的文件,否则后果自负。
当前为
- // ==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 = "完成!";
- }
- })();