// ==UserScript==
// @name Pixiv Downloader
// @namespace https://greasyfork.org/zh-CN/scripts/432150
// @version 0.8.8
// @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 | Webp | 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 | Webp | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。
// @description:zh-TW 一鍵下載Pixiv各頁面原圖。支持多圖下載,動圖下載,按作品標籤下載,畫師作品批次下載。動圖支持格式轉換:Gif | Apng | Webp | 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
// @require https://greasyfork.org/scripts/455256-toanimatedwebp/code/toAnimatedWebp.js?version=1120088
// ==/UserScript==
(function (workerChunk, JSZip, GIF) {
'use strict';
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
var workerChunk__default = /*#__PURE__*/_interopDefaultLegacy(workerChunk);
var JSZip__default = /*#__PURE__*/_interopDefaultLegacy(JSZip);
var GIF__default = /*#__PURE__*/_interopDefaultLegacy(GIF);
var e=[],t=[];function n(n,r){if(n&&"undefined"!=typeof document){var a,s=!0===r.prepend?"prepend":"append",d=!0===r.singleTag,i="string"==typeof r.container?document.querySelector(r.container):document.getElementsByTagName("head")[0];if(d){var u=e.indexOf(i);-1===u&&(u=e.push(i)-1,t[u]={}),a=t[u]&&t[u][s]?t[u][s]:t[u][s]=c();}else a=c();65279===n.charCodeAt(0)&&(n=n.substring(1)),a.styleSheet?a.styleSheet.cssText+=n:a.appendChild(document.createTextNode(n));}function c(){var e=document.createElement("style");if(e.setAttribute("type","text/css"),r.attributes)for(var t=Object.keys(r.attributes),n=0;n<t.length;n++)e.setAttribute(t[n],r.attributes[t[n]]);var a="prepend"===s?"afterbegin":"beforeend";return i.insertAdjacentElement(a,e),e}}
var css$3 = "@property --pdl-progress {\n syntax: '<percentage>';\n inherits: true;\n initial-value: 0%;\n}\n@keyframes pdl_loading {\n 100% {\n transform: translate(-50%, -50%) rotate(360deg);\n }\n}\n.pdl-btn {\n position: relative;\n border-top-right-radius: 8px;\n background: no-repeat center/85%;\n 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\");\n color: #01b468;\n display: inline-block;\n font-size: 13px;\n font-weight: bold;\n height: 32px;\n line-height: 32px;\n margin: 0;\n overflow: hidden;\n padding: 0;\n border: none;\n text-decoration: none !important;\n text-align: center;\n text-overflow: ellipsis;\n user-select: none;\n white-space: nowrap;\n width: 32px;\n z-index: 1;\n cursor: pointer;\n}\n.pdl-btn-main {\n margin: 0 0 0 10px;\n}\n.pdl-btn-sub {\n bottom: 0;\n background-color: rgba(255, 255, 255, 0.5);\n left: 0;\n position: absolute;\n}\n.pdl-btn-sub.artworks {\n position: sticky;\n top: 40px;\n border-radius: 4px;\n}\n.pdl-btn-sub.presentation {\n position: fixed;\n top: 50px;\n right: 16px;\n border-radius: 8px;\n left: auto;\n}\n.pdl-btn-sub-bookmark.pdl-btn-sub-bookmark {\n left: auto;\n right: 0;\n bottom: 34px;\n border-radius: 8px;\n border-top-right-radius: 0px;\n border-bottom-right-radius: 0px;\n}\n.pdl-error {\n 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\") !important;\n}\n.pdl-complete {\n 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\") !important;\n}\n.pdl-progress {\n background-image: none !important;\n cursor: default !important;\n}\n.pdl-progress:after {\n content: '';\n display: inline-block;\n position: absolute;\n top: 50%;\n left: 50%;\n width: 27px;\n height: 27px;\n transform: translate(-50%, -50%);\n -webkit-mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);\n mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);\n border-radius: 50%;\n}\n.pdl-progress:not(:empty):after {\n background: conic-gradient(\n #01b468 0,\n #01b468 var(--pdl-progress),\n transparent var(--pdl-progress),\n transparent\n );\n transition: --pdl-progress 0.2s ease;\n}\n.pdl-progress:empty:after {\n background: conic-gradient(#01b468 0, #01b468 25%, #01b46833 25%, #01b46833);\n animation: 1.5s infinite linear pdl_loading;\n}\n\n.pdl-btn.pdl-tag {\n height: auto;\n border-top-right-radius: 4px;\n border-bottom-right-radius: 4px;\n left: -1px;\n background-color: rgba(0, 0, 0, 0.12);\n transition: background-image 0.5s;\n}\n\n.pdl-btn.pdl-tag.pdl-tag-hide,\n.pdl-btn.pdl-modal-tag.pdl-tag-hide {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3C/svg%3E\");\n pointer-events: none;\n}\n\n.pdl-btn.pdl-modal-tag {\n position: absolute;\n right: 65px;\n top: 6px;\n background-origin: content-box;\n border-radius: 4px;\n padding: 5px;\n width: 42px;\n height: 50px;\n background-color: rgba(0, 0, 0, 0.04);\n transition: 0.25s background-color;\n}\n\n.pdl-btn.pdl-modal-tag:not(.pdl-tag-hide):hover {\n background-color: rgba(0, 0, 0, 0.12);\n}\n\n.pdl-wrap-artworks {\n position: absolute;\n right: 8px;\n top: 0px;\n bottom: 0px;\n margin-top: 40px;\n}\n";
n(css$3,{});
var css$2 = ".pdl-modal * {\n font-family: 'win-bug-omega, system-ui, -apple-system, \"Segoe UI\", Roboto, Ubuntu, Cantarell, \"Noto Sans\", \"Hiragino Kaku Gothic ProN\", Meiryo, sans-serif';\n line-height: 1.15;\n}\n\n.pdl-modal {\n position: fixed;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n display: flex;\n z-index: 99;\n background-color: rgba(0, 0, 0, 0.32);\n user-select: none;\n}\n\n.pdl-dialog {\n position: relative;\n background-color: #fff;\n border-radius: 24px;\n margin: auto;\n padding: 20px 40px 30px 40px;\n width: 600px;\n font-size: 16px;\n}\n\n.pdl-dialog-header > h3 {\n font-weight: bold;\n font-size: 1.17em;\n margin: 1em 0;\n}\n\n.pdl-dialog-close {\n position: absolute;\n top: 10px;\n right: 10px;\n margin: 0;\n padding: 0;\n width: 25px;\n height: 25px;\n border: none;\n cursor: pointer;\n border-radius: 50%;\n background-color: transparent;\n transform: rotate(45deg);\n transition: 0.25s background-color;\n background: linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%)\n center/18px 2px no-repeat,\n linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/2px\n 18px no-repeat;\n}\n\n.pdl-dialog-close:hover {\n background-color: rgba(0, 0, 0, 0.05);\n}\n\n.pdl-dialog-content hr {\n height: 0 !important;\n margin: 0;\n border: none;\n border-top: 1px solid rgba(0, 0, 0, 0.1);\n}\n\n.pdl-dialog-content hr.sub {\n margin-inline-start: 1.5em;\n}\n\n/* button */\n.pdl-dialog-button {\n font-size: 16px;\n background-color: #fff;\n border: 1px solid rgb(125, 125, 125);\n border-radius: 5px;\n padding: 0.5em 1.5em;\n cursor: pointer;\n transition: 0.2s opacity;\n line-height: 1.15;\n}\n.pdl-dialog-button:hover {\n opacity: 0.7;\n}\n\n.pdl-dialog-button.primary {\n color: #fff;\n background-color: #0096fa;\n border-color: #0096fa;\n}\n\n.pdl-dialog-button.icon {\n padding: 0.5em 0.8em;\n}\n\n.pdl-dialog-button[disabled] {\n cursor: not-allowed !important;\n color: #c0c4cc;\n background-color: #fff;\n border-color: #e4e7ed;\n opacity: 1 !important;\n}\n\n.pdl-dialog-button[disabled]:hover {\n opacity: 1 !important;\n}\n\n.pdl-dialog-button.primary[disabled] {\n color: #fff;\n background-color: #a0cfff;\n border-color: #a0cfff;\n}\n\n/* tabs */\n.pdl-tabs-nav {\n display: flex;\n align-items: center;\n border-bottom: 1px solid #dcdfe6;\n position: relative;\n}\n\n.pdl-tabs-nav .pdl-tab-item {\n padding: 0px 16px;\n line-height: 2.5;\n cursor: pointer;\n transition: 0.2s color;\n}\n\n.pdl-tabs-nav .pdl-tab-item:nth-child(2) {\n padding-left: 0px;\n}\n\n.pdl-tabs-nav .pdl-tab-item:last-child {\n padding-right: 0px;\n}\n\n.pdl-tab-item:hover,\n.pdl-tab-item.active {\n color: #0096fa;\n}\n\n.pdl-tab-item.active {\n font-weight: 700;\n}\n\n.pdl-tabs-content {\n padding: 16px;\n min-height: 200px;\n}\n\n.pdl-tabs-nav .pdl-tabs__active-bar {\n position: absolute;\n bottom: 0;\n left: 0;\n height: 2px;\n background-color: #0096fa;\n z-index: 1;\n transition: width 0.2s, transform 0.2s;\n}\n\n/* filename */\n\n#pdl-setting-filename a {\n color: #0096fa;\n text-decoration: underline;\n}\n#pdl-setting-filename .tags-option label,\n#pdl-setting-filename .tags-option input {\n cursor: pointer;\n}\n\n#pdl-setting-filename .tags-content {\n flex: 1;\n display: flex;\n gap: 20px;\n}\n\n#pdl-setting-filename .pdl-input-wrap,\n#pdl-setting-filename .tags-option {\n display: flex;\n align-items: center;\n margin: 12px 0;\n gap: 12px;\n}\n\n#pdl-setting-filename .pdl-input-wrap label,\n#pdl-setting-filename .tags-option .tags-title {\n cursor: default;\n font-weight: 700;\n width: 6em;\n}\n\n#pdl-setting-filename .pdl-input-wrap input[type='text'] {\n height: auto;\n padding: 0.5em;\n font-size: 16px;\n line-height: 1.5;\n flex: 1;\n border: 1px solid #333;\n}\n\n#pdl-setting-filename .pdl-input-wrap input[type='text']:focus {\n background-color: #fff !important;\n}\n\n#pdl-setting-filename .pdl-input-wrap button {\n line-height: 1.5;\n}\n\n/* ugoria */\n#pdl-setting-ugoria p.option-header {\n font-weight: 700;\n}\n\n#pdl-setting-ugoria #pdl-ugoria-format-wrap {\n display: flex;\n justify-content: space-between;\n flex-wrap: nowrap;\n margin: 1.5em 1em;\n}\n\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item,\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item label,\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item input {\n cursor: pointer;\n}\n\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item label {\n padding-left: 4px;\n}\n\n/* history */\n#pdl-setting-history div {\n text-align: center;\n margin: 1em 0;\n}\n\n#pdl-setting-history .btn-history {\n width: 80%;\n}\n\n/* others setting */\n#pdl-setting-others .pdl-options {\n display: flex;\n align-items: center;\n gap: 20px;\n cursor: pointer;\n transition: 0.2s background-color;\n padding: 1em 0.5em;\n border-radius: 4px;\n}\n\n#pdl-setting-others .pdl-options:hover {\n background-color: rgba(0, 0, 0, 0.05);\n}\n\n#pdl-setting-others .pdl-options.sub-option {\n padding: 0.5em;\n padding-inline-start: 2em;\n}\n\n/* donate */\n#pdl-setting-donate {\n text-align: center;\n}\n\n#pdl-setting-donate p {\n margin: 0.5em 0;\n}\n\n/* upgrade msg */\n.pdl-changelog h4 {\n margin: 1em 0;\n font-weight: bold !important;\n}\n\n.pdl-changelog ul {\n padding-inline-start: 40px !important;\n}\n\n.pdl-changelog li {\n line-height: 2;\n list-style-type: disc !important;\n}\n\n.pdl-changelog p {\n margin: 0.5em 0;\n}\n";
n(css$2,{});
var css$1 = ".pdl-dlbar-status_bar {\n flex-grow: 1;\n height: 46px;\n line-height: 46px;\n padding-right: 8px;\n text-align: right;\n font-weight: bold;\n font-size: 16px;\n color: rgb(133, 133, 133);\n cursor: default;\n white-space: nowrap;\n}\n\n.pdl-btn-all.pdl-btn-all,\n.pdl-stop.pdl-stop {\n background-color: transparent;\n border: none;\n padding: 0 10px;\n}\n\n.pdl-btn-all.pdl-btn-all:hover,\n.pdl-stop.pdl-stop:hover {\n color: rgb(31, 31, 31);\n}\n\n.pdl-btn-all::before,\n.pdl-stop::before {\n content: '';\n height: 24px;\n width: 24px;\n transition: background-image 0.2s ease 0s;\n background: no-repeat center/85%;\n}\n\n.pdl-btn-all::before {\n 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\");\n}\n\n.pdl-stop::before {\n 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\");\n}\n\n.pdl-btn-all:hover::before {\n 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\");\n}\n\n.pdl-stop:hover::before {\n 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\");\n}\n\n.pdl-hide {\n display: none !important;\n}\n\n.pdl-filter-wrap {\n display: flex;\n justify-content: flex-end;\n gap: 12px;\n margin: 4px 0;\n font-weight: bold;\n font-size: 14px;\n line-height: 14px;\n color: rgb(133, 133, 133);\n transition: color 0.2s ease 0s;\n}\n\n.pdl-filter-wrap.unavailable {\n pointer-events: none !important;\n opacity: 0.5 !important;\n}\n\n.pdl-filter-wrap .pdl-filter:hover {\n color: rgb(31, 31, 31);\n}\n\n.pdl-filter-wrap label {\n padding-left: 8px;\n cursor: pointer;\n}\n\n.pdl-checkbox.pdl-checkbox {\n vertical-align: top;\n appearance: none;\n position: relative;\n box-sizing: border-box;\n width: 28px;\n border: 2px solid transparent;\n cursor: pointer;\n border-radius: 14px;\n height: 14px;\n background-color: rgba(133, 133, 133);\n transition: background-color 0.2s ease 0s, box-shadow 0.2s ease 0s;\n}\n\n.pdl-checkbox:hover {\n background-color: rgba(31, 31, 31);\n}\n\n.pdl-checkbox::after {\n content: '';\n position: absolute;\n display: block;\n top: 0px;\n left: 0px;\n width: 10px;\n height: 10px;\n transform: translateX(0px);\n background-color: rgb(255, 255, 255);\n border-radius: 10px;\n transition: transform 0.2s ease 0s;\n}\n\n.pdl-checkbox:checked {\n background-color: rgb(0, 150, 250);\n}\n\n.pdl-checkbox:checked::after {\n transform: translateX(14px);\n}\n\n.pdl-dlbar {\n display: flex;\n flex-grow: 1;\n}\n\n.pdl-dlbar-follow_latest {\n padding: 0 8px;\n}\n";
n(css$1,{});
var css = "[data-theme='dark'] .pdl-btn-all::before {\n 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\");\n}\n\n[data-theme='dark'] .pdl-btn-main,\n[data-theme='dark'] .pdl-btn-all:hover::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23D6D6D6' 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\");\n}\n\n[data-theme='dark'] .pdl-stop::before {\n 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\");\n}\n\n[data-theme='dark'] .pdl-stop:hover::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23D6D6D6' 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\");\n}\n\n[data-theme='dark'] .pdl-checkbox:not(:checked):hover {\n background-color: rgba(155, 155, 155);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-tag {\n background-color: rgba(255, 255, 255, 0.4);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-modal-tag {\n background-color: rgba(255, 255, 255, 0.4);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-modal-tag:hover {\n background-color: rgba(255, 255, 255, 0.6);\n}\n\n[data-theme='dark'] .pdl-wrap .pdl-filter:hover,\n[data-theme='dark'] .pdl-stop.pdl-stop:hover,\n[data-theme='dark'] .pdl-btn-all.pdl-btn-all:hover {\n color: rgb(214, 214, 214);\n}\n\n/* modal */\n[data-theme='dark'] .pdl-dialog {\n background-color: rgb(31, 31, 31);\n}\n\n[data-theme='dark'] .pdl-dialog-footer button {\n background-color: rgb(245, 245, 245);\n}\n\n[data-theme='dark'] .pdl-dialog-content hr {\n border-top: 1px solid rgba(255, 255, 255, 0.3);\n}\n\n/* others setting */\n[data-theme='dark'] #pdl-setting-others .pdl-options:hover {\n background-color: rgba(255, 255, 255, 0.1);\n}\n";
n(css,{});
var IllustType;
(function (IllustType) {
IllustType[IllustType["illusts"] = 0] = "illusts";
IllustType[IllustType["manga"] = 1] = "manga";
IllustType[IllustType["ugoira"] = 2] = "ugoira";
})(IllustType || (IllustType = {}));
var BookmarkRestrict;
(function (BookmarkRestrict) {
BookmarkRestrict[BookmarkRestrict["public"] = 0] = "public";
BookmarkRestrict[BookmarkRestrict["private"] = 1] = "private";
})(BookmarkRestrict || (BookmarkRestrict = {}));
const defaultSettings = Object.freeze({
version: '0.8.8',
ugoriaFormat: 'zip',
folderPattern: 'pixiv/{artist}',
filenamePattern: '{artist}_{title}_{id}_p{page}',
tagLang: 'ja',
showMsg: true,
log: false,
filter: {
exclude_downloaded: false,
[IllustType.illusts]: true,
[IllustType.manga]: true,
[IllustType.ugoira]: true
},
bundleIllusts: false,
bundleManga: false,
addBookmark: false,
addBookmarkWithTags: false,
privateR18: false
});
const regexp = {
preloadData: /"meta-preload-data" content='(.*?)'>/,
globalData: /"meta-global-data" content='(.*?)'>/,
artworksPage: /artworks\/(\d+)$/,
userPage: /users\/(\d+)/,
bookmarkPage: /users\/\d+\/bookmarks\/artworks/,
userPageTags: /users\/\d+\/(artworks|illustrations|manga|bookmarks(?!artworks))/,
ppSearchPage: /\/tags\/.*\/(artworks|illustrations|manga)/,
suscribePage: /bookmark_new_illust/,
activityHref: /illust_id=(\d+)/,
originSrcPageNum: /(?<=_p)\d+/,
followLatest: /\/bookmark_new_illust(?:_r18)?\.php/
};
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 creditCode = `<img style="display: block; margin: 1em auto; width: 200px"
src=""
/>`;
const lang = document.documentElement.getAttribute('lang')?.toLowerCase() ||
'en';
const i18nLib = {
en: {
bookmarks: 'Bookmarks',
bookmarks_public: 'Public',
bookmarks_private: 'Private',
stop: 'Stop',
dlbar_category_works: 'Works',
dlbar_filter_illusts: 'Illustrations',
dlbar_filter_manga: 'Manga',
dlbar_filter_ugoria: 'Ugoria',
dlbar_filter_exclude_downloaded: 'Exclude downloaded',
'dlbar.follow_latest.category_all.single': 'All (page)',
'dlbar.follow_latest.category_all.all': 'All (batch)',
'dlbar.follow_latest.category_r18.single': 'R-18 (page)',
'dlbar.follow_latest.category_r18.all': 'R-18 (batch)',
feedback: 'Feedback',
gm_menu_setting: 'Settings',
tabs_nav_filename: 'Filename',
tags_lang: 'Tags language: ',
tags_tips: '{artist}, {artistID}, {title}, {id}, {page}, {tags}',
tags_tips2: 'Note: Tags language may not be the language you selected, <a href="https://crowdin.com/project/pixiv-tags" target="_blank">some tags without translations</a> may still be in other languages.',
folder: 'Folder:',
folder_tips: "I don't need subfolder",
folder_tips2: "If you don't need a subfolder, just leave the folder name blank",
folder_vm_tips: "VM doesn't support",
folder_api_tips: 'Need Browser Api',
filename: 'FileName:',
filename_tips: 'Your Name?',
tabs_nav_history: 'History',
'modal_history_content.import_btn': 'Import',
'modal_history_content.export_btn': 'Export',
'modal_history_content.clear_btn': 'Clear',
clear_history_tips: 'Do you really want to clear history?',
tabs_nav_ugoria: 'Ugoria',
modal_ugoria_format_title: 'Ugoria Format:',
tabs_nav_others: 'Others',
'modal_others_content.bundle_illusts': 'Pack multi-page illustrations into a .zip archive',
'modal_others_content.bundle_manga': 'Pack manga into a .zip archive',
'modal_others_content.add_bookmark': 'Bookmark work when downloading a single work',
'modal_others_content.add_bookmark_tags': 'Add works tags',
'modal_others_content.add_bookmark_private_r18': 'Bookmark R-18 works to private category',
tabs_nav_donate: 'Feedback / Donate'
},
'zh-cn': {
bookmarks: '收藏',
bookmarks_public: '公开',
bookmarks_private: '不公开',
stop: '停止',
dlbar_category_works: '作品',
dlbar_filter_illusts: '插画',
dlbar_filter_manga: '漫画',
dlbar_filter_ugoria: '动图',
dlbar_filter_exclude_downloaded: '排除已下载图片',
'dlbar.follow_latest.category_all.single': '全部(单页)',
'dlbar.follow_latest.category_all.all': '全部(批量)',
'dlbar.follow_latest.category_r18.single': 'R-18(单页)',
'dlbar.follow_latest.category_r18.all': 'R-18(批量)',
feedback: '有问题or想建议?这里反馈',
gm_menu_setting: '设置',
tabs_nav_filename: '文件名',
tags_lang: '标签语言:',
tags_tips: '{artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}:作品pixiv ID, {page}:页码, {tags}:作品标签。',
tags_tips2: '请注意:标签翻译不一定是你选择的语言,部分<a href="https://crowdin.com/project/pixiv-tags" target="_blank">无对应语言翻译的标签</a>仍可能是其他语言。',
folder: '文件夹名:',
folder_tips: '我不想保存到子文件夹',
folder_tips2: '如果不想保存到画师目录,文件夹名留空即可。',
folder_vm_tips: 'Violentmonkey不支持',
folder_api_tips: '需要Browser Api',
filename: '文件名:',
filename_tips: '你的名字?',
tabs_nav_history: '历史记录',
'modal_history_content.import_btn': '导入记录',
'modal_history_content.export_btn': '导出记录',
'modal_history_content.clear_btn': '清除记录',
clear_history_tips: '真的要清除历史记录吗?',
tabs_nav_ugoria: '动图',
modal_ugoria_format_title: '动图格式:',
tabs_nav_others: '其它',
'modal_others_content.bundle_illusts': '将多页插图打包为.zip压缩包',
'modal_others_content.bundle_manga': '将多页漫画作品打包为.zip压缩包',
'modal_others_content.add_bookmark': '下载单个作品时收藏作品',
'modal_others_content.add_bookmark_tags': '收藏时添加作品标签',
'modal_others_content.add_bookmark_private_r18': '将R-18作品收藏到不公开类别',
tabs_nav_donate: '反馈 / 赞赏'
},
zh: {},
ja: {},
ko: {},
'zh-tw': {}
};
i18nLib.en = Object.create(i18nLib['zh-cn'], Object.getOwnPropertyDescriptors(i18nLib.en));
i18nLib.ja = Object.create(i18nLib.en);
i18nLib.ko = Object.create(i18nLib.en);
i18nLib['zh-tw'] = Object.create(i18nLib['zh-cn']);
i18nLib.zh = i18nLib['zh-cn'];
const i18n = (key) => i18nLib[lang]?.[key] || `i18n[${lang}][${key}] not found`;
const modalHtml = {
upgradeMsgTitle: `<h3>Pixiv Downloader ${defaultSettings.version}</h3>`,
upgradeMsgContent: ` <div class="pdl-changelog">
<p>可乐涨价,我的暑假结束了...</p>
<h4>新增</h4>
<ul>
<li>现在可以批量下载“已关注用户的新作”。</li>
<li>——可以选择下载单页或全部(最多34页)</li>
<li>——点击“全部”或“R-18”按钮切换需要下载的作品类别</li>
</ul>
<h4>其它</h4>
<ul>
<li>稍微提高了webp转换速度</li>
<li>一些其它调整</li>
</ul>
</div>`,
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: #0096fa; text-decoration: underline">脚本还行?请我喝杯可乐吧!</summary>
${creditCode}
<p style="text-align: center">愿你每天都能找到对的色图,就像我每天都能喝到香草味可乐</p>
</details>`,
modalFeedback: `<a target="_blank" style="position: absolute; right: 0px; top: 0px; color: #0096fa; text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback">${i18n('feedback')}</a>`,
modalSettingsContent: ` <div>
<div class="pdl-tabs-nav">
<div class="pdl-tabs__active-bar"></div>
</div>
<div class="pdl-tabs-content">
</div>
</div>`,
modalSettingFilename: ` <div id="pdl-setting-filename">
<div>
<div class="pdl-input-wrap">
<label for="pdlfolder">${i18n('folder')}</label>
<input type="text" id="pdlfolder" maxlength="100" />
<button id="pdl-filename-folder-reset" class="pdl-dialog-button icon" disabled>↺</button>
<button id="pdl-filename-folder-confirm" class="pdl-dialog-button icon primary" disabled>✓</button>
</div>
<div class="pdl-input-wrap">
<label for="pdlfilename">${i18n('filename')}</label>
<input
type="text"
id="pdlfilename"
placeholder="${i18n('filename_tips')}"
required
maxlength="100"
/>
<button id="pdl-filename-filename-reset" class="pdl-dialog-button icon" disabled>↺</button>
<button id="pdl-filename-filename-confirm" class="pdl-dialog-button icon primary" disabled>✓</button>
</div>
</div>
<div class="tags-option">
<span class="tags-title">${i18n('tags_lang')}</span>
<div class="tags-content">
<div class="tags-item">
<input type="radio" name="lang" id="lang_ja" value="ja" />
<label for="lang_ja">日本語(default)</label>
</div>
<div class="tags-item">
<input type="radio" name="lang" id="lang_zh" value="zh" />
<label for="lang_zh">简中</label>
</div>
<div class="tags-item">
<input type="radio" name="lang" id="lang_zh_tw" value="zh_tw" />
<label for="lang_zh_tw">繁中</label>
</div>
<div class="tags-item">
<input type="radio" name="lang" id="lang_en" value="en" />
<label for="lang_en">English</label>
</div>
</div>
</div>
<p style="font-size: 14px; margin: 0.5em 0">
${i18n('tags_tips')}
</p>
<p style="font-size: 14px; margin: 0.5em 0">
${i18n('folder_tips2')}
</p>
<p style="font-size: 14px; margin: 0.5em 0">
${i18n('tags_tips2')}
</p>
</div>`,
modalSettingUgoria: ` <div id="pdl-setting-ugoria">
<p class="option-header">${i18n('modal_ugoria_format_title')}</p>
<div id="pdl-ugoria-format-wrap">
<div class="pdl-ugoria-format-item">
<input
type="radio"
id="pdl-ugoria-zip"
value="zip"
name="format"
/><label for="pdl-ugoria-zip">Zip</label>
</div>
<div class="pdl-ugoria-format-item">
<input
type="radio"
id="pdl-ugoria-gif"
value="gif"
name="format"
/><label for="pdl-ugoria-gif">Gif</label>
</div>
<div class="pdl-ugoria-format-item">
<input
type="radio"
id="pdl-ugoria-apng"
value="png"
name="format"
/><label for="pdl-ugoria-apng">Png</label>
</div>
<div class="pdl-ugoria-format-item">
<input
type="radio"
id="pdl-ugoria-webm"
value="webm"
name="format"
/><label for="pdl-ugoria-webm">Webm</label>
</div>
<div class="pdl-ugoria-format-item">
<input
type="radio"
id="pdl-ugoria-webp"
value="webp"
name="format"
/><label for="pdl-ugoria-webp">Webp</label>
</div>
</div>
</div>`,
modalSettingHistory: ` <div id="pdl-setting-history">
<div>
<button id="pdl-export" class="btn-history pdl-dialog-button primary">
${i18n('modal_history_content.export_btn')}
</button>
</div>
<div>
<input
type="file"
id="pdl-import"
accept=".txt"
style="display: none"
/><button id="pdl-import-btn" class="btn-history pdl-dialog-button primary">
${i18n('modal_history_content.import_btn')}
</button>
</div>
<div>
<button id="pdl-clear-history" class="btn-history pdl-dialog-button primary">
${i18n('modal_history_content.clear_btn')}
</button>
</div>
</div>`,
modalSettingOthers: ` <div id="pdl-setting-others">
<div>
<label class="pdl-options"
><input
id="pdl-options-bundle-illusts"
type="checkbox"
class="pdl-checkbox"
/><span
>${i18n('modal_others_content.bundle_illusts')}</span
></label
>
</div>
<hr />
<div>
<label class="pdl-options"
><input
id="pdl-options-bundle-manga"
type="checkbox"
class="pdl-checkbox"
/><span
>${i18n('modal_others_content.bundle_manga')}</span
></label
>
</div>
<hr />
<div>
<label class="pdl-options"
><input
id="pdl-options-add-bookmark"
type="checkbox"
class="pdl-checkbox"
/><span
>${i18n('modal_others_content.add_bookmark')}</span
></label
>
</div>
<hr />
<div>
<label class="pdl-options sub-option"
><input
id="pdl-options-add-bookmark-tags"
type="checkbox"
class="pdl-checkbox"
/><span
>${i18n('modal_others_content.add_bookmark_tags')}</span
></label
>
</div>
<hr class="sub"/>
<div>
<label class="pdl-options sub-option"
><input
id="pdl-options-add-bookmark-private-r18"
type="checkbox"
class="pdl-checkbox"
/><span
>${i18n('modal_others_content.add_bookmark_private_r18')}</span
></label
>
</div>
</div>`,
modalSettingDonate: ` <div id="pdl-setting-donate">
${creditCode}
<p>如果脚本有帮助到你,欢迎扫码请我喝杯可乐 ^_^</p>
<p><a target="_blank" style="color: #0096fa; text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback">${i18n('feedback')}</a></p>
</div>`
};
function add$1(pixivId) {
this._records.add(pixivId);
localStorage.setItem(`pdlTemp-${pixivId}`, '');
}
function has(pixivId) {
return this._records.has(pixivId);
}
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(i18n('clear_history_tips'));
if (!isConfirm)
return;
this.updateHistory();
this._records = new Set();
localStorage.pixivDownloader = '[]';
location.reload();
}
function saveHistory(historyArr) {
if (historyArr) {
localStorage.pixivDownloader = JSON.stringify(historyArr);
}
else {
localStorage.pixivDownloader = JSON.stringify([...this._records]);
}
}
function getAll() {
return [...this._records];
}
const pixivHistory = {
_records: getHistory(),
add: add$1,
has,
getAll,
updateHistory,
saveHistory,
clearHistory
};
function debugLog(...msgs) {
}
function sleep(delay) {
return new Promise((resolve) => {
setTimeout(resolve, delay);
});
}
function wakeableSleep(delay) {
let wake = () => void {};
const sleep = new Promise((r) => {
setTimeout(r, delay);
wake = r;
});
return {
wake,
sleep
};
}
function getSelfId() {
return document.querySelector('#qualtrics_user-id')
?.textContent;
}
const env = {
isViolentmonkey: GM_info.scriptHandler === 'Violentmonkey',
isBlobDlAvaliable: !(navigator.userAgent.includes('Firefox') &&
GM_info.scriptHandler === 'Tampermonkey' &&
parseFloat(GM_info.version ?? '') > 4.17),
isSupportSubpath: GM_info.downloadMode && GM_info.downloadMode === 'browser'
};
function getSettings() {
let settings;
if (!localStorage.pdlSetting) {
settings = { ...defaultSettings };
saveSettings(settings);
}
else {
settings = JSON.parse(localStorage.pdlSetting);
if (settings.version !== defaultSettings.version) {
settings.version = defaultSettings.version;
settings.showMsg = true;
settings = { ...defaultSettings, ...settings };
saveSettings(settings);
}
}
return settings;
}
function saveSettings(settingObj) {
settingObj = settingObj || settings;
localStorage.pdlSetting = JSON.stringify(settingObj);
}
function upgradeSettings(key, value) {
if (key in settings && settings[key] !== value) {
settings[key] = value;
saveSettings();
}
}
const settings = getSettings();
const handleWorker = `
let webpApi = {};
Module.onRuntimeInitialized = () => {
webpApi = {
init: Module.cwrap('init', '', ['number', 'number', 'number']),
createBuffer: Module.cwrap('createBuffer', 'number', ['number']),
addFrame: Module.cwrap('addFrame', 'number', ['number', 'number', 'number']),
generate: Module.cwrap('generate', 'number', []),
freeResult: Module.cwrap('freeResult', '', []),
getResultPointer: Module.cwrap('getResultPointer', 'number', []),
getResultSize: Module.cwrap('getResultSize', 'number', []),
};
postMessage('ok');
};
onmessage = (evt) => {
const { data, delays, lossless = 1, quality = 75, method = 4} = evt.data;
webpApi.init(lossless, quality, method);
data.forEach((u8a, idx) => {
const pointer = webpApi.createBuffer(u8a.length);
Module.HEAPU8.set(u8a, pointer);
webpApi.addFrame(pointer, u8a.length, delays[idx]);
postMessage(idx);
});
webpApi.generate();
const resultPointer = webpApi.getResultPointer();
const resultSize = webpApi.getResultSize();
const result = new Uint8Array(Module.HEAP8.buffer, resultPointer, resultSize);
postMessage(result);
webpApi.freeResult();
};`;
async function _requestJson(url, init) {
const res = await fetch(url, init);
if (!res.ok)
throw new Error('[Error]fail to fetch:' + url + ', code:' + res.status);
const data = await res.json();
if (data.error)
throw new Error('[Error]JSON return error: ' + data.message);
return data.body;
}
async function getJson(url) {
let json;
let retry = 0;
const MAX_RETRY = 3;
do {
try {
debugLog('[Info]fetch url:', url);
json = await _requestJson(url);
}
catch (error) {
retry++;
if (retry === MAX_RETRY)
throw error;
sleep(3000);
}
} while (!json);
return json;
}
async function getArtworkHtml(illustId) {
let params = '';
if (settings.tagLang !== 'ja')
params = '?lang=' + settings.tagLang;
const res = await fetch('https://www.pixiv.net/artworks/' + illustId + params);
if (!res.ok)
throw new Error(String(res.status));
return await res.text();
}
async function _getDeps(url) {
return fetch(url).then((res) => {
if (res.ok)
return res.text();
throw new Error(`[Pixiv Downloader]Fetch dependency ${url} failed: ${res.status} ${res.statusText}`);
});
}
async function getGifWS() {
let gifWS;
if (!(gifWS = await GM_getValue('gifWS'))) {
gifWS = await _getDeps(depsUrls.gifWorker);
GM_setValue('gifWS', gifWS);
}
return gifWS;
}
async function getApngWS() {
let apngWS;
if (!(apngWS = await GM_getValue('apngWS'))) {
const [pako, upng] = await Promise.all([
_getDeps(depsUrls.pako),
_getDeps(depsUrls.upng)
]);
const upngScript = 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 + upngScript;
GM_setValue('apngWS', apngWS);
}
return apngWS;
}
function getWebpWS() {
return workerChunk__default["default"] + handleWorker;
}
function addBookmark$1(illustId, token, tags, restrict) {
return _requestJson('/ajax/illusts/bookmarks/add', {
method: 'POST',
headers: {
accept: 'application/json',
'content-type': 'application/json; charset=utf-8',
'x-csrf-token': token
},
body: JSON.stringify({
illust_id: illustId,
restrict,
comment: '',
tags
})
});
}
function getFollowLatestWorks(page, mode) {
return _requestJson(`/ajax/follow_latest/illust?p=${page}&mode=${mode}&lang=jp`);
}
const api = {
getJson,
getArtworkHtml,
getGifWS,
getApngWS,
getWebpWS,
getFollowLatestWorks,
addBookmark: addBookmark$1
};
function initialDeps() {
return Promise.all([api.getGifWS(), api.getApngWS(), api.getWebpWS()]).then(([gif, apng, webp]) => {
this._deps.gif = URL.createObjectURL(new Blob([gif], { type: 'text/javascript' }));
this._deps.apng = URL.createObjectURL(new Blob([apng], { type: 'text/javascript' }));
this._deps.webp = URL.createObjectURL(new Blob([webp], { type: 'text/javascript' }));
return this;
});
}
function _createImgElements(zip) {
const eles = [];
zip.forEach((_, 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__default["default"]();
const freeApngWorkers = [];
const freeWebpWorkers = [];
const MAX_CONVERT = 2;
let queue = [];
let active = [];
let isStop = false;
const convertTo = {
webp: (frames, convertMeta) => {
return new Promise((resolve, reject) => {
let worker;
let reuse = false;
if (freeWebpWorkers.length) {
worker = freeWebpWorkers.shift();
reuse = true;
}
else {
worker = new Worker(createConverter._deps.webp);
}
convertMeta.abort = () => {
reject('[Info]Convert stop manually, reject when convert webp. ' +
convertMeta.id);
convertMeta.isAborted = true;
worker.terminate();
};
const workerLoad = new Promise((resolve) => {
if (reuse)
return resolve();
worker.onmessage = (evt) => {
if (evt.data === 'ok') {
resolve();
}
};
});
const delays = convertMeta.source.framesInfo.map((frameInfo) => {
return Number(frameInfo.delay);
});
const data = [];
let completed = 0;
frames.forEach((frame, idx) => {
const canvas = document.createElement('canvas');
const width = (canvas.width = frame.naturalWidth);
const height = (canvas.height = frame.naturalHeight);
const context = canvas.getContext('2d');
if (!context)
return;
context.drawImage(frame, 0, 0, width, height);
data.push(new Promise((resolve, reject) => {
canvas.toBlob((blob) => {
if (!blob)
return reject('[Pixiv Downloader]toBlob failed: ' + idx);
blob.arrayBuffer().then((buffer) => {
const u8a = new Uint8Array(buffer);
resolve(u8a);
convertMeta.onProgress?.((++completed / frames.length) * 0.5, 'webp');
});
}, 'image/webp', 1);
}));
});
workerLoad
.then(() => Promise.all(data))
.then((u8arrs) => {
if (convertMeta.isAborted)
return;
worker.onmessage = (evt) => {
if (typeof evt.data !== 'object') {
debugLog('[Info]Webp convert phrase 2:', convertMeta.id, evt.data);
convertMeta.onProgress?.(0.5 + (evt.data / frames.length) * 0.5, 'webp');
}
else {
freeWebpWorkers.push(worker);
resolve(new Blob([evt.data], { type: 'image/webp' }));
}
};
worker.postMessage({ data: u8arrs, delays });
})
.catch(console.error);
});
},
gif: (frames, convertMeta) => {
return new Promise((resolve, reject) => {
const gif = new GIF__default["default"]({
workers: 2,
quality: 10,
workerScript: createConverter._deps.gif
});
convertMeta.abort = () => {
gif.abort();
convertMeta.isAborted = true;
};
debugLog('[Info]Start convert:', convertMeta.id);
frames.forEach((frame, i) => {
gif.addFrame(frame, {
delay: convertMeta.source.framesInfo[i].delay
});
});
gif.on('progress', (progress) => {
debugLog('[Info]Convert progress:', convertMeta.id);
if (typeof convertMeta.onProgress === 'function')
convertMeta.onProgress(progress, 'gif');
});
gif.on('finished', (gifBlob) => {
resolve(gifBlob);
});
gif.on('abort', () => {
reject('[Info]Convert stop: abort. ' + convertMeta.id);
});
gif.render();
});
},
png: (frames, convertMeta) => {
return new Promise((resolve, reject) => {
const 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 });
if (!context)
return reject('[Error]Can not get canvas context');
const data = [];
const delay = convertMeta.source.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);
});
debugLog('[Info]Start convert:', convertMeta.id);
let worker;
if (freeApngWorkers.length) {
worker = freeApngWorkers.shift();
}
else {
worker = new Worker(createConverter._deps.apng);
}
convertMeta.abort = () => {
reject('[Info]Convert stop manually, reject when convert apng. ' +
convertMeta.id);
convertMeta.isAborted = true;
worker.terminate();
};
worker.onmessage = function (e) {
freeApngWorkers.push(worker);
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) => {
const canvas = document.createElement('canvas');
const width = (canvas.width = frames[0].naturalWidth);
const height = (canvas.height = frames[0].naturalHeight);
const context = canvas.getContext('2d');
if (!context)
return reject('[Error]Can not get canvas context');
const stream = canvas.captureStream();
const recorder = new MediaRecorder(stream, {
mimeType: 'video/webm',
videoBitsPerSecond: 80000000
});
const delay = convertMeta.source.framesInfo.map((frame) => {
return Number(frame.delay);
});
const 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 = () => {
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, source, resolve, reject } = convertMeta;
let frames;
active.push(convertMeta);
if (typeof convertMeta.onProgress === 'function')
convertMeta.onProgress(0, 'zip');
const newFolder = zip.folder(id);
if (!newFolder)
throw '[Error]Can not get new root folder';
newFolder
.loadAsync(source.data)
.then(_createImgElements)
.then((imgEles) => {
zip.remove(id);
frames = imgEles;
if (convertMeta.isAborted)
throw '[Info]Convert stop manually, reject when unzip. ' + id;
return convertTo[source.format](frames, convertMeta);
})
.then(resolve)
.catch(reject)
.finally(() => {
frames.forEach((frame) => URL.revokeObjectURL(frame.src));
active.splice(active.indexOf(convertMeta), 1);
if (queue.length)
convert(queue.shift());
});
};
return {
add: (convertSource, handler) => {
debugLog('[Info]Converter add', convertSource.id);
return new Promise((resolve, reject) => {
queue.push({
id: convertSource.id,
isAborted: false,
source: convertSource,
onProgress: handler?.onProgress,
resolve,
reject,
abort() {
this.isAborted = true;
}
});
while (active.length < MAX_CONVERT && queue.length && !isStop) {
convert(queue.shift());
}
});
},
del: (taskIds) => {
if (!taskIds.length)
return;
isStop = true;
active = active.filter((convertMeta) => {
if (taskIds.includes(convertMeta.id)) {
convertMeta.abort();
}
else {
return true;
}
});
queue = queue.filter((convertMeta) => !taskIds.includes(convertMeta.id));
isStop = false;
while (active.length < MAX_CONVERT && queue.length) {
convert(queue.shift());
}
}
};
}
const createConverter = {
_deps: {
gif: '',
apng: '',
webp: ''
},
initialDeps,
createInstance
};
const zip = new JSZip__default["default"]();
function add(id, name, data) {
zip.folder(id)?.file(name, data);
}
function bundle(id) {
const folder = zip.folder(id);
if (!folder)
throw new Error('[Error]no folder:' + id);
return folder.generateAsync({ type: 'blob' });
}
function remove(ids) {
if (typeof ids === 'string') {
zip.remove(ids);
}
else {
const dirs = zip.filter((_, file) => file.dir).map((dir) => dir.name);
const dirsToDel = ids.filter((id) => dirs.some((dir) => dir.includes(id)));
dirsToDel.forEach((dir) => zip.remove(dir));
}
}
function fileCount(id) {
let count = 0;
zip.folder(id)?.forEach(() => count++);
return count;
}
const compressor = {
add,
bundle,
remove,
fileCount
};
const _saveWithoutSubpath = (blob, downloadMeta) => {
const dlEle = document.createElement('a');
dlEle.href = URL.createObjectURL(blob);
dlEle.download = downloadMeta.source.path;
dlEle.click();
URL.revokeObjectURL(dlEle.href);
downloadMeta.resolve(downloadMeta.taskId);
};
const _saveWithSubpath = (blob, downloadMeta) => {
const imgUrl = URL.createObjectURL(blob);
const request = {
url: imgUrl,
name: downloadMeta.source.path,
onerror: (error) => {
console.log('[pixiv downloader]Error when saving', downloadMeta.source.path);
URL.revokeObjectURL(imgUrl);
downloadMeta.reject(error);
},
onload: () => {
if (typeof downloadMeta.onLoad === 'function')
downloadMeta.onLoad();
URL.revokeObjectURL(imgUrl);
downloadMeta.resolve(downloadMeta.taskId);
}
};
downloadMeta.abort = GM_download(request).abort;
};
function createDownloader(converter) {
const MAX_DOWNLOAD = 5;
const MAX_RETRY = 3;
const INTERVAL = 500;
const TIMEOUT = 20000;
let isStop = false;
let queue = [];
let active = [];
let save;
if (env.isBlobDlAvaliable && env.isSupportSubpath) {
save = _saveWithSubpath;
}
else {
debugLog('[Info]scriptHandler:', GM_info.scriptHandler, GM_info.version);
save = _saveWithoutSubpath;
}
const download = (downloadMeta) => {
const { taskId, source } = downloadMeta;
debugLog('[Info]Start download:', source.path);
active.push(downloadMeta);
let abortObj;
const errorHandler = errorHandlerFactory(downloadMeta);
if ((!env.isBlobDlAvaliable || env.isViolentmonkey) &&
!('kind' in source)) {
abortObj = GM_download({
url: source.src,
name: source.path,
headers: {
referer: 'https://www.pixiv.net'
},
ontimeout: errorHandler,
onerror: errorHandler,
onload: async () => {
debugLog('[Info]Download complete', source.path);
if (typeof downloadMeta.onLoad === 'function')
downloadMeta.onLoad();
downloadMeta.resolve(taskId);
await sleep(INTERVAL);
active.splice(active.indexOf(downloadMeta), 1);
if (queue.length && !isStop)
download(queue.shift());
}
});
}
else {
abortObj = GM_xmlhttpRequest({
url: source.src,
timeout: TIMEOUT,
method: 'GET',
headers: {
referer: 'https://www.pixiv.net'
},
responseType: 'blob',
ontimeout: errorHandler,
onerror: errorHandler,
onprogress: (e) => {
if (e.lengthComputable &&
typeof downloadMeta.onProgress === 'function') {
downloadMeta.onProgress(e.loaded / e.total);
}
},
onload: async (e) => {
if (downloadMeta.state === 0 )
return debugLog();
if (!('kind' in source)) {
save(e.response, downloadMeta);
}
else if (source.kind === 'convert') {
const convertSource = {
id: taskId,
data: e.response,
format: settings.ugoriaFormat,
framesInfo: source.ugoiraMeta?.frames
};
converter
.add(convertSource, { onProgress: downloadMeta.onProgress })
.then((blob) => {
save(blob, downloadMeta);
}, downloadMeta.reject);
}
else if (source.kind === 'bundle') {
compressor.add(taskId, source.filename, e.response);
if (compressor.fileCount(taskId) === source.pageCount) {
compressor.bundle(taskId).then((blob) => {
save(blob, downloadMeta);
compressor.remove(taskId);
});
}
else {
downloadMeta.resolve(taskId);
if (typeof downloadMeta.onLoad === 'function')
downloadMeta.onLoad();
}
}
await sleep(INTERVAL);
active.splice(active.indexOf(downloadMeta), 1);
if (queue.length && !isStop)
download(queue.shift());
}
});
}
downloadMeta.abort = () => {
downloadMeta.state = 0 ;
abortObj.abort();
downloadMeta.reject('[Warning]xhr abort manually. ' + taskId);
};
};
function errorHandlerFactory(downloadMeta) {
return function () {
const { source } = downloadMeta;
debugLog('[Error]xmlhttpRequest timeout:', source.src);
downloadMeta.retry++;
if (downloadMeta.retry > MAX_RETRY) {
downloadMeta.reject('[Error]xmlhttpRequest failed: ' + source.src);
console.log('[pixiv downloader]Network error:', source.path, source.src);
active.splice(active.indexOf(downloadMeta), 1);
if (queue.length && !isStop)
download(queue.shift());
}
else {
debugLog('[Warning]retry xhr:', downloadMeta.retry, source.src);
download(downloadMeta);
}
};
}
function add(metas, handler = {}) {
if (metas.length < 1)
return Promise.resolve('');
const promises = [];
metas.forEach((source) => {
let downloadMeta;
promises.push(new Promise((resolve, reject) => {
downloadMeta = {
taskId: source.taskId,
source,
state: 1 ,
retry: 0,
onProgress: handler.onProgress,
onLoad: handler.onLoad,
resolve,
reject
};
queue.push(downloadMeta);
}));
});
while (active.length < MAX_DOWNLOAD && queue.length && !isStop) {
download(queue.shift());
}
return Promise.all(promises).then(([taskId]) => taskId);
}
function del(taskIds) {
if (!taskIds.length)
return;
isStop = true;
active = active.filter((downloadMeta) => {
if (taskIds.includes(downloadMeta.taskId)) {
downloadMeta.abort?.();
}
else {
return true;
}
});
queue = queue.filter((downloadMeta) => !taskIds.includes(downloadMeta.taskId));
isStop = false;
while (active.length < MAX_DOWNLOAD && queue.length) {
download(queue.shift());
}
}
return {
add,
del
};
}
const needBundle = (type) => {
return ((type === IllustType.manga && settings.bundleManga) ||
(type === IllustType.illusts && settings.bundleIllusts));
};
const replaceInvalidChar = (string) => {
if (!string)
return '';
const temp = document.createElement('div');
temp.innerHTML = string;
if (!temp.textContent)
return '';
return temp.textContent
.trim()
.replace(/^\.|\.$/g, '')
.replace(/[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?|]/g, '')
.replace(/"/g, "'")
.replace(/</g, '﹤')
.replace(/>/g, '﹥');
};
const getFilePath = ({ user, userId, title, tagStr, illustId, page, ext }, option = { needBundle: false, needConvert: false }) => {
let pathPattern;
if (settings.folderPattern &&
env.isSupportSubpath &&
(!option.needConvert || env.isBlobDlAvaliable) &&
!option.needBundle) {
pathPattern = settings.folderPattern + '/' + settings.filenamePattern;
}
else {
pathPattern = settings.filenamePattern;
}
if (option.needBundle && !settings.filenamePattern.includes('{page}')) {
pathPattern += '_{page}';
}
return (pathPattern
.replaceAll('{artist}', user)
.replaceAll('{artistID}', userId)
.replaceAll('{title}', title)
.replaceAll('{tags}', tagStr)
.replaceAll('{page}', String(page))
.replaceAll('{id}', illustId) + ext);
};
const makeTagsStr = (prev, cur, index, tagsArr) => {
const tag = settings.tagLang === 'ja' ? cur.tag : cur.translation?.['en'] || cur.tag;
if (index < tagsArr.length - 1) {
return prev + tag + '_';
}
else {
return prev + tag;
}
};
function filterWorks(works, option) {
const obj = {
unavaliable: [],
avaliable: [],
invalid: []
};
works.forEach((work) => {
if (!work.isBookmarkable) {
obj.unavaliable.push(work.id);
}
else if (option.exclude_downloaded && pixivHistory.has(work.id)) {
obj.invalid.push(work.id);
}
else if (!option[work.illustType]) {
obj.invalid.push(work.id);
}
else {
obj.avaliable.push(work.id);
}
});
return obj;
}
async function getFollowLatestGenerator(filterOption, mode, page) {
const MAX_PAGE = 34;
const MAX_ILLUSTS_PER_PAGE = 60;
let lastId;
let total;
let data;
let cache;
function findLastId(ids) {
return Math.min(...ids.map((id) => Number(id)));
}
if (page === undefined) {
data = await api.getFollowLatestWorks(1, mode);
const ids = data.page.ids;
total = ids.length;
lastId = findLastId(ids);
if (total === MAX_ILLUSTS_PER_PAGE) {
const secondPageData = await api.getFollowLatestWorks(2, mode);
const secondIds = secondPageData.page.ids;
const secondLastId = findLastId(secondIds);
if (secondLastId < lastId) {
lastId = secondLastId;
cache = secondPageData;
total += secondIds.length;
}
}
}
else {
data = await api.getFollowLatestWorks(page, mode);
total = data.page.ids.length;
}
async function* generateIds() {
yield filterWorks(data.thumbnails.illust, filterOption);
if (page === undefined) {
if (total === MAX_ILLUSTS_PER_PAGE)
return;
if (total < MAX_ILLUSTS_PER_PAGE * 2) {
yield filterWorks(cache.thumbnails.illust, filterOption);
return;
}
let currentPage = 3;
while (currentPage <= MAX_PAGE) {
const data = await api.getFollowLatestWorks(currentPage, mode);
const ids = data.page.ids;
const pageLastId = findLastId(ids);
if (pageLastId >= lastId) {
yield filterWorks(cache.thumbnails.illust, filterOption);
break;
}
lastId = pageLastId;
total += ids.length;
yield { ...filterWorks(cache.thumbnails.illust, filterOption), total };
cache = data;
currentPage++;
await sleep(3000);
}
}
}
return {
total,
generator: generateIds()
};
}
async function getChunksGenerator(userId, category, tag, rest, filterOption) {
const OFFSET = 48;
let requestUrl;
if (category === 'bookmarks') {
requestUrl = `https://www.pixiv.net/ajax/user/${userId}/illusts/bookmarks?tag=${tag}&offset=0&limit=${OFFSET}&rest=${rest}&lang=ja`;
}
else {
requestUrl = `https://www.pixiv.net/ajax/user/${userId}/${category}/tag?tag=${tag}&offset=0&limit=${OFFSET}&lang=ja`;
}
let head = 0;
const firstPageData = await api.getJson(requestUrl);
const total = firstPageData.total;
async function* generateIds() {
yield filterWorks(firstPageData.works, filterOption);
head += OFFSET;
while (head < total) {
const data = await api.getJson(requestUrl.replace('offset=0', 'offset=' + head));
head += OFFSET;
await sleep(3000);
yield filterWorks(data.works, filterOption);
}
}
return {
total,
generator: generateIds()
};
}
async function getAllWorksGenerator(userId, filterOption) {
const requestUrl = 'https://www.pixiv.net/ajax/user/' + userId + '/profile/all';
const profile = await api.getJson(requestUrl);
let illustIds = [];
let mangaIds = [];
if ((filterOption[IllustType.illusts] || filterOption[IllustType.ugoira]) &&
typeof profile.illusts === 'object') {
illustIds.push(...Object.keys(profile.illusts).reverse());
}
if (filterOption[IllustType.manga] && typeof profile.manga === 'object') {
mangaIds.push(...Object.keys(profile.manga).reverse());
}
if (filterOption.exclude_downloaded) {
illustIds = illustIds.filter((id) => !pixivHistory.has(id));
mangaIds = mangaIds.filter((id) => !pixivHistory.has(id));
}
async function* generateIds() {
const OFFSET = 48;
const baseUrl = 'https://www.pixiv.net/ajax/user/' + userId + '/profile/illusts';
let workCategory = 'illust';
while (illustIds.length > 0) {
let searchStr = '?';
const chunk = illustIds.splice(0, OFFSET);
searchStr +=
chunk.map((id) => 'ids[]=' + id).join('&') +
`&work_category=${workCategory}&is_first_page=0&lang=ja`;
const data = await api.getJson(baseUrl + searchStr);
await sleep(3000);
yield filterWorks(Object.values(data.works), filterOption);
}
workCategory = 'manga';
while (mangaIds.length > 0) {
let searchStr = '?';
const chunk = mangaIds.splice(0, OFFSET);
searchStr +=
chunk.map((id) => 'ids[]=' + id).join('&') +
`&work_category=${workCategory}&is_first_page=0&lang=ja`;
const data = await api.getJson(baseUrl + searchStr);
await sleep(3000);
yield filterWorks(Object.values(data.works), filterOption);
}
}
return {
total: illustIds.length + mangaIds.length,
generator: generateIds()
};
}
async function getArtworkData(illustId) {
const htmlText = await api.getArtworkHtml(illustId);
const preloadDataText = htmlText.match(regexp.preloadData);
if (!preloadDataText)
throw new Error('[Error]Fail to parse preload data.');
const preloadData = JSON.parse(preloadDataText[1]);
const illustData = preloadData.illust[illustId];
const globalDataText = htmlText.match(regexp.globalData);
if (!globalDataText)
throw new Error('[Error]Fail to parse global data.');
const globalData = JSON.parse(globalDataText[1]);
let ugoiraMeta;
if (illustData.illustType === IllustType.ugoira) {
ugoiraMeta = await api.getJson('/ajax/illust/' + illustId + '/ugoira_meta');
}
return {
illustData,
globalData,
ugoiraMeta
};
}
function getDownloadSource(artworkData, seletedPage) {
const { illustData, ugoiraMeta } = artworkData;
const { illustType, userName, userId, illustTitle, illustId, tags, pageCount } = illustData;
const pathInfo = {
user: replaceInvalidChar(userName) || 'userId-' + userId,
title: replaceInvalidChar(illustTitle) || 'illustId-' + illustId,
tagStr: replaceInvalidChar(tags.tags.reduce(makeTagsStr, '')),
illustId,
userId,
ext: '',
page: 0
};
const metas = [];
const taskId = illustId + '_' + Math.random().toString(36).slice(2);
if (illustType === IllustType.illusts || illustType === IllustType.manga) {
const firstImgSrc = illustData.urls.original;
const srcPrefix = firstImgSrc.slice(0, firstImgSrc.indexOf('_') + 2);
const extendName = firstImgSrc.slice(-4);
pathInfo.ext = extendName;
if (pageCount > 1 && seletedPage === undefined) {
if (needBundle(illustType)) {
const path = getFilePath({ ...pathInfo, ext: '.zip', page: pageCount });
for (let i = 0; i < pageCount; i++) {
pathInfo.page = i;
metas.push({
kind: 'bundle',
taskId,
path,
src: srcPrefix + i + extendName,
filename: getFilePath(pathInfo, { needBundle: true }),
pageCount
});
}
}
else {
for (let i = 0; i < pageCount; i++) {
pathInfo.page = i;
metas.push({
taskId,
path: getFilePath(pathInfo),
src: srcPrefix + i + extendName
});
}
}
}
else {
let src = firstImgSrc;
if (seletedPage !== undefined) {
src = srcPrefix + seletedPage + extendName;
pathInfo.page = seletedPage;
}
metas.push({
taskId,
path: getFilePath(pathInfo),
src
});
}
}
else if (illustType === IllustType.ugoira && ugoiraMeta) {
pathInfo.ext = '.' + settings.ugoriaFormat;
if (settings.ugoriaFormat !== 'zip') {
metas.push({
kind: 'convert',
ugoiraMeta,
taskId,
src: ugoiraMeta.originalSrc,
path: getFilePath(pathInfo, { needConvert: true })
});
}
else {
metas.push({
taskId,
src: ugoiraMeta.originalSrc,
path: getFilePath(pathInfo)
});
}
}
return metas;
}
const parser = {
getChunksGenerator,
getAllWorksGenerator,
getFollowLatestGenerator,
getArtworkData,
getDownloadSource
};
let converter;
let downloader;
async function initial() {
converter = await createConverter
.initialDeps()
.then((createConverter) => createConverter.createInstance());
downloader = createDownloader(converter);
}
function addBookmark(pdlBtn, illustId, token, tags) {
if (!settings.addBookmark)
return;
api
.addBookmark(illustId, token, settings.addBookmarkWithTags ? tags : [], settings.privateR18 && tags.includes('R-18')
? BookmarkRestrict.private
: BookmarkRestrict.public)
.then(() => {
const bookmarkBtnRef = findBookmarkBtn(pdlBtn);
if (!bookmarkBtnRef)
return;
switch (bookmarkBtnRef.kind) {
case "main" : {
const pathBorder = bookmarkBtnRef.button.querySelector('svg g path');
pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)');
break;
}
case "sub" : {
const pathBorder = bookmarkBtnRef.button.querySelector('path');
pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)');
break;
}
case "rank" : {
bookmarkBtnRef.button.style.backgroundColor = 'rgb(255, 64, 96)';
break;
}
}
})
.catch((reason) => {
console.error(reason.message);
});
}
function findBookmarkBtn(pdlBtn) {
const bookmarkBtnRef = {};
if (pdlBtn.classList.contains('pdl-btn-sub')) {
const btn = pdlBtn.parentElement?.nextElementSibling?.querySelector('button[type="button"]');
if (btn) {
bookmarkBtnRef.kind = "sub" ;
bookmarkBtnRef.button = btn;
}
else {
const btn = pdlBtn.parentElement?.querySelector('div._one-click-bookmark');
if (btn) {
bookmarkBtnRef.kind = "rank" ;
bookmarkBtnRef.button = btn;
}
}
}
else if (pdlBtn.classList.contains('pdl-btn-main')) {
const btn = pdlBtn.parentElement?.parentElement?.querySelector('button.gtm-main-bookmark');
if (btn) {
bookmarkBtnRef.kind = "main" ;
bookmarkBtnRef.button = btn;
}
}
else {
return debugLog();
}
return bookmarkBtnRef;
}
function handleDownload(pdlBtn, illustId) {
let pageCount, pageComplete = 0, shouldDownloadPage;
const pageAttr = pdlBtn.getAttribute('should-download');
if (pageAttr) {
shouldDownloadPage = Number(pageAttr);
}
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':
case 'webp':
pdlBtn.textContent = String(progress);
break;
case 'zip':
pdlBtn.textContent = '';
break;
}
};
const onLoad = function () {
if (pageCount < 2)
return;
const progress = Math.floor((++pageComplete / pageCount) * 100);
pdlBtn.textContent = String(progress);
pdlBtn.style.setProperty('--pdl-progress', progress + '%');
};
pdlBtn.classList.add('pdl-progress');
parser
.getArtworkData(illustId)
.then((artworkData) => {
const { illustData, globalData } = artworkData;
const { illustId, tags, bookmarkData } = illustData;
if (!bookmarkData) {
const { token } = globalData;
const tagsArr = tags.tags.map((item) => item.tag);
addBookmark(pdlBtn, illustId, token, tagsArr);
}
return parser.getDownloadSource(artworkData, shouldDownloadPage);
})
.then((sources) => {
pageCount = sources.length;
return downloader.add(sources, { onLoad, onProgress });
})
.then(() => {
pixivHistory.add(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 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 (!ele.classList.contains('pdl-progress')) {
handleDownload(ele, ele.getAttribute('pdl-id'));
}
});
}
return ele;
}
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 createThunbnailsBtn(nodes) {
const isBookmarkPage = regexp.bookmarkPage.test(location.pathname);
nodes.forEach((e) => {
if (e.childElementCount !== 0 &&
!e.querySelector('.pdl-btn-sub')) {
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 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();
});
}
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 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': String(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) ?? [];
if (!pageNum)
throw new Error('[Error]Invalid Image Element.');
btn?.setAttribute('should-download', String(pageNum));
}
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');
if (!img)
return;
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);
if (!img.parentElement)
return;
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');
const 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': String(idx) },
classList: ['pdl-btn', 'pdl-btn-sub']
};
node.appendChild(createPdlBtn(attrs));
});
}, 300);
}
function changeDlbarDisplay() {
document.querySelectorAll('.pdl-dlbar .pdl-btn-all').forEach((ele) => {
ele.classList.toggle('pdl-hide');
});
document.querySelector('.pdl-dlbar .pdl-stop')?.classList.toggle('pdl-hide');
document.querySelectorAll('.pdl-tag').forEach((ele) => {
ele.classList.toggle('pdl-tag-hide');
});
document.querySelector('.pdl-filter-wrap')?.classList.toggle('unavailable');
}
let isDownloading = false;
const dlBarRef = {
filter: {
exclude_downloaded: undefined,
[IllustType.illusts]: undefined,
[IllustType.manga]: undefined,
[IllustType.ugoira]: undefined
},
statusBar: undefined,
abortBtn: undefined
};
function getFilterOption() {
return {
exclude_downloaded: dlBarRef.filter.exclude_downloaded?.checked ??
defaultSettings.filter.exclude_downloaded,
[IllustType.illusts]: dlBarRef.filter[IllustType.illusts]?.checked ??
defaultSettings.filter[IllustType.illusts],
[IllustType.manga]: dlBarRef.filter[IllustType.manga]?.checked ??
defaultSettings.filter[IllustType.manga],
[IllustType.ugoira]: dlBarRef.filter[IllustType.ugoira]?.checked ??
defaultSettings.filter[IllustType.ugoira]
};
}
function updateStatus(str) {
dlBarRef.statusBar && (dlBarRef.statusBar.textContent = str);
}
function onProgressCB(progressData) {
if (typeof progressData === 'string') {
updateStatus(progressData);
}
else {
debugLog('update progress by', progressData.illustId);
updateStatus(`Downloading: ${progressData.completed} / ${progressData.avaliable}`);
}
}
async function useDownload(chunksGenerators) {
if (!dlBarRef.abortBtn)
return;
isDownloading = true;
changeDlbarDisplay();
let total = 0;
let failedResult;
const idsGenerators = [];
try {
if (chunksGenerators instanceof Array) {
await Promise.all(chunksGenerators).then((chunksGenerator) => {
chunksGenerator.forEach((val) => {
total += val.total;
idsGenerators.push(val.generator);
});
});
}
else {
await chunksGenerators.then((val) => {
total = val.total;
idsGenerators.push(val.generator);
});
}
}
catch (error) {
console.error(error);
updateStatus('Network error, see console.');
changeDlbarDisplay();
isDownloading = false;
return;
}
try {
if (total === 0) {
throw 'No works.';
}
debugLog('[Info]Total:', total);
const { failed, unavaliable } = await downloadByIds(total, idsGenerators, dlBarRef.abortBtn, onProgressCB);
if (failed.length || unavaliable.length) {
updateStatus(`Failed: ${failed.length + unavaliable.length}. See console.`);
console.log('[Pixiv Downloader]Failed: ', failed.join(', '));
console.log('[Pixiv Downloader]Unavaliable: ', unavaliable.join(', '));
if (failed.length)
failedResult = failed;
}
else {
updateStatus('Complete');
}
}
catch (error) {
updateStatus(error);
}
changeDlbarDisplay();
isDownloading = false;
return failedResult;
}
async function downloadByIds(total, idsGenerators, abortBtn, onProgress) {
let resolve;
let reject;
const done = new Promise((r, j) => {
resolve = r;
reject = j;
});
let wakeFn;
let completed = 0;
const failed = [], unavaliable = [], invalid = [];
let isCanceled = false;
const tasks = [];
let tooManyRequests = false;
abortBtn.onclick = () => {
isCanceled = true;
abortBtn.onclick = null;
reject(`Stopped. ${completed} / ${total - failed.length - unavaliable.length - invalid.length}`);
wakeFn && wakeFn();
if (tasks.length) {
downloader.del(tasks);
converter.del(tasks);
compressor.remove(tasks);
tasks.length = 0;
}
};
const afterEach = (illustId) => {
const avaliable = total - failed.length - unavaliable.length - invalid.length;
onProgress({
illustId,
avaliable,
completed
});
if (completed === avaliable) {
resolve({ failed, unavaliable });
}
};
onProgress('Downloading...');
try {
for (const idsGenerator of idsGenerators) {
if (isCanceled)
return done;
for await (const ids of idsGenerator) {
debugLog('[Info]ids:', ids);
if (isCanceled)
return done;
if (ids.unavaliable.length) {
unavaliable.push(...ids.unavaliable);
}
if (ids.invalid.length) {
invalid.push(...ids.invalid);
}
if (typeof ids.total === 'number' && !Number.isNaN(ids.total)) {
total = ids.total;
}
if (ids.avaliable.length) {
for (const id of ids.avaliable) {
if (isCanceled)
return done;
if (tooManyRequests) {
onProgress('Too many requests, wait 30s');
const { wake, sleep } = wakeableSleep(30000);
wakeFn = wake;
await sleep;
tooManyRequests = false;
if (isCanceled)
return done;
onProgress('Downloading...');
}
parser
.getArtworkData(id)
.then(parser.getDownloadSource)
.then((sources) => {
if (isCanceled) {
throw '[Warning]Download stop manually: ' + id;
}
tasks.push(sources[0].taskId);
return downloader.add(sources);
})
.then((taskId) => {
pixivHistory.add(id);
if (!isCanceled) {
tasks.splice(tasks.indexOf(taskId), 1);
completed++;
afterEach(id);
}
}, (reason) => {
if (!isCanceled) {
if (reason.message && reason.message === '429')
tooManyRequests = true;
if (reason.message &&
reason.message === '[Error]Fail to parse preload data.') {
unavaliable.push(id);
}
else {
failed.push(id);
}
afterEach(id);
}
});
await sleep(1000);
}
}
else {
afterEach();
}
}
}
}
catch (error) {
console.error(error);
reject('Network Error, see console');
abortBtn.click();
}
return done;
}
function downloadWorks(evt) {
evt.preventDefault();
evt.stopPropagation();
if (isDownloading)
return;
const btn = evt.target;
const userId = btn.getAttribute('pdl-userid');
const filterOption = getFilterOption();
const ids = parser.getAllWorksGenerator(userId, filterOption);
useDownload(ids).then((failed) => {
if (failed instanceof Array && failed.length) {
const gen = async function* () {
yield {
avaliable: failed,
unavaliable: [],
invalid: []
};
};
useDownload(Promise.resolve({ total: failed.length, generator: gen() }));
}
});
}
async function downloadBookmarksOrTags(evt) {
evt.preventDefault();
evt.stopPropagation();
if (isDownloading)
return;
const btn = evt.target;
const userId = btn.getAttribute('pdl-userid');
const category = btn.getAttribute('category');
const tag = btn.getAttribute('tag') || '';
const rest = (btn.getAttribute('rest') || 'show');
const filterOption = getFilterOption();
let idsGenerators;
if (rest === 'all') {
const idsShowPromise = parser.getChunksGenerator(userId, 'bookmarks', '', 'show', filterOption);
const idsHidePromise = parser.getChunksGenerator(userId, 'bookmarks', '', 'hide', filterOption);
idsGenerators = [idsShowPromise, idsHidePromise];
}
else {
idsGenerators = parser.getChunksGenerator(userId, category, tag, rest, filterOption);
}
useDownload(idsGenerators).then((failed) => {
if (failed instanceof Array && failed.length) {
const gen = async function* () {
yield {
avaliable: failed,
unavaliable: [],
invalid: []
};
};
useDownload(Promise.resolve({ total: failed.length, generator: gen() }));
}
});
}
function downloadFollowLatest(evt) {
evt.preventDefault();
evt.stopPropagation();
if (isDownloading)
return;
const btn = evt.target;
let idsGenerators;
const mode = location.pathname.includes('r18') ? 'r18' : 'all';
const filterOption = getFilterOption();
if (btn.classList.contains('pdl-dl-all')) {
idsGenerators = parser.getFollowLatestGenerator(filterOption, mode);
}
else {
const params = new URLSearchParams(location.search);
const page = Number(params.get('p')) || 1;
idsGenerators = parser.getFollowLatestGenerator(filterOption, mode, page);
}
useDownload(idsGenerators).then((failed) => {
if (failed instanceof Array && failed.length) {
const gen = async function* () {
yield {
avaliable: failed,
unavaliable: [],
invalid: []
};
};
useDownload(Promise.resolve({ total: failed.length, generator: gen() }));
}
});
}
function createFilterEl(id, filterType, text) {
const checkbox = document.createElement('input');
const label = document.createElement('label');
checkbox.id = id;
checkbox.type = 'checkbox';
checkbox.classList.add('pdl-checkbox');
checkbox.setAttribute('category', String(filterType));
checkbox.checked = settings.filter[filterType];
label.setAttribute('for', id);
label.setAttribute('category', String(filterType));
label.textContent = text;
checkbox.addEventListener('change', (evt) => {
const checkbox = evt.currentTarget;
const category = checkbox.getAttribute('category');
upgradeSettings('filter', {
...settings.filter,
[category]: checkbox.checked
});
});
dlBarRef.filter[filterType] = checkbox;
const wrap = document.createElement('div');
wrap.classList.add('pdl-filter');
wrap.appendChild(checkbox);
wrap.appendChild(label);
return wrap;
}
function createFilter() {
const wrapper = document.createElement('div');
wrapper.classList.add('pdl-filter-wrap');
wrapper.appendChild(createFilterEl('pdl-filter-exclude_downloaded', 'exclude_downloaded', i18n('dlbar_filter_exclude_downloaded')));
wrapper.appendChild(createFilterEl('pdl-filter-illusts', IllustType.illusts, i18n('dlbar_filter_illusts')));
wrapper.appendChild(createFilterEl('pdl-filter-manga', IllustType.manga, i18n('dlbar_filter_manga')));
wrapper.appendChild(createFilterEl('pdl-filter-ugoria', IllustType.ugoira, i18n('dlbar_filter_ugoria')));
return wrapper;
}
function createDownloadBar(userId) {
const nav = document.querySelector('nav[class~="sc-192ftwf-0"]');
if (!nav)
return;
const dlBtn = nav.querySelector('.pdl-btn-all');
if (dlBtn) {
if (dlBtn.getAttribute('pdl-userid') === userId)
return;
removeDownloadBar();
}
const dlBar = document.createElement('div');
dlBar.classList.add('pdl-dlbar');
const statusBar = document.createElement('div');
statusBar.classList.add('pdl-dlbar-status_bar');
dlBarRef.statusBar = dlBar.appendChild(statusBar);
const baseClasses = nav.querySelector('a:not([aria-current])').classList;
dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
attrs: { 'pdl-userId': userId },
classList: [...baseClasses, 'pdl-stop', 'pdl-hide']
}, i18n('stop'), { addEvent: false }));
if (userId !== getSelfId()) {
const hasWorks = ["a[href$='illustrations']", "a[href$='manga']"].some((selector) => !!nav.querySelector(selector));
if (hasWorks) {
const el = createPdlBtn({
attrs: { 'pdl-userid': userId },
classList: [...baseClasses, 'pdl-btn-all']
}, i18n('dlbar_category_works'), { addEvent: false });
el.addEventListener('click', downloadWorks);
dlBar.appendChild(el);
}
if (nav.querySelector("a[href*='bookmarks']")) {
const el = createPdlBtn({
attrs: { 'pdl-userid': userId, category: 'bookmarks' },
classList: [...baseClasses, 'pdl-btn-all']
}, i18n('bookmarks'), { addEvent: false });
el.addEventListener('click', downloadBookmarksOrTags);
dlBar.appendChild(el);
}
}
else {
if (nav.querySelector("a[href*='bookmarks']")) {
dlBar.appendChild(createPdlBtn({
attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'all' },
classList: [...baseClasses, 'pdl-btn-all']
}, i18n('bookmarks'), { addEvent: false }));
dlBar.appendChild(createPdlBtn({
attrs: {
'pdl-userid': userId,
category: 'bookmarks',
rest: 'show'
},
classList: [...baseClasses, 'pdl-btn-all']
}, i18n('bookmarks_public'), { addEvent: false }));
dlBar.appendChild(createPdlBtn({
attrs: {
'pdl-userid': userId,
category: 'bookmarks',
rest: 'hide'
},
classList: [...baseClasses, 'pdl-btn-all']
}, i18n('bookmarks_private'), { addEvent: false }));
dlBar
.querySelectorAll('.pdl-btn-all')
.forEach((node) => {
node.addEventListener('click', downloadBookmarksOrTags);
});
}
}
const filter = createFilter();
nav.parentElement.insertBefore(filter, nav);
nav.appendChild(dlBar);
}
function removeDownloadBar() {
const dlBarWrap = document.querySelector('.pdl-dlbar');
if (dlBarWrap) {
dlBarWrap.remove();
document.querySelector('.pdl-filter-wrap')?.remove();
}
}
function createTagsBtn(userId, category) {
const tagsEles = document.querySelectorAll('section> div:nth-child(2) > div > div');
if (!tagsEles.length)
return;
let cate;
if (category === 'illustrations' || category === 'artworks') {
cate = 'illusts';
}
else {
cate = category;
}
let rest = 'show';
if (userId === getSelfId() &&
category === 'bookmarks' &&
location.search.includes('rest=hide'))
rest = 'hide';
tagsEles.forEach((ele) => {
const tagBtn = ele.querySelector('.pdl-btn');
if (tagBtn) {
const btnRest = tagBtn.getAttribute('rest');
if (rest !== btnRest)
tagBtn.setAttribute('rest', rest);
return;
}
let tag;
const tagLink = ele.querySelector('a');
if (!tagLink)
return;
if (tagLink.getAttribute('status') !== 'active') {
if (rest === 'hide') {
tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1, tagLink.href.lastIndexOf('?'));
}
else {
tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1);
}
}
else {
const tagTextEles = ele.querySelectorAll('div[title]');
if (!tagTextEles.length)
return console.log('[Info]No Tags Element found.');
tag = tagTextEles[tagTextEles.length - 1].getAttribute('title').slice(1);
}
const attrs = {
attrs: { 'pdl-userId': userId, category: cate, tag, rest },
classList: ['pdl-btn', 'pdl-tag']
};
if (isDownloading)
attrs.classList.push('pdl-tag-hide');
const dlBtn = createPdlBtn(attrs, '', { addEvent: false });
if (!(tagLink.href.includes('bookmarks') &&
tagLink.getAttribute('status') !== 'active')) {
dlBtn.style.backgroundColor = tagLink.getAttribute('color') + '80';
}
dlBtn.addEventListener('click', downloadBookmarksOrTags);
ele.appendChild(dlBtn);
});
let modalTagsEles;
let modal;
if (category === 'bookmarks') {
modal = document.querySelector('div[role="presentation"]');
if (!modal)
return;
modalTagsEles = modal.querySelectorAll('a');
}
else {
const charcoalTokens = document.querySelectorAll('.charcoal-token');
modal = charcoalTokens[charcoalTokens.length - 1];
if (!modal)
return;
modalTagsEles = modal.querySelectorAll('a');
}
if (!regexp.userPageTags.exec(modalTagsEles[0]?.href))
return;
modalTagsEles.forEach((ele) => {
if (ele.querySelector('.pdl-btn'))
return;
let tag;
if (rest === 'hide') {
tag = ele.href.slice(ele.href.lastIndexOf('/') + 1, ele.href.lastIndexOf('?'));
}
else {
tag = ele.href.slice(ele.href.lastIndexOf('/') + 1);
}
const attrs = {
attrs: { 'pdl-userId': userId, category: cate, tag, rest },
classList: ['pdl-btn', 'pdl-modal-tag']
};
if (isDownloading)
attrs.classList.push('pdl-tag-hide');
const dlBtn = createPdlBtn(attrs, '', { addEvent: false });
dlBtn.addEventListener('click', (evt) => {
modal.querySelector('svg').parentElement.click();
downloadBookmarksOrTags(evt);
});
ele.appendChild(dlBtn);
});
}
function createFollowLatestDownloadBar() {
const prevDlBtn = document.querySelector('.pdl-btn-all');
if (prevDlBtn) {
const prevDlAllBtn = document.querySelector('.pdl-dl-all');
if (location.pathname.includes('r18')) {
prevDlBtn.textContent !==
i18n('dlbar.follow_latest.category_r18.single') &&
(prevDlBtn.textContent = i18n('dlbar.follow_latest.category_r18.single'));
prevDlAllBtn.textContent !==
i18n('dlbar.follow_latest.category_r18.all') &&
(prevDlAllBtn.textContent = i18n('dlbar.follow_latest.category_r18.all'));
}
else {
prevDlBtn.textContent !==
i18n('dlbar.follow_latest.category_all.single') &&
(prevDlBtn.textContent = i18n('dlbar.follow_latest.category_all.single'));
prevDlAllBtn.textContent !==
i18n('dlbar.follow_latest.category_all.all') &&
(prevDlAllBtn.textContent = i18n('dlbar.follow_latest.category_all.all'));
}
return;
}
const nav = document.querySelector('nav');
if (!nav || nav.parentElement.childElementCount === 1)
return;
const navBar = nav.parentElement;
const modeSwitch = nav.nextElementSibling;
const filter = createFilter();
navBar.parentElement.insertBefore(filter, navBar);
const dlBar = document.createElement('div');
dlBar.classList.add('pdl-dlbar');
dlBar.classList.add('pdl-dlbar-follow_latest');
const statusBar = document.createElement('div');
statusBar.classList.add('pdl-dlbar-status_bar');
dlBarRef.statusBar = dlBar.appendChild(statusBar);
const baseClasses = nav.querySelector('a:not([aria-current])').classList;
dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
attrs: { 'pdl-userid': '' },
classList: [...baseClasses, 'pdl-stop', 'pdl-hide']
}, i18n('stop'), { addEvent: false }));
const dlBtn = createPdlBtn({
attrs: { 'pdl-userid': '' },
classList: [...baseClasses, 'pdl-btn-all']
},
i18n('dlbar_category_works'), { addEvent: false });
dlBtn.addEventListener('click', downloadFollowLatest);
dlBar.appendChild(dlBtn);
const dlAllBtn = createPdlBtn({
attrs: { 'pdl-userid': '' },
classList: [...baseClasses, 'pdl-btn-all', 'pdl-dl-all']
},
i18n('dlbar_category_works'), { addEvent: false });
dlAllBtn.addEventListener('click', downloadFollowLatest);
dlBar.appendChild(dlAllBtn);
navBar.insertBefore(dlBar, modeSwitch);
}
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) {
createThunbnailsBtn(document.querySelectorAll('a'));
firstRun = false;
}
else {
compatPixivPreviewer(addedNodes);
const thunmnails = addedNodes.reduce((prev, current) => {
return prev.concat(Array.from(current.querySelectorAll('a')));
}, []);
createThunbnailsBtn(thunmnails);
}
const isArtworksPage = regexp.artworksPage.exec(location.pathname);
const isUserPage = regexp.userPage.exec(location.pathname);
const isTagsPage = regexp.userPageTags.exec(location.pathname);
if (isArtworksPage) {
const id = isArtworksPage[1];
createMainBtn(id);
createMultyWorksBtn(id);
createPresentationBtn(id);
createPreviewModalBtn();
}
else if (isUserPage) {
createDownloadBar(isUserPage[1]);
if (isTagsPage) {
createTagsBtn(isUserPage[1], isTagsPage[1]);
}
}
else if (regexp.followLatest.test(location.pathname)) {
createFollowLatestDownloadBar();
}
else {
removeDownloadBar();
}
}
function createModal({ header = '', content, footer = '' }, option) {
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: modalHtml.upgradeMsgTitle,
content: modalHtml.upgradeMsgContent,
footer: modalHtml.modalCreditFooter + modalHtml.modalFeedback
}, { closeOnClickModal: false }));
}
function createTabsPart({ tabName, paneHtml }) {
const tab = document.createElement('div');
tab.classList.add('pdl-tab-item');
tab.textContent = tabName;
const pane = document.createElement('div');
pane.classList.add('pdl-tab-pane');
pane.innerHTML = paneHtml;
return {
tab,
pane
};
}
function createTabFilename({ tabName, paneHtml }) {
const { tab, pane } = createTabsPart({ tabName, paneHtml });
const folder = pane.querySelector('#pdlfolder');
const folderReset = pane.querySelector('#pdl-filename-folder-reset');
const folderUpdate = pane.querySelector('#pdl-filename-folder-confirm');
const filename = pane.querySelector('#pdlfilename');
const filenameReset = pane.querySelector('#pdl-filename-filename-reset');
const filenameUpdate = pane.querySelector('#pdl-filename-filename-confirm');
if (!folder || !filename)
throw new Error('[Error]Can not create modal.');
filename.value = settings.filenamePattern;
if (!env.isSupportSubpath) {
folder.setAttribute('disabled', '');
folder.value = '';
}
else {
folder.value = settings.folderPattern;
}
folder.placeholder = env.isViolentmonkey
? i18n('folder_vm_tips')
: !env.isSupportSubpath
? i18n('folder_api_tips')
: i18n('folder_tips');
folder.addEventListener('input', () => {
folderReset?.removeAttribute('disabled');
folderUpdate?.removeAttribute('disabled');
});
folderReset?.addEventListener('click', () => {
folder.value = settings.folderPattern;
folderReset?.setAttribute('disabled', '');
folderUpdate?.setAttribute('disabled', '');
});
folderUpdate?.addEventListener('click', () => {
const folderPattern = folder.value
.split('/')
.map((path) => path
.trim()
.replace(/^\.+|\.+$|[\u200b-\u200f\uFEFF\u202a-\u202e\\:*?"|<>]/g, ''))
.filter((path) => !!path)
.join('/') ?? '';
upgradeSettings('folderPattern', folderPattern);
folder.value = folderPattern;
folderReset?.setAttribute('disabled', '');
folderUpdate?.setAttribute('disabled', '');
});
filename.addEventListener('input', () => {
filenameReset?.removeAttribute('disabled');
filenameUpdate?.removeAttribute('disabled');
});
filenameReset?.addEventListener('click', () => {
filename.value = settings.filenamePattern;
filenameReset?.setAttribute('disabled', '');
filenameUpdate?.setAttribute('disabled', '');
});
filenameUpdate?.addEventListener('click', () => {
const filenamePattern = filename.value
.trim()
.replace(/^\.+|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, '') ??
'';
if (filenamePattern === '')
return filenameReset?.click();
upgradeSettings('filenamePattern', filenamePattern);
filename.value = filenamePattern;
filenameReset?.setAttribute('disabled', '');
filenameUpdate?.setAttribute('disabled', '');
});
pane
.querySelectorAll('.tags-content .tags-item input')
.forEach((input) => {
if (settings.tagLang === input.value)
input.checked = true;
input.addEventListener('change', (ev) => {
upgradeSettings('tagLang', ev.currentTarget.value);
});
});
return {
tab,
pane
};
}
function createTabUgoria({ tabName, paneHtml }) {
const { tab, pane } = createTabsPart({ tabName, paneHtml });
pane
.querySelectorAll('.pdl-ugoria-format-item input[type="radio"]')
.forEach((el) => {
if (settings.ugoriaFormat === el.value)
el.checked = true;
el.addEventListener('change', (ev) => {
upgradeSettings('ugoriaFormat', ev.currentTarget.value);
});
});
return {
tab,
pane
};
}
function createTabHistory({ tabName, paneHtml }) {
const { tab, pane } = createTabsPart({ tabName, paneHtml });
const file = pane.querySelector('#pdl-import');
file?.addEventListener('change', (evt) => {
const file = evt.currentTarget.files?.[0];
if (!file)
return;
if (file.type === 'text/plain') {
const reader = new FileReader();
reader.readAsText(file);
reader.onload = (readEvt) => {
const text = readEvt.target?.result;
try {
if (typeof text !== 'string')
throw new Error('Invalid file');
const history = JSON.parse(text);
if (!(history instanceof Array))
throw new Error('Invalid file');
if (history.length &&
!history.every((id) => typeof id === 'string')) {
throw new Error('Invalid id type');
}
pixivHistory.saveHistory(history);
location.reload();
}
catch (error) {
alert(error.message);
}
};
}
else {
alert('Invalid file');
}
});
const importBtn = pane.querySelector('#pdl-import-btn');
importBtn?.addEventListener('click', () => file?.click());
const exportBtn = pane.querySelector('#pdl-export');
exportBtn?.addEventListener('click', () => {
const dlEle = document.createElement('a');
const history = JSON.stringify(pixivHistory.getAll());
dlEle.href = URL.createObjectURL(new Blob([history], { type: 'text/plain' }));
dlEle.download = 'Pixiv Downloader ' + new Date().toLocaleString() + '.txt';
dlEle.click();
URL.revokeObjectURL(dlEle.href);
});
const clearBtn = pane.querySelector('#pdl-clear-history');
clearBtn?.addEventListener('click', () => pixivHistory.clearHistory());
return {
tab,
pane
};
}
function createTabOthers({ tabName, paneHtml }) {
const { tab, pane } = createTabsPart({ tabName, paneHtml });
[
{ selector: '#pdl-options-bundle-illusts', settingKey: 'bundleIllusts' },
{ selector: '#pdl-options-bundle-manga', settingKey: 'bundleManga' },
{ selector: '#pdl-options-add-bookmark', settingKey: 'addBookmark' },
{
selector: '#pdl-options-add-bookmark-tags',
settingKey: 'addBookmarkWithTags'
},
{
selector: '#pdl-options-add-bookmark-private-r18',
settingKey: 'privateR18'
}
].forEach(({ selector, settingKey }) => {
const optionEl = pane.querySelector(selector);
optionEl && (optionEl.checked = settings[settingKey]);
optionEl?.addEventListener('change', (ev) => {
upgradeSettings(settingKey, ev.currentTarget.checked);
});
});
return {
tab,
pane
};
}
function showSettings() {
if (document.querySelector('.pdl-modal'))
return;
const modal = createModal({
content: modalHtml.modalSettingsContent
}, { closeOnClickModal: false });
const tabsNav = modal.querySelector('.pdl-tabs-nav');
const tabContent = modal.querySelector('.pdl-tabs-content');
[
createTabFilename({
tabName: i18n('tabs_nav_filename'),
paneHtml: modalHtml.modalSettingFilename
}),
createTabUgoria({
tabName: i18n('tabs_nav_ugoria'),
paneHtml: modalHtml.modalSettingUgoria
}),
createTabHistory({
tabName: i18n('tabs_nav_history'),
paneHtml: modalHtml.modalSettingHistory
}),
createTabOthers({
tabName: i18n('tabs_nav_others'),
paneHtml: modalHtml.modalSettingOthers
}),
createTabsPart({
tabName: i18n('tabs_nav_donate'),
paneHtml: modalHtml.modalSettingDonate
})
].forEach(({ tab, pane }) => {
tabsNav.appendChild(tab);
tabContent.appendChild(pane);
});
const panes = Array.from(tabContent.querySelectorAll('.pdl-tab-pane'));
panes.forEach((el) => {
el.style.setProperty('display', 'none');
});
const activeBar = tabsNav.querySelector('.pdl-tabs__active-bar');
const tabs = Array.from(modal.querySelectorAll('.pdl-tabs-nav .pdl-tab-item'));
tabs.forEach((el) => {
el.addEventListener('click', (ev) => {
const tab = ev.currentTarget;
if (!tab)
return;
tabs.forEach((tab) => tab.classList.remove('active'));
tab.classList.add('active');
activeBar.style.width = getComputedStyle(tab).width;
activeBar.style.transform = `translateX(${tab.offsetLeft + parseFloat(getComputedStyle(tab).paddingLeft)}px)`;
panes.forEach((pane) => pane.style.setProperty('display', 'none'));
panes[tabs.indexOf(tab)].style.removeProperty('display');
});
});
tabs[0].classList.add('active');
panes[0].style.removeProperty('display');
document.body.appendChild(modal);
activeBar.style.width = getComputedStyle(tabs[0]).width;
activeBar.style.transform = `translateX(${tabs[0].offsetLeft + parseFloat(getComputedStyle(tabs[0]).paddingLeft)}px)`;
}
pixivHistory.updateHistory();
GM_registerMenuCommand(i18n('gm_menu_setting'), showSettings, 's');
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'));
}
}
}
});
})
.catch(console.error);
})(workerChunk, JSZip, GIF);