您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
一键下载Pixiv各页面原图。支持多图下载,动图下载,画师作品批量下载。动图支持格式转换:Gif | Apng | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。
当前为
// ==UserScript== // @name Pixiv Downloader // @namespace https://greasyfork.org/zh-CN/scripts/432150 // @version 0.5.4 // @description:en Download the original images of Pixiv pages with one click. Supports:multiple illustrations, ugoira(animation), and batch downloads of artists' work. Ugoira support format conversion: Gif | Apng | Webm. The downloaded images will be saved in a separate folder named after the artist (you need to adjust the tampermonkey "Download" setting to "Browser API"). A record of downloaded images is kept. // @description 一键下载Pixiv各页面原图。支持多图下载,动图下载,画师作品批量下载。动图支持格式转换:Gif | Apng | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。 // @description:zh-TW 一鍵下載Pixiv各頁面原圖。支持多圖下載,動圖下載,畫師作品批次下載。動圖支持格式轉換:Gif | Apng | Webm。下載的圖片將保存到以畫師名命名的單獨文件夾(需要調整tampermonkey“下載”設置為“瀏覽器API”)。保留已下載圖片的紀錄。 // @author ruaruarua // @match https://www.pixiv.net/* // @icon https://www.pixiv.net/favicon.ico // @noframes // @grant GM_xmlhttpRequest // @grant GM_download // @grant GM_setValue // @grant GM_getValue // @grant GM_info // @grant GM_registerMenuCommand // @connect i.pximg.net // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/gif.js/0.2.0/gif.js // ==/UserScript== (function () { 'use strict'; const style = ` @property --pdl-progress { syntax: '<percentage>'; inherits: true; initial-value: 0%; } @keyframes pdl_loading { 100% { transform: translate(-50%, -50%) rotate(360deg); } } .pdl-btn { position: relative; border-top-right-radius: 8px; background: no-repeat center/85%; background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%233C3C3C' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E"); color: #01b468; display: inline-block; font-size: 13px; font-weight: bold; height: 32px; line-height: 32px; margin: 0; overflow: hidden; padding: 0; border: none; text-decoration: none!important; text-align: center; text-overflow: ellipsis; user-select: none; white-space: nowrap; width: 32px; z-index: 1; cursor: pointer; } .pdl-btn-main { margin: 0 0 0 10px; } .pdl-btn-sub { bottom: 0; background-color: rgba(255, 255, 255, .5); left: 0; position: absolute; } .pdl-btn-sub.artworks{ position: sticky; top: 40px; border-radius: 4px; } .pdl-btn-sub.presentation{ position: fixed; top: 50px; right: 16px; border-radius: 8px; left: auto; } .pdl-btn-sub-bookmark.pdl-btn-sub-bookmark { left: auto; right: 0; bottom: 34px; border-radius: 8px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; } .pdl-error { background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23EA0000' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E"); } .pdl-complete { background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%2301B468' d='M256 8C119.033 8 8 119.033 8 256s111.033 248 248 248 248-111.033 248-248S392.967 8 256 8zm0 48c110.532 0 200 89.451 200 200 0 110.532-89.451 200-200 200-110.532 0-200-89.451-200-200 0-110.532 89.451-200 200-200m140.204 130.267l-22.536-22.718c-4.667-4.705-12.265-4.736-16.97-.068L215.346 303.697l-59.792-60.277c-4.667-4.705-12.265-4.736-16.97-.069l-22.719 22.536c-4.705 4.667-4.736 12.265-.068 16.971l90.781 91.516c4.667 4.705 12.265 4.736 16.97.068l172.589-171.204c4.704-4.668 4.734-12.266.067-16.971z'%3E%3C/path%3E %3C/svg%3E"); } .pdl-progress { background-image: none; cursor: default; } .pdl-progress:after{ content: ''; display: inline-block; position: absolute; top: 50%; left: 50%; width: 27px; height: 27px; transform: translate(-50%, -50%); -webkit-mask: radial-gradient(transparent, transparent 54%, #000 57%, #000); mask: radial-gradient(transparent, transparent 54%, #000 57%, #000); border-radius: 50%; } .pdl-progress:not(:empty):after { background: conic-gradient(#01B468 0, #01B468 var(--pdl-progress), transparent var(--pdl-progress), transparent); transition: --pdl-progress .2s ease; } .pdl-progress:empty:after { background: conic-gradient(#01B468 0, #01B468 25%, #01B46833 25%, #01B46833); animation: 1.5s infinite linear pdl_loading; } .pdl-nav-placeholder { flex-grow: 1; height: 42px; line-height: 42px; text-align: right; font-weight: bold; font-size: 16px; color: rgb(133, 133, 133); border-top: 4px solid transparent; cursor: default; white-space: nowrap; } .pdl-btn-all, .pdl-stop { background-color: transparent; border: none; } .pdl-btn-all::before, .pdl-stop::before { content: ''; height: 24px; width: 24px; transition: background-image 0.2s ease 0s; background: no-repeat center/85%; } .pdl-btn-all::before { background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E"); } .pdl-stop::before { background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E"); } .pdl-btn-all:hover::before{ background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E"); } .pdl-stop:hover::before { background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E"); } .pdl-hide { display: none!important; } .pdl-wrap { text-align: right; padding-right: 24px; font-weight: bold; font-size: 14px; line-height: 14px; color: rgb(133, 133, 133); transition: color 0.2s ease 0s; } .pdl-wrap:hover { color: rgb(31, 31, 31); } .pdl-wrap label { padding-left: 8px; cursor: pointer; } .pdl-wrap input { vertical-align: top; appearance: none; position: relative; box-sizing: border-box; width: 28px; border: 2px solid transparent; cursor: pointer; border-radius: 14px; height: 14px; background-color: rgba(133, 133, 133); transition: background-color 0.2s ease 0s, box-shadow 0.2s ease 0s; } .pdl-wrap input:hover { background-color: rgba(31, 31, 31); } .pdl-wrap input::after { content: ""; position: absolute; display: block; top: 0px; left: 0px; width: 10px; height: 10px; transform: translateX(0px); background-color: rgb(255, 255, 255); border-radius: 10px; transition: transform 0.2s ease 0s; } .pdl-wrap input:checked { background-color: rgb(0, 150, 250); } .pdl-wrap input:checked::after { transform: translateX(14px); } .pdl-wrap-artworks { position: absolute; right: 8px; top: 0px; bottom: 0px; margin-top: 40px; } .pdl-modal * { font-family: 'win-bug-omega, system-ui, -apple-system, "Segoe UI", Roboto, Ubuntu, Cantarell, "Noto Sans", "Hiragino Kaku Gothic ProN", Meiryo, sans-serif'; line-height: 1.15; } .pdl-modal { position: fixed; top: 0; left: 0; width: 100%; height: 100%; display: flex; z-index: 99; background-color: rgba(0, 0, 0, 0.32); user-select: none; } .pdl-dialog { position: relative; background-color: #fff; border-radius: 24px; margin: auto; padding: 20px 40px 30px 40px; max-width: 720px; font-size: 16px; } .pdl-dialog-header > h3 { font-weight: bold; font-size: 1.17em; margin: 1em 0; } .pdl-dialog p { margin: 1em 0px; overflow-wrap: break-word; } .pdl-dialog-close { position: absolute; top: 10px; right: 10px; margin: 0; padding: 0; width: 25px; height: 25px; border: none; cursor: pointer; border-radius: 50%; background-color: transparent; transform: rotate(45deg); transition: 0.25s background-color; background: linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/18px 2px no-repeat, linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/2px 18px no-repeat; } .pdl-dialog-close:hover { background-color: rgba(0, 0, 0, 0.05); } .pdl-dialog-content { user-select: text; }`; function addStyle() { const sty = document.createElement("style"); sty.innerHTML = style; document.head.appendChild(sty); } function debugLog(...msgs) { } const defaultSettings = { version: "0.5.4", ugoriaFormat: "zip", folderPattern: "{artist}", filenamePattern: "{artist}_{title}_{id}_p{page}", showMsg: true, log: false, }; function migraSettings(settings) { if (localStorage.pdlFormat) { settings.ugoriaFormat = localStorage.pdlFormat; localStorage.removeItem("pdlFormat"); } if (localStorage.pdlFilename) { settings.filenamePattern = localStorage.pdlFilename.replace("{author}", "{artist}"); localStorage.removeItem("pdlFilename"); } } function getSettings() { let settings; if (!localStorage.pdlSetting) { settings = defaultSettings; migraSettings(settings); saveSettings(settings); } else { settings = JSON.parse(localStorage.pdlSetting); if (settings.version !== defaultSettings.version) { settings.version = defaultSettings.version; settings.showMsg = true; for (const key in defaultSettings) { if (!(key in settings)) { settings[key] = defaultSettings[key]; } } saveSettings(settings); } } return settings; } function saveSettings(settingObj) { settingObj = settingObj || settings; localStorage.pdlSetting = JSON.stringify(settingObj); } function upgradeSettings(key, value) { if (key in settings) { if (settings[key] === value) return; settings[key] = value; saveSettings(); } } const settings = getSettings(); function createSetFormatFn(format) { return () => { if (settings.ugoriaFormat !== format) { upgradeSettings("ugoriaFormat", format); } }; } const regexp = { artworksPage: /artworks\/(\d+)$/, userPage: /users\/(\d+)/, ppSearchPage: /\/tags\/.*\/(artworks|illustrations|manga)/, bookmarkPage: /users\/\d+\/bookmarks\/artworks/, suscribePage: /bookmark_new_illust/, activityHref: /illust_id=(\d+)/, originSrcPageNum: /(?<=_p)\d+/, }; const artworkType = { ILLUSTS: 0, MANGA: 1, UGOIRA: 2, }; const depsUrls = { gifWorker: "https://cdnjs.cloudflare.com/ajax/libs/gif.js/0.2.0/gif.worker.js", pako: "https://cdnjs.cloudflare.com/ajax/libs/pako/2.0.4/pako.min.js", upng: "https://cdnjs.cloudflare.com/ajax/libs/upng-js/2.1.0/UPNG.min.js", }; const text = { upgradeMsgTitle: `<h3>Pixiv Downloader ${settings.version}</h3>`, upgradeMsgContent: `<p>1. 现在可以设置保存图片的文件夹名字了。</p><p>2. 现在漫画的多图预览对话框有下载按钮了。</p><p>请注意:FireFox浏览器 Tampermonkey 升级4.18.0后,需转换格式的动图无法保存到画师目录,请理解。</p>`, modalCreditFooter: `<style>.pdl-dialog-footer { position: relative; font-size: 12px; }</style><details style="margin-top: 1.5em;"> <summary style="display: inline-block; list-style: none; cursor: pointer; color: rgb(0, 0, 238); text-decoration: underline">脚本还行?请我喝杯可乐吧!</summary> <img style="display: block; margin: 1em auto; width: 200px" src="" /> <p style="text-align: center">祝你每天都能找到对的色图,就像我每天都能喝到香草味可乐</p> </details>`, modalFeedback: `<a target="_blank" style="position: absolute; right: 0px; top: 0px; color: rgb(0, 0, 238); text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback">有问题or想建议?这里反馈</a>`, filePathSettingTitle: `<h3>设置文件名</h3>`, filePathSettingContent: `<style>.pdl-dialog-content input {height: auto; padding: 0.5em; line-height: 1.5; margin: 0.6em 0 0.3em 0; font-size: 16px;} .pdl-dialog-content label {cursor: default; line-height: 1.15;}</style><div style="display: flex; gap: 20px; justify-content: space-between;"> <div> <label style="display: block;" for="pdlfolder">文件夹名:</label> <input type="text" id="pdlfolder" style="width: 200px;" placeholder="我不想保存到画师文件夹"> </div> <div> <label style="display: block;" for="pdlfilename">文件名:</label> <input type="text" id="pdlfilename" style="width: 300px;" placeholder="你的名字?"> </div> </div> <p style="font-size: 14px; margin: 0.5em 0"> {artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}: 作品pixiv ID, {page}: 页码。 </p> <p style="font-size: 14px; margin: 0.5em 0">如果不想保存到画师目录,文件夹名留空即可。</p> </div>`, modalOperationBar: `<style> .pdl-dialog-footer button { font-size: 16px; background-color: transparent; border: 1px solid; color: rgb(125,125,125); border-radius: 5px; padding: 0.5em 1.5em; cursor: pointer; transition: .2s opacity; line-height: 1.15; } .pdl-dialog-footer button:hover{ opacity: 0.7; } </style> <div style="display: flex; justify-content: flex-end; margin-top: 1.5em; gap: 1.5em;"> <button id="pdlcancel">取消</button><button id="pdlconfirm" style="border-color: #01b468; background-color: #01b468; color: #fff;">确认</button></div>`, }; function initialDeps(urls) { return Promise.all([_getGifWS(urls.gifWorker), _getApngWS(urls.pako, urls.upng)]).then( ([gif, apng]) => { this._deps.gif = URL.createObjectURL(new Blob([gif], { type: "text/javascript" })); this._deps.apng = URL.createObjectURL(new Blob([apng], { type: "text/javascript" })); return this; } ); } function _fetchDeps(url) { return fetch(url) .then((res) => { if (res.ok) return res.text(); throw new Error(res.status + res.statusText); }) .catch((err) => { console.log("[Pixiv Downloader]Fetch dependency failed.", url, err); return ""; }); } async function _getGifWS(url) { let gifWS; if (!(gifWS = await GM_getValue("gifWS"))) { gifWS = await _fetchDeps(url); if (!gifWS) throw new Error("[Pixiv Downloader]Can not fetch gif worker script."); GM_setValue("gifWS", gifWS); } return gifWS; } async function _getApngWS(pakoUrl, upngUrl) { let apngWS; if (!(apngWS = await GM_getValue("apngWS"))) { let pako = _fetchDeps(pakoUrl); let upng = _fetchDeps(upngUrl); pako = await pako; upng = await upng; if (!pako || !upng) throw new Error("[Pixiv Downloader]Can not fetch apng script."); upng = upng.replace("window.UPNG", "UPNG").replace("window.pako", "pako"); const workerEvt = `onmessage = (evt) => { const {data, width, height, delay } = evt.data; const png = UPNG.encode(data, width, height, 0, delay, {loop: 0}); if (!png) console.log('Convert Apng failed.'); postMessage(png); };`; apngWS = workerEvt + pako + upng; GM_setValue("apngWS", apngWS); } return apngWS; } function _createImgElements(zip) { const eles = []; zip.forEach((relativePath, file) => { eles.push( new Promise((resolve) => { const image = new Image(); image.onload = () => { resolve(image); }; file.async("blob").then((blob) => void (image.src = URL.createObjectURL(blob))); }) ); }); return Promise.all(eles); } function createInstance() { const zip = new JSZip(); const freeApngWorkers = []; const apngWorkers = []; const MAX_CONVERT = 2; let queue = []; let active = []; let isStop = false; const convertTo = { gif: (frames, convertMeta) => { return new Promise((resolve, reject) => { let gif = new GIF({ workers: 2, quality: 10, workerScript: this._deps.gif, }); convertMeta.abort = convertMeta._baseAbort.bind(null, gif.abort.bind(gif)); debugLog("[Info]Start convert:", convertMeta.id); frames.forEach((frame, i) => { gif.addFrame(frame, { delay: convertMeta.framesInfo[i].delay }); }); gif.on( "progress", (() => { const type = "gif"; return (progress) => { debugLog("[Info]Convert progress:", convertMeta.id); if (typeof convertMeta.onProgress === "function") convertMeta.onProgress(progress, type); }; })() ); gif.on("finished", (gifBlob) => { gif = null; resolve(gifBlob); }); gif.on("abort", () => { gif = null; reject("[Info]Convert stop: abort. " + convertMeta.id); }); gif.render(); }); }, png: (frames, convertMeta) => { return new Promise((resolve, reject) => { let canvas = document.createElement("canvas"); const width = (canvas.width = frames[0].naturalWidth); const height = (canvas.height = frames[0].naturalHeight); const context = canvas.getContext("2d", { willReadFrequently: true }); const data = []; const delay = convertMeta.framesInfo.map((frameInfo) => { return Number(frameInfo.delay); }); frames.forEach((frame) => { if (convertMeta.isAborted) throw "[Info]Convert stop manually, reject when drawImage. " + convertMeta.id; context.clearRect(0, 0, width, height); context.drawImage(frame, 0, 0, width, height); data.push(context.getImageData(0, 0, width, height).data); }); canvas = null; debugLog("[Info]Start convert:", convertMeta.id); let worker; if (apngWorkers.length === MAX_CONVERT) { worker = freeApngWorkers.shift(); } else { worker = new Worker(this._deps.apng); apngWorkers.push(worker); } convertMeta.abort = convertMeta._baseAbort.bind(null, () => { reject("[Info]Convert stop manually, reject when convert apng. " + convertMeta.id); worker.terminate(); apngWorkers.splice(apngWorkers.indexOf(worker), 1); }); worker.onmessage = function (e) { if (queue.length) { freeApngWorkers.push(worker); } else { worker.terminate(); apngWorkers.splice(apngWorkers.indexOf(worker), 1); } if (!e.data) { return reject("[Error]apng data is null. " + convertMeta.id); } const pngBlob = new Blob([e.data], { type: "image/png" }); resolve(pngBlob); }; const cfg = { data, width, height, delay }; worker.postMessage(cfg); }); }, webm: (frames, convertMeta) => { return new Promise((resolve, reject) => { let canvas = document.createElement("canvas"); const width = (canvas.width = frames[0].naturalWidth); const height = (canvas.height = frames[0].naturalHeight); const context = canvas.getContext("2d"); const stream = canvas.captureStream(); const recorder = new MediaRecorder(stream, { mimeType: "video/webm", videoBitsPerSecond: 80000000, }); const delay = convertMeta.framesInfo.map((frame) => { return Number(frame.delay); }); let data = []; let frame = 0; const displayFrame = () => { context.clearRect(0, 0, width, height); context.drawImage(frames[frame], 0, 0); if (convertMeta.isAborted) { return recorder.stop(); } setTimeout(() => { if (typeof convertMeta.onProgress === "function") convertMeta.onProgress((frame + 1) / frames.length, "webm"); if (frame === frames.length - 1) { return recorder.stop(); } else { frame++; } displayFrame(); }, delay[frame]); }; recorder.ondataavailable = (event) => { if (event.data && event.data.size) { data.push(event.data); } }; recorder.onstop = () => { canvas = null; if (convertMeta.isAborted) { return reject( "[info]Convert stop manually, reject when convert webm." + convertMeta.id ); } resolve(new Blob(data, { type: "video/webm" })); }; displayFrame(); recorder.start(); }); }, }; const convert = (convertMeta) => { const { id, data, convertResolve, convertReject } = convertMeta; let frames; active.push(convertMeta); if (typeof convertMeta.onProgress === "function") convertMeta.onProgress(0, "zip"); zip .folder(id) .loadAsync(data) .then(_createImgElements) .then((imgEles) => { zip.remove(id); frames = imgEles; if (convertMeta.isAborted) throw "[Info]Convert stop manually, reject when unzip. " + id; return convertTo[convertMeta.format](frames, convertMeta); }) .then(convertResolve) .catch(convertReject) .finally(() => { frames.forEach((frame) => URL.revokeObjectURL(frame.src)); frames = null; active.splice(active.indexOf(convertMeta), 1); if (queue.length) convert(queue.shift()); }); }; return { add: (convertMeta) => { debugLog("[Info]Converter add", convertMeta.id); return new Promise((convertResolve, convertReject) => { convertMeta.isAborted = false; convertMeta.convertResolve = convertResolve; convertMeta.convertReject = convertReject; convertMeta._baseAbort = (callBack) => { if (typeof callBack === "function") callBack(); convertMeta.isAborted = true; }; convertMeta.abort = convertMeta._baseAbort; queue.push(convertMeta); while (active.length < MAX_CONVERT && queue.length && !isStop) { convert(queue.shift()); } }); }, del: (metas) => { if (!metas.length) return; isStop = true; active = active.filter((convertMeta) => { if (metas.find((meta) => meta.id === convertMeta.id)) { convertMeta.abort(); } else { return true; } }); queue = queue.filter((convertMeta) => !metas.find((meta) => meta.id === convertMeta.id)); isStop = false; while (active.length < MAX_CONVERT && queue.length) { convert(queue.shift()); } }, }; } const createConverter = { _deps: { gif: "", apng: "", }, initialDeps, createInstance, }; const _isBlobDlAvaliable = !( navigator.userAgent.includes("Firefox") && GM_info.scriptHandler === "Tampermonkey" && parseFloat(GM_info.version) > 4.17 ); const _isNeedConvert = (meta) => { return meta.illustType === artworkType.UGOIRA && settings.ugoriaFormat !== "zip"; }; const _ffSave = (blob, meta) => { const dlEle = document.createElement("a"); dlEle.href = URL.createObjectURL(blob); dlEle.download = meta.path.slice(meta.path.indexOf("/") + 1); dlEle.click(); URL.revokeObjectURL(dlEle.href); meta.resolve(meta); }; const _normalSave = (blob, meta) => { const imgUrl = URL.createObjectURL(blob); const request = { url: imgUrl, name: meta.path, onerror: (error) => { console.log("[pixiv downloader]Error when saving", meta.path); URL.revokeObjectURL(imgUrl); meta.reject && meta.reject(error); }, onload: () => { if (typeof meta.onLoad === "function") meta.onLoad(); URL.revokeObjectURL(imgUrl); meta.resolve(meta); }, }; meta.abort = GM_download(request).abort; }; function createDownloader(converter) { const MAX_DOWNLOAD = 5; const MAX_RETRY = 3; let isStop = false; let queue = []; let active = []; let save; if (_isBlobDlAvaliable) { save = _normalSave; } else { debugLog("[Info]Run at firefox && TM version:", GM_info.version); save = _ffSave; } const download = (meta) => { debugLog("[Info]Start download:", meta.path); active.push(meta); let abortObj; if (!_isBlobDlAvaliable && !_isNeedConvert(meta)) { abortObj = GM_download({ url: meta.src, name: meta.path, headers: { referer: "https://www.pixiv.net", }, ontimeout: errHandler.bind(null, meta), onerror: errHandler.bind(null, meta), onload: () => { debugLog("[Info]Download complete", meta.path); if (typeof meta.onLoad === "function") meta.onLoad(); meta.resolve(meta); }, }); } else { const request = { url: meta.src, timeout: 20000, method: "GET", headers: { referer: "https://www.pixiv.net", }, responseType: "blob", ontimeout: errHandler.bind(null, meta), onprogress: (e) => { if (e.lengthComputable && typeof meta.onProgress === "function") { meta.onProgress(e.loaded / e.total); } }, onload: (e) => { debugLog("[Info]Download complete", meta.id); if (!meta.state) return debugLog("[Warning]But download was canceled.", meta.id); if (_isNeedConvert(meta)) { const convertMeta = { id: meta.id, data: e.response, format: settings.ugoriaFormat, framesInfo: meta.ugoiraMeta.frames, onProgress: meta.onProgress, }; converter.add(convertMeta).then((blob) => { save(blob, meta); }, meta.reject); } else { save(e.response, meta); } active.splice(active.indexOf(meta), 1); if (queue.length && !isStop) download(queue.shift()); }, onerror: errHandler.bind(null, meta), }; abortObj = GM_xmlhttpRequest(request); } meta.abort = () => { meta.state = 0; abortObj.abort(); meta.reject("[Warning]xhr abort manually. " + meta.id); }; }; const add = (metas) => { if (metas.length < 1) return; const promises = []; metas.forEach((meta) => { promises.push( new Promise((resolve, reject) => { meta.state = 1; meta.resolve = resolve; meta.reject = reject; }) ); }); queue = queue.concat(metas); while (active.length < MAX_DOWNLOAD && queue.length && !isStop) { download(queue.shift()); } return Promise.all(promises); }; const del = (metas) => { if (!metas.length) return; isStop = true; active = active.filter((meta) => { if (metas.includes(meta)) { meta.abort(); } else { return true; } }); queue = queue.filter((meta) => !metas.includes(meta)); isStop = false; while (active.length < MAX_DOWNLOAD && queue.length) { download(queue.shift()); } }; const errHandler = (meta) => { debugLog("[Error]xmlhttpRequest timeout:", meta.src); if (!meta.retries) { meta.retries = 1; } else { meta.retries++; } if (meta.retries > MAX_RETRY) { meta.reject("[Error]xmlhttpRequest failed: " + meta.src); console.log("[pixiv downloader]Network error:", meta.path, meta.src); active.splice(active.indexOf(meta), 1); if (queue.length && !isStop) download(queue.shift()); } else { debugLog("[Warning]retry xhr:", meta.retries, meta.src); download(meta); } }; return { add: add, del: del, }; } function createParser() { const replaceInvalidChar = (string) => { if (!string) return; const temp = document.createElement("div"); temp.innerHTML = string; string = temp.textContent; return string .trim() .replace(/^\.|\.$/g, "") .replace(/[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?|]/g, "") .replace(/"/g, "'") .replace(/</g, "﹤") .replace(/>/g, "﹥"); }; const getFilePath = ({ user, userId, title, illustId, page, ext }) => { const path = settings.folderPattern ? settings.folderPattern + "/" + settings.filenamePattern : settings.filenamePattern; return ( path .replaceAll("{author}", user) .replaceAll("{artist}", user) .replaceAll("{artistID}", userId) .replaceAll("{title}", title) .replaceAll("{page}", page) .replaceAll("{id}", illustId) + ext ); }; const parseByIllust = async (illustId) => { const res = await fetch("https://www.pixiv.net/artworks/" + illustId); if (!res.ok) throw new Error("[Error]fetch artworksURL failed: " + res.status); const htmlText = await res.text(); const preloadData = JSON.parse(htmlText.match(/"meta-preload-data" content='(.*)'>/)[1]); if (!preloadData.illust) throw new Error("[Error]Fail to parse meta preload data."); const illustInfo = preloadData.illust[illustId]; const user = replaceInvalidChar(illustInfo.userName) || "userId-" + illustInfo.userId; const title = replaceInvalidChar(illustInfo.illustTitle) || "illustId-" + illustInfo.illustId; const illustType = illustInfo.illustType; let metas = []; const pathInfo = { user, title, illustId, userId: illustInfo.userId, ext: "", page: 0, }; if (illustType === artworkType.ILLUSTS || illustType === artworkType.MANGA) { const firstImgSrc = illustInfo.urls.original; const srcPrefix = firstImgSrc.slice(0, firstImgSrc.indexOf("_") + 2); const srcSuffix = firstImgSrc.slice(-4); pathInfo.ext = srcSuffix; for (let i = 0; i < illustInfo.pageCount; i++) { pathInfo.page = i; metas.push({ id: illustId, illustType: illustType, path: getFilePath(pathInfo), src: srcPrefix + i + srcSuffix, }); } } if (illustType === artworkType.UGOIRA) { const ugoiraRes = await fetch( "https://www.pixiv.net/ajax/illust/" + illustId + "/ugoira_meta" ); if (!ugoiraRes.ok) throw new Error("[Error]fetch ugoira meta failed: " + res.status); const ugoira = await ugoiraRes.json(); pathInfo.ext = "." + settings.ugoriaFormat; metas.push({ id: illustId, illustType: illustType, path: getFilePath(pathInfo), src: ugoira.body.originalSrc, ugoiraMeta: ugoira.body, }); } return metas; }; const parseByUser = async (userId, type) => { const res = await fetch("https://www.pixiv.net/ajax/user/" + userId + "/profile/all"); if (!res.ok) throw new Error("fetch user profile failed: " + res.status); const profile = await res.json(); let illustIds; if (type) { illustIds = Object.keys(profile.body[type]); } else { illustIds = Object.keys(profile.body.illusts).concat(Object.keys(profile.body.manga)); } return illustIds; }; return { id: parseByIllust, user: parseByUser, }; } let converter; let downloader; let parser; async function initial() { converter = await createConverter .initialDeps(depsUrls) .then((createConverter) => createConverter.createInstance()); parser = createParser(); downloader = createDownloader(converter); } function add(ele) { this._records.add(ele); } function has(ele) { return this._records.has(ele); } function getHistory() { const storage = localStorage.pixivDownloader || "[]"; return new Set(JSON.parse(storage)); } function updateHistory() { Object.keys(localStorage).forEach((key) => { const matchResult = /pdlTemp-(\d+)/.exec(key); if (matchResult) { this._records.add(matchResult[1]); localStorage.removeItem(matchResult[0]); } }); this.saveHistory(); } function clearHistory() { const isConfirm = confirm("Do you really want to clear history?"); if (!isConfirm) return; this.updateHistory(); this._records = new Set(); localStorage.pixivDownloader = "[]"; } function saveHistory() { localStorage.pixivDownloader = JSON.stringify([...this._records]); } const pixivHistory = { _records: getHistory(), add, has, updateHistory, saveHistory, clearHistory, }; function handleDownload(pdlBtn, illustId) { let pageCount, pageComplete = 0; const onProgress = (progress = 0, type = null) => { if (pageCount > 1) return; progress = Math.floor(progress * 100); switch (type) { case null: pdlBtn.style.setProperty("--pdl-progress", progress + "%"); case "gif": case "webm": pdlBtn.textContent = progress; break; case "zip": pdlBtn.textContent = ""; break; } }; const onLoad = function () { if (pageCount < 2) return; const progress = Math.floor((++pageComplete / pageCount) * 100); pdlBtn.textContent = progress; pdlBtn.style.setProperty("--pdl-progress", progress + "%"); }; pdlBtn.classList.add("pdl-progress"); parser .id(illustId) .then((metas) => { let shouldDownloadPage; if ((shouldDownloadPage = pdlBtn.getAttribute("should-download"))) { metas = [metas[shouldDownloadPage]]; } pageCount = metas.length; metas.forEach((meta) => { meta.onProgress = onProgress; meta.onLoad = onLoad; }); return downloader.add(metas); }) .then(() => { pixivHistory.add(illustId); localStorage.setItem(`pdlTemp-${illustId}`, ""); pdlBtn.classList.remove("pdl-error"); pdlBtn.classList.add("pdl-complete"); }) .catch((err) => { if (err) console.log(err); pdlBtn.classList.remove("pdl-complete"); pdlBtn.classList.add("pdl-error"); }) .finally(() => { pdlBtn.innerHTML = ""; pdlBtn.style.removeProperty("--pdl-progress"); pdlBtn.classList.remove("pdl-progress"); }); } function handleDownloadAll(userId, type = "") { let worksCount = 0, worksComplete = 0, failed = []; let isCanceled = false; let metasRecord = []; const timers = []; const placeholder = document.querySelector(".pdl-nav-placeholder"); const control = document.querySelector(".pdl-stop"); const isExcludeDled = document.querySelector("#pdl-filter").checked; return new Promise((resolve, reject) => { control.onclick = () => { isCanceled = true; for (const timer of timers) { if (timer) clearTimeout(timer); } if (metasRecord.length) { downloader.del(metasRecord); converter.del(metasRecord); metasRecord = []; } control.onclick = null; reject("Download stopped"); }; const onProgressCB = (illustId) => { placeholder.textContent = `Downloading: ${++worksComplete} / ${worksCount}`; if (worksComplete === worksCount - failed.length) { if (failed.length) { placeholder.textContent = `Complete. Failed: ${failed.length} / ${worksCount}. See console.`; console.log("Failed: ", failed.join(", ")); } else { placeholder.textContent = "Complete"; } resolve(); } }; placeholder.textContent = "Download..."; parser .user(userId, type) .then((illustIds) => { if (isCanceled) throw "Download stopped"; if (isExcludeDled) { pixivHistory.updateHistory(); debugLog("Before filter", illustIds.length); illustIds = illustIds.filter((illustId) => !pixivHistory.has(illustId)); debugLog("After filter", illustIds.length); } if (!illustIds.length) throw "All Exclude"; worksCount = illustIds.length; illustIds.forEach((illustId, idx) => { if (isCanceled) throw "[Warning]Download stopped"; let timer = setTimeout(() => { timer = null; parser .id(illustId) .then((metas) => { if (isCanceled) { throw "[Warning]Download stop manually: " + metas[0].id; } metasRecord = metasRecord.concat(metas); return downloader.add(metas); }) .then((metas) => { pixivHistory.add(illustId); localStorage.setItem(`pdlTemp-${illustId}`, ""); if (isCanceled) { return; } metasRecord = metasRecord.filter((meta) => !metas.includes(meta)); onProgressCB(); }) .catch((err) => { failed.push(illustId); }); }, idx * 600); timers.push(timer); }); }) .catch((err) => { reject(err); }); }); } function toggleDlAll(evt) { const target = evt.target; if (target.classList.contains("pdl-btn-all")) { evt.preventDefault(); evt.stopPropagation(); const dlBarsBtn = target.parentElement.querySelectorAll("[pdl-userid]"); const placeholder = document.querySelector(".pdl-nav-placeholder"); const userId = target.getAttribute("pdl-userid"); dlBarsBtn.forEach((ele) => { ele.classList.toggle("pdl-hide"); }); handleDownloadAll(userId, target.getAttribute("pdl-type")) .catch((err) => { placeholder.textContent = err; }) .finally(() => { dlBarsBtn.forEach((ele) => { ele.classList.toggle("pdl-hide"); }); }); } } function createModal({ header, content, footer = "" }, option = { closeOnClickModal: true }) { const modal = document.createElement("div"); const dialog = document.createElement("div"); modal.classList.add("pdl-modal"); dialog.classList.add("pdl-dialog"); if (option.closeOnClickModal) { dialog.onclick = (e) => { e.stopPropagation(); }; modal.onclick = () => { modal.remove(); }; } dialog.innerHTML = ` <header class="pdl-dialog-header">${header}</header> <div class="pdl-dialog-content">${content}</div> <footer class="pdl-dialog-footer">${footer}</footer>`; const closeBtn = document.createElement("button"); closeBtn.classList.add("pdl-dialog-close"); closeBtn.onclick = () => { modal.remove(); }; dialog.insertBefore(closeBtn, dialog.firstChild); modal.appendChild(dialog); return modal; } function showUpgradeMsg() { document.body.appendChild( createModal({ header: text.upgradeMsgTitle, content: text.upgradeMsgContent, footer: text.modalCreditFooter + text.modalFeedback, }) ); } function showFilePathSetting() { if (document.querySelector("#pdlfolder")) return; const modal = createModal( { header: text.filePathSettingTitle, content: text.filePathSettingContent, footer: text.modalOperationBar, }, { closeOnClickModal: false } ); const folder = modal.querySelector("#pdlfolder"); const filename = modal.querySelector("#pdlfilename"); modal.querySelector("#pdlcancel").onclick = () => { modal.remove(); }; modal.querySelector("#pdlconfirm").onclick = () => { if (filename.value === "") return; const folderPattern = folder.value .trim() .replace(/^\.|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, ""); const filenamePattern = filename.value .trim() .replace(/^\.|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, ""); if (filenamePattern === "") return; upgradeSettings("folderPattern", folderPattern); upgradeSettings("filenamePattern", filenamePattern); modal.remove(); }; folder.value = settings.folderPattern; filename.value = settings.filenamePattern; document.body.appendChild(modal); } function getIllustId(node) { const isLinkToArtworksPage = regexp.artworksPage.exec(node.href); if (isLinkToArtworksPage) { if ( node.getAttribute("data-gtm-value") || node.classList.contains("gtm-illust-recommend-node-node") || node.classList.contains("gtm-discover-user-recommend-node") || node.classList.contains("work") ) { return isLinkToArtworksPage[1]; } } else { const isActivityThumb = regexp.activityHref.exec(node.href); if (isActivityThumb && node.classList.contains("work")) { return isActivityThumb[1]; } } return ""; } function createPdlBtn(attributes, textContent = "", { addEvent } = { addEvent: true }) { const ele = document.createElement("button"); ele.textContent = textContent; if (!attributes) return ele; const { attrs, classList } = attributes; if (classList && classList.length > 0) { for (const cla of classList) { ele.classList.add(cla); } } if (attrs) { for (const key in attrs) { ele.setAttribute(key, attrs[key]); } } if (addEvent) { ele.addEventListener("click", (evt) => { evt.preventDefault(); evt.stopPropagation(); const ele = evt.currentTarget; if (!evt.currentTarget.classList.contains("pdl-progress")) { handleDownload(ele, ele.getAttribute("pdl-id")); } }); } return ele; } function createMainBtn(id) { if (document.querySelector(".pdl-btn-main")) return; const handleBar = document.querySelector("main section section"); if (handleBar) { const pdlBtnWrap = handleBar.lastElementChild.cloneNode(); const attrs = { attrs: { "pdl-id": id }, classList: ["pdl-btn", "pdl-btn-main"], }; if (pixivHistory.has(id)) attrs.classList.push("pdl-complete"); pdlBtnWrap.appendChild(createPdlBtn(attrs)); handleBar.appendChild(pdlBtnWrap); } } function createDownloadBar(userId) { const nav = document.querySelector("nav"); if (!nav || document.querySelector(".pdl-nav-placeholder")) return; const fragment = document.createDocumentFragment(); const placeholder = document.createElement("div"); placeholder.classList.add("pdl-nav-placeholder"); fragment.appendChild(placeholder); const baseClasses = nav.querySelector("a:not([aria-current])").classList; fragment.appendChild( createPdlBtn( { attrs: { "pdl-userId": userId }, classList: [...baseClasses, "pdl-stop", "pdl-hide"], }, "Stop", { addEvent: false } ) ); fragment.appendChild( createPdlBtn( { attrs: { "pdl-userId": userId }, classList: [...baseClasses, "pdl-btn-all"], }, "All", { addEvent: false } ) ); if (nav.querySelector("a[href$=illustrations]") && nav.querySelector("a[href$=manga]")) { fragment.appendChild( createPdlBtn( { attrs: { "pdl-userid": userId, "pdl-type": "illusts" }, classList: [...baseClasses, "pdl-btn-all"], }, "Illusts", { addEvent: false } ) ); fragment.appendChild( createPdlBtn( { attrs: { "pdl-userid": userId, "pdl-type": "manga" }, classList: [...baseClasses, "pdl-btn-all"], }, "Manga", { addEvent: false } ) ); } const wrapper = document.createElement("div"); const checkbox = document.createElement("input"); const label = document.createElement("label"); wrapper.classList.add("pdl-wrap"); checkbox.id = "pdl-filter"; checkbox.type = "checkbox"; label.setAttribute("for", "pdl-filter"); label.textContent = "Exclude downloaded"; wrapper.appendChild(checkbox); wrapper.appendChild(label); nav.parentElement.insertBefore(wrapper, nav); nav.appendChild(fragment); nav.addEventListener("click", toggleDlAll); } function createSubBtn(nodes) { const isBookmarkPage = regexp.bookmarkPage.test(location.pathname); nodes.forEach((e) => { if (e.childElementCount !== 0) { const illustId = getIllustId(e); if (illustId) { const attrs = { attrs: { "pdl-id": illustId }, classList: ["pdl-btn", "pdl-btn-sub"], }; if (pixivHistory.has(illustId)) attrs.classList.push("pdl-complete"); if (isBookmarkPage) attrs.classList.push("pdl-btn-sub-bookmark"); e.appendChild(createPdlBtn(attrs)); } } }); } function createMultyWorksBtn(id) { const works = document.querySelectorAll("[role='presentation'] > a"); if (works.length < 2) return; const containers = Array.from(works).map((node) => node.parentElement.parentElement); if (containers[0].querySelector(".pdl-btn")) return; containers.forEach((node, idx) => { const wrapper = document.createElement("div"); wrapper.classList.add("pdl-wrap-artworks"); const attrs = { attrs: { "pdl-id": id, "should-download": idx }, classList: ["pdl-btn", "pdl-btn-sub", "artworks"], }; wrapper.appendChild(createPdlBtn(attrs)); node.appendChild(wrapper); }); } const createPresentationBtn = (() => { let observer, btn; function cb(mutationList) { const newImg = mutationList[1]["addedNodes"][0]; const [pageNum] = regexp.originSrcPageNum.exec(newImg.src); const containers = btn.parentElement; const attrs = { attrs: { "pdl-id": btn.getAttribute("pdl-id"), "should-download": pageNum, }, classList: ["pdl-btn", "pdl-btn-sub", "presentation"], }; btn.remove(); btn = createPdlBtn(attrs); containers.appendChild(btn); } return (id) => { const containers = document.querySelector("body > [role='presentation'] > div"); if (!containers) { if (observer) { observer.disconnect(); observer = null; btn = null; } return; } if (containers.querySelector(".pdl-btn")) return; const img = containers.querySelector("img"); const isOriginImg = regexp.originSrcPageNum.exec(img.src); if (!isOriginImg) return; const [pageNum] = isOriginImg; const attrs = { attrs: { "pdl-id": id, "should-download": pageNum }, classList: ["pdl-btn", "pdl-btn-sub", "presentation"], }; btn = createPdlBtn(attrs); containers.appendChild(btn); observer = new MutationObserver(cb); observer.observe(img.parentElement, { childList: true, subtree: true }); }; })(); function createPreviewModalBtn() { const illustModalBtn = document.querySelectorAll(".gtm-manga-viewer-preview-modal-open"); const mangaModalBtn = document.querySelectorAll(".gtm-manga-viewer-open-preview"); let mangaViewerModalBtn = document.querySelectorAll(".gtm-manga-viewer-close-icon")?.[1]; if (!illustModalBtn.length && !mangaModalBtn.length) return; const btns = [...illustModalBtn, ...mangaModalBtn]; if (mangaViewerModalBtn) btns.push(mangaViewerModalBtn); btns.forEach((node) => { node.addEventListener("click", handleModalClick); }); } function handleModalClick() { const timer = setInterval(() => { const ulList = document.querySelectorAll("ul"); const previewList = ulList[ulList.length - 1]; if (getComputedStyle(previewList).display !== "grid") return; clearInterval(timer); const [, id] = regexp.artworksPage.exec(location.pathname); previewList.childNodes.forEach((node, idx) => { node.style.position = "relative"; const attrs = { attrs: { "pdl-id": id, "should-download": idx }, classList: ["pdl-btn", "pdl-btn-sub"], }; node.appendChild(createPdlBtn(attrs)); }); }, 300); } function compatPixivPreviewer(nodes) { const isPpSearchPage = regexp.ppSearchPage.test(location.pathname); if (!isPpSearchPage) return; nodes.forEach((node) => { const pdlEle = node.querySelector(".pdl-btn"); if (!pdlEle) return false; pdlEle.remove(); }); } let firstRun = true; function observerCallback(records) { const addedNodes = []; records.forEach((record) => { if (!record.addedNodes.length) return; record.addedNodes.forEach((node) => { if ( node.nodeType === Node.ELEMENT_NODE && node.tagName !== "BUTTON" && node.tagName !== "IMG" ) { addedNodes.push(node); } }); }); if (!addedNodes.length) { return; } if (firstRun) { createSubBtn(document.querySelectorAll("a")); firstRun = false; } else { compatPixivPreviewer(addedNodes); const thunmnails = addedNodes.reduce((prev, current) => { return prev.concat(Array.from(current.querySelectorAll("a"))); }, []); createSubBtn(thunmnails); } const isArtworksPage = regexp.artworksPage.exec(location.pathname); const isUserPage = regexp.userPage.exec(location.pathname); if (isArtworksPage) { const id = isArtworksPage[1]; createMainBtn(id); createMultyWorksBtn(id); createPresentationBtn(id); createPreviewModalBtn(); } else if (isUserPage) { createDownloadBar(isUserPage[1]); } } addStyle(); pixivHistory.updateHistory(); GM_registerMenuCommand("Apng", createSetFormatFn("png"), "a"); GM_registerMenuCommand("Gif", createSetFormatFn("gif"), "g"); GM_registerMenuCommand("Zip", createSetFormatFn("zip"), "z"); GM_registerMenuCommand("Webm", createSetFormatFn("webm"), "w"); GM_registerMenuCommand("Clear history", pixivHistory.clearHistory.bind(pixivHistory), "c"); GM_registerMenuCommand("Edit filename", showFilePathSetting, "e"); initial().then(() => { if (settings.showMsg) { showUpgradeMsg(); upgradeSettings("showMsg", false); } new MutationObserver(observerCallback).observe(document.body, { childList: true, subtree: true, }); document.addEventListener("keydown", (e) => { if (e.ctrlKey && e.key === "q") { const pdlMainBtn = document.querySelector(".pdl-btn-main"); if (pdlMainBtn) { e.preventDefault(); if (!e.repeat) { pdlMainBtn.dispatchEvent(new MouseEvent("click", { bubbles: true })); } } } }); }); })();