// ==UserScript==
// @name GB688 downloader;国标网下载
// @version 2024.12.23.01
// @license Apache-2.0
// @namespace https://github.com/yikuaibaiban/gb688_downloader
// @description 基于国标网显示规律在本地拼合成PDF并提供下载
// @author yikuaibaiban
// @icon 
// @match http://c.gb688.cn/*
// @grant none
// @run-at document-start
// ==/UserScript==
(function initStyles() {
let style = document.createElement("style");
style.appendChild(document.createTextNode(`.downloader_container { position: fixed; height: 40px; width: 120px; top: calc(50% - 20px); right: 20px;}.downloader_container button { width: 100%; height: 40px; background-color: blueviolet; border: none; color: white; line-height: 40px; text-align: center; font-size: 20px; border-radius: 5px; transition: all 0.3s; display: flex; justify-content: center; align-items: center;}.downloader_container button:hover { background-color: dodgerblue;}@keyframes loading { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); }}.loading { width: 18px; height: 18px; margin-right: 5px; animation: loading 1s infinite;}`));
document.head.appendChild(style);
})();
let viewGbImgCache = [];
async function interceptsXHRCallback(url, response, method, readyState) {
if (readyState !== 4) return;
if (url.includes("viewGbImg?fileName=")) {
let finds = viewGbImgCache.filter(item => item.url === url);
if (!finds.length) {
let img = new Image();
img.src = await blobToBase64(response);//URL.createObjectURL(response);
viewGbImgCache.push({url, img});
}
}
}
interceptsXHR(interceptsXHRCallback);
function interceptsXHR(callback) {
const open = window.XMLHttpRequest.prototype.open;
window.XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
this.addEventListener('readystatechange', function () {
callback(url, this.response, method, this.readyState);
});
open.apply(this, arguments);
}
}
function blobToBase64(blob) {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.readAsDataURL(blob);
reader.onloadend = () => {
const base64String = reader.result;
resolve(base64String);
};
reader.onerror = (error) => {
reject(error);
};
});
}
function downloadPDF() {
let downloadBtn = document.querySelector('#downloadBtn');
downloadBtn.disabled = true;
let loadingIcon = new Image();
loadingIcon.src =
'';
loadingIcon.className = 'loading';
downloadBtn.insertBefore(loadingIcon, downloadBtn.children[0]);
let pheight = document.querySelector('div.page').clientHeight;
let pwidth = document.querySelector('div.page').clientWidth;
let qwidth = pwidth * 0.1;
let qheight = pheight * 0.1;
let canvasArray = new Array();
let progressText = document.querySelector('#progressText');
progressText.innerHTML = "正在提取图片...";
document.querySelectorAll('.page').forEach(function (elem, i) {
let canvas = document.createElement('canvas');
canvas.width = pwidth;
canvas.height = pheight;
canvas.setAttribute('complete', '0');
canvasArray.push(canvas);
if (elem.hasAttribute('bg')) {
let url = "viewGbImg?fileName=" + elem.getAttribute('bg');
let finds = viewGbImgCache.filter(item => item.url === url);
if (!finds.length) {
viewGbImgCache.push({url, img: null});
}
}
});
progressText.innerHTML = "正在下载图片...";
viewGbImgCache.forEach((item, i) => {
if (item.img !== null) {
return;
}
fetch(item.url, {headers: {'Cache-Alive': 'chunked'}})
.then((response) => {
if (response.status === 200) {
return response.blob();
}
})
.then(async (blob) => {
let img = new Image();
img.src = await blobToBase64(blob) // URL.createObjectURL(blob);
viewGbImgCache[i].img = img;
});
});
let timer1 = setInterval(() => {
for (let i = 0; i < viewGbImgCache.length; i++) {
if (viewGbImgCache[i].img === null) {
return;
}
}
clearInterval(timer1);
progressText.innerHTML = "正在拼合图像...";
document.querySelectorAll('.page').forEach((elem, i) => {
let canvas = canvasArray[i];
let ctx = canvas.getContext('2d');
ctx.fillStyle = 'white';
ctx.fillRect(0, 0, pwidth, pheight);
try {
$(elem)
.children('span')
.each(function (j, s) {
ctx.drawImage(
viewGbImgCache[i === 0 ? 0 : Math.ceil(i / 4) - 1].img,
-parseInt($(s).css('background-position-x')),
-parseInt($(s).css('background-position-y')),
qwidth,
qheight,
$(s).attr('class').split('-')[1] * qwidth,
$(s).attr('class').split('-')[2] * qheight,
qwidth + 1,
qheight
);
});
} catch (e) {
console.log(viewGbImgCache[i === 0 ? 0 : Math.ceil(i / 4) - 1].img);
}
canvas.setAttribute('complete', '1');
});
}, 500);
let timer = setInterval(() => {
for (let i = 0; i < canvasArray.length; i++) {
const element = canvasArray[i];
if (element.getAttribute('complete') != '1') {
return;
}
}
progressText.innerHTML = "正在转换PDF...";
const {jsPDF} = window.jspdf;
const pdf = new jsPDF('p', 'px', [pwidth, pheight]);
let title = $('title').text().split('|')[1].toString().trim();
canvasArray.forEach(function (e, i) {
progressText.innerHTML = `正在转换PDF(${i + 1}/${canvasArray.length})...`;
pdf.addImage(
e.toDataURL('image/jpeg'),
'jpeg',
0,
0,
pwidth,
pheight,
'',
'MEDDIUM'
);
pdf.addPage();
});
let targetPage = pdf.internal.getNumberOfPages();
pdf.deletePage(targetPage);
progressText.innerHTML = "PDF转换完成...";
pdf.save(title + '.pdf');
clearInterval(timer);
downloadBtn.disabled = false;
downloadBtn.removeChild(downloadBtn.children[0]);
}, 500);
}
function init() {
var jspdfScript = document.createElement('script');
jspdfScript.src =
'https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js';
document.querySelector('head').appendChild(jspdfScript);
var container = document.createElement('div');
document.querySelector('body').appendChild(container);
container.className = 'downloader_container';
container.style.zIndex = "999";
var downloadButton = document.createElement('button');
container.appendChild(downloadButton);
downloadButton.innerHTML = '<span>下载PDF</span>';
downloadButton.addEventListener('click', downloadPDF);
downloadButton.id = 'downloadBtn';
var progressText = document.createElement('span');
container.appendChild(progressText);
progressText.id = 'progressText';
progressText.innerHTML = '如需滚动请缓慢滚动,以免缓存图像错位';
}
(async function () {
init()
})();