// ==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 // @downloadURL https://update.greasyfork.icu/scripts/510224/GB688%20downloader%3B%E5%9B%BD%E6%A0%87%E7%BD%91%E4%B8%8B%E8%BD%BD.user.js // @updateURL https://update.greasyfork.icu/scripts/510224/GB688%20downloader%3B%E5%9B%BD%E6%A0%87%E7%BD%91%E4%B8%8B%E8%BD%BD.meta.js // ==/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 = '下载PDF'; downloadButton.addEventListener('click', downloadPDF); downloadButton.id = 'downloadBtn'; var progressText = document.createElement('span'); container.appendChild(progressText); progressText.id = 'progressText'; progressText.innerHTML = '如需滚动请缓慢滚动,以免缓存图像错位'; } (async function () { init() })();