// ==UserScript==
// @name Pixiv Downloader
// @namespace https://greasyfork.org/zh-CN/scripts/432150
// @version 0.9.4
// @description:en Download the original images of Pixiv pages with one click. Supports:multiple illustrations, ugoira(animation), and batch downloads of artists' work. Ugoira support format conversion: Gif | Apng | 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 unsafeWindow
// @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://cdn.jsdelivr.net/npm/dayjs@1/dayjs.min.js
// @require https://greasyfork.org/scripts/455256-toanimatedwebp/code/toAnimatedWebp.js?version=1120088
// ==/UserScript==
(function (workerChunk, GIF, JSZip, dayjs) {
'use strict';
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
var workerChunk__default = /*#__PURE__*/_interopDefaultLegacy(workerChunk);
var GIF__default = /*#__PURE__*/_interopDefaultLegacy(GIF);
var JSZip__default = /*#__PURE__*/_interopDefaultLegacy(JSZip);
var dayjs__default = /*#__PURE__*/_interopDefaultLegacy(dayjs);
var e=[],t$1=[];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$1[u]={}),a=t$1[u]&&t$1[u][s]?t$1[u][s]:t$1[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$4 = "@property --pdl-progress {\n syntax: '<percentage>';\n inherits: true;\n initial-value: 0%;\n}\n\n@keyframes pdl_loading {\n 100% {\n transform: translate(-50%, -50%) rotate(360deg);\n }\n}\n\n:root {\n --pdl-btn-top: 100;\n --pdl-btn-left: 0;\n --pdl-btn-self-bookmark-top: 75;\n --pdl-btn-self-bookmark-left: 100;\n}\n\n.pdl-btn {\n position: relative;\n border-radius: 4px;\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: 10;\n cursor: pointer;\n}\n\n.pdl-btn-main {\n margin: 0 0 0 10px;\n}\n\n.pdl-btn-sub {\n position: absolute;\n background-color: rgba(255, 255, 255, 0.5);\n top: calc((100% - 32px) * var(--pdl-btn-top) / 100);\n left: calc((100% - 32px) * var(--pdl-btn-left) / 100);\n}\n\n.pdl-btn-sub.artworks {\n position: sticky;\n top: 40px;\n left: 0px;\n}\n\n.pdl-btn-sub.presentation {\n position: fixed;\n top: 50px;\n right: 20px;\n left: auto;\n border-radius: 8px;\n}\n\n.pdl-btn-sub.self-bookmark {\n top: calc((100% - 32px) * var(--pdl-btn-self-bookmark-top) / 100);\n left: calc((100% - 32px) * var(--pdl-btn-self-bookmark-left) / 100);\n}\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\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\n.pdl-progress {\n background-image: none !important;\n cursor: default !important;\n}\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\n.pdl-progress:not(:empty):after {\n background: conic-gradient(#01b468 0, #01b468 var(--pdl-progress), transparent var(--pdl-progress), transparent);\n transition: --pdl-progress 0.2s ease;\n}\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$4,{});
var css$3 = ".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%) center/18px 2px no-repeat,\n linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/2px 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.pdl-dialog-content hr.vertical {\n height: 1.15em !important;\n border: none;\n border-left: 1px solid rgba(0, 0, 0, 0.1);\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: 7em;\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#pdl-setting-filename .pdl-options {\n display: flex;\n align-items: center;\n justify-content: space-between;\n cursor: pointer;\n padding: 0.6em 0;\n}\n\n#pdl-setting-filename .pdl-options:hover {\n background-color: rgba(0, 0, 0, 0.05);\n}\n\n#pdl-setting-filename .pdl-unavailable {\n pointer-events: none !important;\n opacity: 0.5 !important;\n cursor: not-allowed !important;\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/* adjust button position */\n.pdl-adjust-button {\n display: flex;\n justify-content: space-between;\n gap: 32px;\n}\n\n.pdl-adjust-content {\n flex: 1;\n}\n\n.pdl-adjust-content .pdl-adjust-item {\n margin-bottom: 1em;\n}\n\n.pdl-adjust-content .pdl-adjust-item .pdl-adjust-title {\n font-weight: 700;\n margin-bottom: 0.8em;\n}\n\n.pdl-adjust-content .pdl-adjust-item .pdl-adjust-select {\n display: flex;\n align-items: center;\n margin: 0.6em 0;\n padding: 0 0.4em;\n gap: 20px;\n}\n\n.pdl-adjust-select input[type='range'] {\n flex: 1;\n}\n\n.pdl-adjust-preview {\n align-self: center;\n}\n\n.pdl-adjust-preview .pdl-thumbnail-sample {\n position: relative;\n width: 184px;\n height: 184px;\n background-color: rgba(0, 150, 250, 0.15);\n border-radius: 4px;\n}\n";
n(css$3,{});
var css$2 = ".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$2,{});
var css$1 = "[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$1,{});
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))/,
searchPage: /\/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 langZh = {
downloadBar: {
button: {
stop: '停止',
works: '作品',
bookmarks: '收藏',
bookmarks_public: '公开',
bookmarks_private: '不公开',
all_one_page: '全部(单页)',
all: '全部(批量)',
r18_one_page: 'R-18(单页)',
r18: 'R-18(批量)'
},
filter: {
exclude_downloaded: '排除已下载图片',
illusts: '插画',
manga: '漫画',
ugoria: '动图'
}
},
modals: {
upgradeMsg: {
feedback: '有问题or想建议?这里反馈'
},
setting: {
filename: {
tab_title: '文件名',
input: {
folder_label: '文件夹名:',
folder_placeholder: '我不想保存到子文件夹',
folder_placeholder_vm: 'Violentmonkey不支持',
folder_placeholder_need_api: '需要Browser Api',
filename_label: '文件名:',
filename_placeholder: '你的名字?',
tag_label: '标签语言:',
fsa_label: '使用FileSystemAccess API',
fsa_placeholder: '根文件夹名',
filename_conflict_label: '文件名重复时:',
filename_conflict_option_uniquify: '重命名',
filename_conflict_option_overwrite: '覆盖',
filename_conflict_option_prompt: '提示'
},
tips: {
filename_pattern: '{artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}:作品ID, {page}:页码, {tags}:作品标签,{date} / {date(占位符)}: 创建时间',
empty_folder: '如果不想保存到画师目录,文件夹名留空即可。',
tag_translation: '请注意:标签翻译不一定是你选择的语言,部分<a href="https://crowdin.com/project/pixiv-tags" target="_blank">无对应语言翻译的标签</a>仍可能是其他语言。'
},
button: {
fsa_change_dir: '更改'
}
},
ugoria: {
tab_title: '动图',
format_label: '动图格式:'
},
history: {
tab_title: '历史记录',
button: {
import: '导入记录(替换)',
merge: '导入记录(合并)',
export: '导出记录',
clear: '清除记录'
},
tips: {
clear: '真的要清除历史记录吗?'
}
},
button_pos: {
tab_title: '按钮',
self_bookmark_title: '预览图(我的收藏)',
preview_title: '预览图',
input: {
horizon_label: '水平:',
vertical_label: '垂直:'
}
},
others: {
tab_title: '其它',
input: {
bundle_illusts: '将多页插图打包为.zip压缩包',
bundle_manga: '将多页漫画作品打包为.zip压缩包',
add_bookmark: '下载单个作品时收藏作品',
add_bookmark_with_tags: '收藏时添加作品标签',
add_bookmark_private_r18: '将R-18作品收藏到不公开类别',
show_popup_button: '显示设置按钮'
}
},
feedback: {
tab_title: '反馈 / 赞赏',
tips: {
feedback: '有问题or想建议?这里反馈'
}
}
}
},
gm_menu: {
setting: '设置'
}
};
const langEn = {
downloadBar: {
button: {
stop: 'Stop',
works: 'Works',
bookmarks: 'Bookmarks',
bookmarks_public: 'Public',
bookmarks_private: 'Private',
all_one_page: 'All (one page)',
all: 'All',
r18_one_page: 'R-18 (one page)',
r18: 'R-18'
},
filter: {
exclude_downloaded: 'Exclude downloaded',
illusts: 'Illustrations',
manga: 'Manga',
ugoria: 'Ugoria'
}
},
modals: {
upgradeMsg: {
feedback: 'Feedback'
},
setting: {
filename: {
tab_title: 'Filename',
input: {
folder_label: 'FileName:',
folder_placeholder: "I don't need subfolder",
folder_placeholder_vm: "VM doesn't support",
folder_placeholder_need_api: 'Need Browser Api',
filename_label: 'FileName:',
filename_placeholder: 'Your Name?',
tag_label: 'Tags language:',
fsa_label: 'FileSystemAccess API',
fsa_placeholder: 'Root directory',
filename_conflict_label: 'Conflict Action: ',
filename_conflict_option_uniquify: 'Uniquify',
filename_conflict_option_overwrite: 'Overwrite',
filename_conflict_option_prompt: 'Prompt'
},
tips: {
filename_pattern: '{artist}, {artistID}, {title}, {id}, {page}, {tags}, {date} / {date(format)}',
empty_folder: "If you don't need a subfolder, just leave the folder name blank",
tag_translation: '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.'
},
button: {
fsa_change_dir: 'Change'
}
},
ugoria: {
tab_title: 'Ugoria',
format_label: 'Ugoria Format:'
},
history: {
tab_title: 'History',
button: {
import: 'Import (Replace)',
merge: 'Import (Merge)',
export: 'Export',
clear: 'Clear'
},
tips: {
clear: 'Do you really want to clear history?'
}
},
button_pos: {
tab_title: 'Button',
self_bookmark_title: 'Thumbnail(My bookmarks)',
preview_title: 'Thumbnail',
input: {
horizon_label: 'X:',
vertical_label: 'Y:'
}
},
others: {
tab_title: 'Others',
input: {
bundle_illusts: 'Pack multi-page illustrations into a .zip archive',
bundle_manga: 'Pack manga into a .zip archive',
add_bookmark: 'Bookmark work when downloading a single work',
add_bookmark_with_tags: 'Add works tags',
add_bookmark_private_r18: 'Bookmark R-18 works to private category',
show_popup_button: 'Show setting button'
}
},
feedback: {
tab_title: 'Feedback',
tips: {
feedback: 'Feedback'
}
}
}
},
gm_menu: {
setting: '设置'
}
};
const messages = {
'zh-cn': langZh,
'zh-tw': langZh,
zh: langZh,
en: langEn
};
const curLang = document.documentElement.getAttribute('lang')?.toLowerCase() || 'en';
const defaultLang = 'en';
function t(key) {
const lang = (curLang in messages ? curLang : defaultLang);
const paths = key.split('.');
let last = messages[lang];
for (let i = 0; i < paths.length; i++) {
const value = last[paths[i]];
if (value === undefined || value === null)
return null;
last = value;
}
return last;
}
function createHistory() {
let records = (function getHistory() {
const storage = localStorage.pixivDownloader || '[]';
return new Set(JSON.parse(storage));
})();
function readHistoryFile(file, cb) {
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');
cb(history);
location.reload();
}
catch (error) {
alert(error.message);
}
};
}
else {
alert('Invalid file');
}
}
return {
add(pixivId) {
if (records.has(pixivId))
return;
records.add(pixivId);
localStorage.setItem(`pdlTemp-${pixivId}`, '');
},
has(pixivId) {
return records.has(pixivId);
},
getAll() {
return [...records];
},
updateHistory() {
const validKeys = Object.keys(localStorage).filter((key) => /(?<=^pdlTemp-)\d+$/.test(key));
if (!validKeys.length)
return;
validKeys.forEach((key) => {
const [id] = /(?<=^pdlTemp-)\d+$/.exec(key);
records.add(id);
localStorage.removeItem(key);
});
this.saveHistory();
},
saveHistory(historyArr) {
if (historyArr) {
if (historyArr.length && !historyArr.every((id) => typeof id === 'string')) {
throw new Error('Invalid id type');
}
this.updateHistory();
localStorage.pixivDownloader = JSON.stringify(historyArr);
}
else {
localStorage.pixivDownloader = JSON.stringify([...records]);
}
},
clearHistory() {
const isConfirm = confirm(t('modals.setting.history.tips.clear'));
if (!isConfirm)
return;
this.updateHistory();
records = new Set();
localStorage.pixivDownloader = '[]';
location.reload();
},
replace(file) {
readHistoryFile(file, this.saveHistory.bind(this));
},
merge(file) {
readHistoryFile(file, (historyArr) => {
if (!historyArr.length)
throw new Error('No id found');
if (!historyArr.every((id) => typeof id === 'string')) {
throw new Error('Invalid id type');
}
historyArr.forEach((id) => records.add(id));
this.saveHistory();
});
}
};
}
const pixivHistory = createHistory();
function getSelfId() {
return document.querySelector('#qualtrics_user-id')?.textContent ?? '';
}
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 getLogger() {
const methods = ['info', 'warn', 'error'];
const style = ['color: green;', 'color: orange;', 'color: red;'];
const logLevel = 2 ;
const namePrefix = '[Pixiv Downlaoder] ';
function log(level, args) {
if (logLevel <= level)
console[methods[level]]('%c[Pixiv Downloader]', style[level], ...args);
}
return {
info(...args) {
log(0 , args);
},
warn(...args) {
log(1 , args);
},
error(...args) {
log(2 , args);
},
time(label) {
console.time(namePrefix + label);
},
timeLog(label) {
console.timeLog(namePrefix + label);
},
timeEnd(label) {
console.timeEnd(namePrefix + label);
}
};
}
const logger = getLogger();
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 = {}));
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 replaceInvalidChar(str) {
if (typeof str !== 'string')
throw new TypeError('expect string but got ' + typeof str);
if (!str)
return '';
return str
.replace(/\p{C}/gu, '')
.replace(/\\/g, '\')
.replace(/\//g, '/')
.replace(/:/g, ':')
.replace(/\*/g, '*')
.replace(/\?/g, '?')
.replace(/\|/g, '|')
.replace(/"/g, '"')
.replace(/</g, '﹤')
.replace(/>/g, '﹥')
.replace(/~/g, '~')
.trim()
.replace(/^\.|\.$/g, '.');
}
function unescapeHtml(str) {
if (typeof str !== 'string')
throw new TypeError('expect string but got ' + typeof str);
if (!str)
return '';
const el = document.createElement('p');
el.innerHTML = str;
return el.innerText;
}
function stringToFragment(string) {
const renderer = document.createElement('template');
renderer.innerHTML = string;
return renderer.content;
}
function loadConfig() {
const defaultConfig = Object.freeze({
version: "0.9.4",
ugoriaFormat: 'zip',
folderPattern: 'pixiv/{artist}',
filenamePattern: '{artist}_{title}_{id}_p{page}',
tagLang: 'ja',
showMsg: true,
filterExcludeDownloaded: false,
filterIllusts: true,
filterManga: true,
filterUgoria: true,
bundleIllusts: false,
bundleManga: false,
addBookmark: false,
addBookmarkWithTags: false,
privateR18: false,
useFileSystemAccess: false,
fileSystemFilenameConflictAction: 'uniquify',
showPopupButton: true
});
const config = (() => {
if (!localStorage.pdlSetting)
return {};
let config;
try {
config = JSON.parse(localStorage.pdlSetting);
}
catch (error) {
console.log(error);
return {};
}
if (config.version !== defaultConfig.version) {
config.version = defaultConfig.version;
config.showMsg = true;
}
return config;
})();
return {
get(key) {
return config[key] ?? defaultConfig[key];
},
set(key, value) {
if (config[key] !== value) {
config[key] = value;
localStorage.pdlSetting = JSON.stringify(config);
logger.info('Config set:', key, value);
}
}
};
}
const config = loadConfig();
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();
};`;
class RequestError extends Error {
response;
constructor(message, response) {
super(message);
this.name = 'RequestError';
this.response = response;
}
}
class CancelError extends Error {
constructor() {
super('User aborted');
this.name = 'CancelError';
}
}
class JsonDataError extends Error {
constructor(msg) {
super(msg);
this.name = 'JsonDataError';
}
}
function createService() {
async function _requestJson(url, init) {
logger.info('fetch url:', url);
const res = await fetch(url, init);
if (!res.ok)
throw new RequestError('Request ' + url + ' failed with status code ' + res.status, res);
const data = await res.json();
if (data.error)
throw new JsonDataError(data.message);
return data.body;
}
async function _getDeps(url) {
return fetch(url).then((res) => {
if (res.ok)
return res.text();
throw new RequestError(`Fetch dependency ${url} failed with status code ${res.status}`, res);
});
}
return {
async getJson(url) {
let json;
let retry = 0;
const MAX_RETRY = 3;
do {
try {
json = await _requestJson(url);
}
catch (error) {
logger.error(error);
retry++;
if (retry === MAX_RETRY)
throw error;
sleep(3000);
}
} while (!json);
return json;
},
async getArtworkHtml(illustId) {
logger.info('Fetch illust:', illustId);
let params = '';
const tagLang = config.get('tagLang');
if (tagLang !== 'ja')
params = '?lang=' + tagLang;
const res = await fetch('https://www.pixiv.net/artworks/' + illustId + params);
if (!res.ok)
throw new RequestError('Request failed with status code ' + res.status, res);
return await res.text();
},
async getGifWS() {
let gifWS;
if (!(gifWS = await GM_getValue('gifWS'))) {
gifWS = await _getDeps(depsUrls.gifWorker);
GM_setValue('gifWS', gifWS);
}
return gifWS;
},
async 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.error('Convert Apng failed.');
postMessage(png);
};`;
apngWS = workerEvt + pako + upngScript;
GM_setValue('apngWS', apngWS);
}
return apngWS;
},
getWebpWS() {
return workerChunk__default["default"] + handleWorker;
},
addBookmark(illustId, token, tags = [], restrict = BookmarkRestrict.public) {
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
})
});
},
getFollowLatestWorks(page, mode = 'all') {
return _requestJson(`/ajax/follow_latest/illust?p=${page}&mode=${mode}&lang=jp`);
},
getUserAllProfile(userId) {
return _requestJson('/ajax/user/' + userId + '/profile/all');
},
getUgoriaMeta(illustId) {
return _requestJson('/ajax/illust/' + illustId + '/ugoira_meta');
},
getArtworkDetail(illustId) {
let params = '';
const tagLang = config.get('tagLang');
if (tagLang !== 'ja')
params = '?lang=' + tagLang;
return _requestJson('/ajax/illust/' + illustId + params);
}
};
}
const api = createService();
function addBookmark(pdlBtn, illustId, token, tags) {
if (!config.get('addBookmark'))
return;
api
.addBookmark(illustId, token, config.get('addBookmarkWithTags') ? tags : [], config.get('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) => {
logger.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 logger.error(new Error('Can not find bookmark button.'));
}
return bookmarkBtnRef;
}
const env = {
isFirefox() {
return navigator.userAgent.includes('Firefox');
},
isViolentmonkey() {
return GM_info.scriptHandler === 'Violentmonkey';
},
isTampermonkey() {
return GM_info.scriptHandler === 'Tampermonkey';
},
isBlobDlAvaliable() {
return !this.isFirefox() || (this.isFirefox() && this.isTampermonkey() && parseFloat(GM_info.version ?? '') < 4.18);
},
isSupportSubpath() {
return this.isBrowserDownloadMode();
},
isBrowserDownloadMode() {
return GM_info.downloadMode === 'browser';
},
isConflictActionEnable() {
return this.isTampermonkey() && parseFloat(GM_info.version ?? '') >= 4.18 && this.isBrowserDownloadMode();
},
isConflictActionPromptEnable() {
return !this.isFirefox() && this.isConflictActionEnable();
},
isFileSystemAccessAvaliable() {
return (typeof unsafeWindow.showDirectoryPicker === 'function' && typeof unsafeWindow.showSaveFilePicker === 'function');
}
};
function createCompressor() {
const zip = new JSZip__default["default"]();
return {
add(id, name, data) {
zip.folder(id)?.file(name, data);
},
bundle(id) {
const folder = zip.folder(id);
if (!folder)
throw new TypeError('no such folder:' + id);
return folder.generateAsync({ type: 'blob' });
},
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));
logger.info('Compressor: Remove', zip);
}
},
fileCount(id) {
let count = 0;
zip.folder(id)?.forEach(() => count++);
return count;
},
async unzip(data) {
const id = Math.random().toString(36);
let folder = zip.folder(id);
if (!folder)
throw TypeError('Can not get new root folder');
const filesPromises = [];
folder = await folder.loadAsync(data);
folder.forEach((_, file) => {
filesPromises.push(file.async('blob'));
});
const files = await Promise.all(filesPromises);
zip.remove(id);
return files;
}
};
}
const compressor = createCompressor();
function createConverter() {
const freeApngWorkers = [];
const freeWebpWorkers = [];
const MAX_CONVERT = 2;
let isStop = false;
let queue = [];
let active = [];
const cachedQueue = {
gif: [],
png: []
};
const depsUrl = {
gif: '',
png: '',
webp: URL.createObjectURL(new Blob([api.getWebpWS()], { type: 'text/javascript' }))
};
let LoadStatus;
(function (LoadStatus) {
LoadStatus[LoadStatus["unloaded"] = 0] = "unloaded";
LoadStatus[LoadStatus["loading"] = 1] = "loading";
LoadStatus[LoadStatus["loaded"] = 2] = "loaded";
})(LoadStatus || (LoadStatus = {}));
const depsStatus = {
gif: {
loaded: LoadStatus.unloaded,
load() {
logger.info('开始加载gif依赖');
this.loaded = LoadStatus.loading;
return api.getGifWS().then((str) => {
depsUrl.gif = URL.createObjectURL(new Blob([str], { type: 'text/javascript' }));
this.loaded = LoadStatus.loaded;
logger.info('加载gif依赖完成');
});
}
},
png: {
loaded: LoadStatus.unloaded,
load() {
logger.info('开始加载png依赖');
this.loaded = LoadStatus.loading;
return api.getApngWS().then((str) => {
depsUrl.png = URL.createObjectURL(new Blob([str], { type: 'text/javascript' }));
this.loaded = LoadStatus.loaded;
logger.info('加载png依赖完成');
});
}
}
};
const convertTo = {
webp: (frames, convertMeta) => {
return new Promise((resolve, reject) => {
let worker;
let reuse = false;
logger.time(convertMeta.id);
if (freeWebpWorkers.length) {
logger.info('Reuse webp workers.');
worker = freeWebpWorkers.shift();
reuse = true;
}
else {
worker = new Worker(depsUrl.webp);
}
convertMeta.abort = () => {
logger.timeEnd(convertMeta.id);
logger.warn('Convert stop manually.' + convertMeta.id);
reject(new CancelError());
convertMeta.isAborted = true;
worker.terminate();
};
const workerLoad = new Promise((onLoaded) => {
if (reuse)
return onLoaded();
worker.onmessage = (evt) => {
if (evt.data === 'ok') {
logger.info('Webp worker loaded.');
onLoaded();
}
};
});
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((onFulfilled, onRejected) => {
canvas.toBlob((blob) => {
if (!blob)
return onRejected(new TypeError('Convert failed when invoke canvas.toBlob() ' + idx));
blob.arrayBuffer().then((buffer) => {
const u8a = new Uint8Array(buffer);
onFulfilled(u8a);
convertMeta.onProgress?.((++completed / frames.length) * 0.5, 'webp');
});
}, 'image/webp', 1);
}));
});
workerLoad
.then(() => Promise.all(data))
.then((u8arrs) => {
if (convertMeta.isAborted)
return;
logger.timeLog(convertMeta.id);
worker.onmessage = (evt) => {
const data = evt.data;
if (typeof data !== 'object') {
convertMeta.onProgress?.(0.5 + (evt.data / frames.length) * 0.5, 'webp');
}
else {
logger.timeEnd(convertMeta.id);
freeWebpWorkers.push(worker);
resolve(new Blob([evt.data], { type: 'image/webp' }));
}
};
worker.postMessage({ data: u8arrs, delays });
}, (reason) => {
logger.timeLog(convertMeta.id);
reject(reason);
});
});
},
gif: (frames, convertMeta) => {
return new Promise((resolve, reject) => {
const gif = new GIF__default["default"]({
workers: 2,
quality: 10,
workerScript: depsUrl.gif
});
convertMeta.abort = () => {
gif.abort();
};
logger.info('Start convert:', convertMeta.id);
logger.time(convertMeta.id);
frames.forEach((frame, i) => {
gif.addFrame(frame, {
delay: convertMeta.source.framesInfo[i].delay
});
});
gif.on('progress', (progress) => {
if (typeof convertMeta.onProgress === 'function')
convertMeta.onProgress(progress, 'gif');
});
gif.on('finished', (gifBlob) => {
logger.timeEnd(convertMeta.id);
resolve(gifBlob);
});
gif.on('abort', () => {
logger.timeEnd(convertMeta.id);
logger.warn('Convert stop manually. ' + convertMeta.id);
convertMeta.isAborted = true;
reject(new CancelError());
});
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(new TypeError('Can not get canvas context'));
const data = [];
const delay = convertMeta.source.framesInfo.map((frameInfo) => {
return Number(frameInfo.delay);
});
logger.info('Start convert:', convertMeta.id);
logger.time(convertMeta.id);
for (const frame of frames) {
if (convertMeta.isAborted) {
logger.timeEnd(convertMeta.id);
logger.warn('Convert stop manually. ' + convertMeta.id);
return reject(new CancelError());
}
context.clearRect(0, 0, width, height);
context.drawImage(frame, 0, 0, width, height);
data.push(context.getImageData(0, 0, width, height).data);
}
logger.timeLog(convertMeta.id);
let worker;
if (freeApngWorkers.length) {
worker = freeApngWorkers.shift();
logger.info('Reuse apng workers.');
}
else {
worker = new Worker(depsUrl.png);
}
convertMeta.abort = () => {
logger.timeEnd(convertMeta.id);
logger.warn('Convert stop manually. ' + convertMeta.id);
reject(new CancelError());
convertMeta.isAborted = true;
worker.terminate();
};
worker.onmessage = function (e) {
freeApngWorkers.push(worker);
logger.timeEnd(convertMeta.id);
if (!e.data) {
return reject(new TypeError('Failed to get png data. ' + 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(new TypeError('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) {
logger.warn('Convert stop manually.' + convertMeta.id);
return reject(new CancelError());
}
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');
compressor
.unzip(source.data)
.then((files) => {
const imagePromises = files.map((blob) => {
return new Promise((resolve) => {
const image = new Image();
image.onload = () => {
resolve(image);
};
image.src = URL.createObjectURL(blob);
});
});
return Promise.all(imagePromises);
})
.then((imgEles) => {
frames = imgEles;
if (convertMeta.isAborted) {
logger.warn('Convert stop manually.' + id);
throw new CancelError();
}
return convertTo[source.format](frames, convertMeta);
})
.then(resolve, 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) => {
logger.info('Converter add', convertSource.id);
return new Promise((resolve, reject) => {
const meta = {
id: convertSource.id,
isAborted: false,
source: convertSource,
onProgress: handler?.onProgress,
resolve,
reject,
abort() {
this.isAborted = true;
}
};
const format = convertSource.format;
if ((format === 'gif' || format === 'png') && depsStatus[format].loaded !== LoadStatus.loaded) {
switch (depsStatus[format].loaded) {
case LoadStatus.unloaded:
cachedQueue[format].push(meta);
depsStatus[format].load().then(() => {
logger.info(`添加${cachedQueue[format].length}个任务`);
queue.push(...cachedQueue[format]);
cachedQueue[format].length = 0;
while (active.length < MAX_CONVERT && queue.length && !isStop) {
convert(queue.shift());
}
}, (reason) => {
cachedQueue[format].forEach((meta) => {
meta.reject(reason);
});
depsStatus[format].loaded = LoadStatus.unloaded;
cachedQueue[format].length = 0;
});
break;
case LoadStatus.loading:
cachedQueue[format].push(meta);
break;
default:
throw new RangeError('Invalid deps status.');
}
}
else {
queue.push(meta);
while (active.length < MAX_CONVERT && queue.length && !isStop) {
convert(queue.shift());
}
}
});
},
del: (taskIds) => {
if (!taskIds.length)
return;
logger.info('Converter del, active:', active.map((meta) => meta.id), 'queue:', queue.map((meta) => meta.id));
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 converter = createConverter();
const updateDirHandleChannel = new BroadcastChannel('update_dir_channel');
updateDirHandleChannel.onmessage = (evt) => {
const data = evt.data;
switch (data.kind) {
case 1 :
dirHandleStatus = 1 ;
logger.info('正在选择目录');
break;
case 0 :
logger.warn('取消更新dirHandle');
if (dirHandle) {
dirHandleStatus = 2 ;
processCachedSave();
}
else {
dirHandleStatus = 0 ;
rejectCachedSave();
}
break;
case 2 :
dirHandleStatus = 2 ;
dirHandle = data.handle;
logger.info('更新dirHandle', dirHandle);
processCachedSave();
break;
case 'request':
if (dirHandle) {
updateDirHandleChannel.postMessage({
kind: 'response',
handle: dirHandle
});
logger.info('响应请求dirHandle');
}
break;
case 'response':
if (!dirHandle) {
if (dirHandleStatus === 0 )
dirHandleStatus = 2 ;
dirHandle = data.handle;
logger.info('首次获取dirHandle', dirHandle);
}
break;
default:
throw new Error('Invalid data kind.');
}
};
updateDirHandleChannel.postMessage({ kind: 'request' });
async function getDirHandleRecursive(dirs) {
let handler = dirHandle;
if (typeof dirs === 'string') {
if (dirs.indexOf('/') === -1)
return await handler.getDirectoryHandle(dirs, { create: true });
dirs = dirs.split('/').filter((dir) => !!dir);
}
for await (const dir of dirs) {
handler = await handler.getDirectoryHandle(dir, { create: true });
}
return handler;
}
const duplicateFilenameCached = {};
async function getFilenameHandle(dirHandle, filename) {
const conflictAction = config.get('fileSystemFilenameConflictAction');
if (conflictAction === 'overwrite')
return await dirHandle.getFileHandle(filename, { create: true });
if (!(filename in duplicateFilenameCached)) {
duplicateFilenameCached[filename] = [];
try {
await dirHandle.getFileHandle(filename);
logger.warn('存在同名文件', filename);
}
catch (error) {
return await dirHandle.getFileHandle(filename, { create: true });
}
}
const extIndex = filename.lastIndexOf('.');
const ext = filename.slice(extIndex + 1);
const name = filename.slice(0, extIndex);
if (conflictAction === 'prompt') {
return await unsafeWindow.showSaveFilePicker({
suggestedName: filename,
types: [{ description: 'Image file', accept: { ['image/' + ext]: ['.' + ext] } }]
});
}
else {
for (let suffix = 1; suffix < 1000; suffix++) {
const newName = `${name} (${suffix}).${ext}`;
try {
await dirHandle.getFileHandle(newName);
}
catch (error) {
if (duplicateFilenameCached[filename].includes(newName)) {
continue;
}
else {
duplicateFilenameCached[filename].push(newName);
}
logger.info('使用文件名:', newName);
return await dirHandle.getFileHandle(newName, { create: true });
}
}
throw new RangeError('Oops, you have too many duplicate files.');
}
}
function clearFilenameCached(duplicateName, actualName) {
if (!(duplicateName in duplicateFilenameCached))
return;
const usedNameArr = duplicateFilenameCached[duplicateName];
logger.info('清理重名文件名', usedNameArr, actualName);
if (usedNameArr.length === 0) {
delete duplicateFilenameCached[duplicateName];
return;
}
const index = usedNameArr.indexOf(actualName);
if (index === -1)
return;
usedNameArr.splice(index, 1);
if (usedNameArr.length === 0)
delete duplicateFilenameCached[duplicateName];
}
async function updateDirHandle() {
try {
dirHandleStatus = 1 ;
updateDirHandleChannel.postMessage({ kind: 1 });
dirHandle = await unsafeWindow.showDirectoryPicker({ id: 'pdl', mode: 'readwrite' });
logger.info('更新dirHandle', dirHandle);
dirHandleStatus = 2 ;
updateDirHandleChannel.postMessage({
kind: 2 ,
handle: dirHandle
});
processCachedSave();
return true;
}
catch (error) {
logger.warn(error);
updateDirHandleChannel.postMessage({ kind: 0 });
if (dirHandle) {
dirHandleStatus = 2 ;
processCachedSave();
}
else {
dirHandleStatus = 0 ;
rejectCachedSave();
}
return false;
}
}
let dirHandleStatus = 0 ;
let dirHandle;
const cachedSaveProcess = [];
async function saveWithFileSystemAccess(blob, downloadMeta) {
try {
if (downloadMeta.state === 0 )
return;
if (dirHandleStatus === 1 ) {
cachedSaveProcess.push([blob, downloadMeta]);
return;
}
if (dirHandleStatus === 0 ) {
const isSuccess = await updateDirHandle();
if (!isSuccess)
throw new TypeError('Failed to get dir handle.');
}
let currenDirHandle;
let filename;
const path = downloadMeta.source.path;
const index = path.lastIndexOf('/');
if (index === -1) {
filename = path;
currenDirHandle = dirHandle;
}
else {
filename = path.slice(index + 1);
currenDirHandle = await getDirHandleRecursive(path.slice(0, index));
}
const fileHandle = await getFilenameHandle(currenDirHandle, filename);
const writableStream = await fileHandle.createWritable();
await writableStream.write(blob);
await writableStream.close();
clearFilenameCached(filename, fileHandle.name);
downloadMeta.resolve(downloadMeta.taskId);
logger.info('Download complete:', downloadMeta.source.path);
}
catch (error) {
downloadMeta.reject(error);
logger.error(error);
}
downloadMeta.state = 2 ;
}
function processCachedSave() {
cachedSaveProcess.forEach((args) => saveWithFileSystemAccess(...args));
logger.info(`执行${cachedSaveProcess.length}个缓存任务`);
cachedSaveProcess.length = 0;
}
function rejectCachedSave() {
cachedSaveProcess.forEach(([, downloadMeta]) => downloadMeta.reject(new CancelError()));
logger.info(`取消${cachedSaveProcess.length}个缓存任务`);
cachedSaveProcess.length = 0;
}
function getCurrentDirName() {
return dirHandle?.name ?? '';
}
function isShouldGetDirHandle() {
return isUseFileSystemAccess() && dirHandleStatus === 0 ;
}
function isUseFileSystemAccess() {
return env.isFileSystemAccessAvaliable() && config.get('useFileSystemAccess');
}
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.state = 2 ;
downloadMeta.resolve(downloadMeta.taskId);
};
const _saveWithSubpath = (blob, downloadMeta) => {
const imgUrl = URL.createObjectURL(blob);
const request = {
url: imgUrl,
name: downloadMeta.source.path,
onerror: (error) => {
if (downloadMeta.state !== 0 ) {
downloadMeta.reject(new Error(`Download error when saving ${downloadMeta.source.path} because ${error.error} ${error.details ?? ''} `));
}
URL.revokeObjectURL(imgUrl);
},
onload: () => {
if (typeof downloadMeta.onLoad === 'function')
downloadMeta.onLoad();
URL.revokeObjectURL(imgUrl);
downloadMeta.state = 2 ;
downloadMeta.resolve(downloadMeta.taskId);
logger.info('Download complete:', downloadMeta.source.path);
}
};
downloadMeta.abort = GM_download(request).abort;
};
function createDownloader() {
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 {
logger.warn('Download function not full support:', GM_info.scriptHandler, GM_info.version);
save = _saveWithoutSubpath;
}
const download = (downloadMeta) => {
const { taskId, source } = downloadMeta;
logger.info('Start download:', source.path);
active.push(downloadMeta);
let fileSaveFn = save;
const isUseFSA = isUseFileSystemAccess();
if (isUseFSA) {
fileSaveFn = saveWithFileSystemAccess;
}
let abortObj;
const errorHandler = errorHandlerFactory(downloadMeta);
if ((!env.isBlobDlAvaliable() || (env.isViolentmonkey() && !isUseFSA)) && !('kind' in source)) {
abortObj = GM_download({
url: source.src,
name: source.path,
headers: {
referer: 'https://www.pixiv.net'
},
ontimeout: errorHandler,
onerror: errorHandler,
onload: async () => {
logger.info('Download complete:', taskId, 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) => {
logger.info('Xhr complete:', source.path);
if (downloadMeta.state === 0 )
return logger.warn('Download was canceled.', taskId, source.path);
if (!('kind' in source)) {
fileSaveFn(e.response, downloadMeta);
}
else if (source.kind === 'convert') {
const convertSource = {
id: taskId,
data: e.response,
format: source.format,
framesInfo: source.ugoiraMeta?.frames
};
converter.add(convertSource, { onProgress: downloadMeta.onProgress }).then((blob) => {
fileSaveFn(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) => {
fileSaveFn(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 = abortObj.abort;
};
function errorHandlerFactory(downloadMeta) {
return function (error) {
const { taskId, source, state } = downloadMeta;
if (state === 0 )
return;
if (error) {
logger.error('Download ' + taskId + ' error', error.error ? ' with reason: ' + error.error : '', 'details' in error ? error.details : error);
if ('status' in error && error.status === 429) {
downloadMeta.reject(new RequestError('Too many request', error));
active.splice(active.indexOf(downloadMeta), 1);
return;
}
}
else {
logger.warn('Download timeout:', source.src);
}
downloadMeta.retry++;
if (downloadMeta.retry > MAX_RETRY) {
downloadMeta.reject(new Error('Download failed: ' + source.src));
active.splice(active.indexOf(downloadMeta), 1);
if (queue.length && !isStop)
download(queue.shift());
}
else {
logger.info('Download retry:', downloadMeta.retry, source.src);
download(downloadMeta);
}
};
}
return {
add(metas, handler = {}) {
logger.info('Downloader add:', metas);
if (metas.length < 1)
return Promise.resolve('');
const promises = [];
metas.forEach((source) => {
promises.push(new Promise((resolve, reject) => {
const 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);
},
del(taskIds) {
if (!taskIds.length)
return;
isStop = true;
logger.info('Downloader delete. active:', active.length, 'queue', queue.length);
active = active.filter((downloadMeta) => {
if (taskIds.includes(downloadMeta.taskId) && downloadMeta.state !== 2 ) {
downloadMeta.abort?.();
downloadMeta.state = 0 ;
downloadMeta.reject(new CancelError());
logger.warn('Download abort manually.', downloadMeta.source.path);
}
else {
return true;
}
});
converter.del(taskIds);
compressor.remove(taskIds);
queue = queue.filter((downloadMeta) => !taskIds.includes(downloadMeta.taskId));
isStop = false;
while (active.length < MAX_DOWNLOAD && queue.length) {
download(queue.shift());
}
}
};
}
const downloader = createDownloader();
const needBundle = (type) => {
return ((type === IllustType.manga && config.get('bundleManga')) ||
(type === IllustType.illusts && config.get('bundleIllusts')));
};
const getFilePath = ({ user, userId, title, tagStr, illustId, createDate, page, ext }, option = { needBundle: false, needConvert: false }) => {
let pathPattern;
const folderPattern = config.get('folderPattern');
const filenamePattern = config.get('filenamePattern');
const isUseFSA = isUseFileSystemAccess();
const shouldAddFolder = !!folderPattern &&
!option.needBundle &&
((env.isSupportSubpath() && (!option.needConvert || env.isBlobDlAvaliable())) || isUseFSA);
if (shouldAddFolder) {
pathPattern = folderPattern + '/' + filenamePattern;
}
else {
pathPattern = filenamePattern;
}
if (option.needBundle && !filenamePattern.includes('{page}')) {
pathPattern += '_{page}';
}
function replaceDate(match, p1) {
const format = p1 || 'YYYY-MM-DD';
return dayjs__default["default"](createDate).format(format);
}
return (pathPattern
.replaceAll(/\{date\((.*?)\)\}|\{date\}/g, replaceDate)
.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 = config.get('tagLang') === 'ja' ? cur.tag : cur.translation?.['en'] || cur.tag;
if (index < tagsArr.length - 1) {
return prev + tag + '_';
}
else {
return prev + tag;
}
};
function isValidIllustType(illustType, option) {
switch (illustType) {
case IllustType.illusts:
if (option.filterIllusts)
return true;
break;
case IllustType.manga:
if (option.filterManga)
return true;
break;
case IllustType.ugoira:
if (option.filterUgoria)
return true;
break;
default:
throw new Error('Invalid filter type');
}
return false;
}
function filterWorks(works, option) {
const obj = {
unavaliable: [],
avaliable: [],
invalid: []
};
works.forEach((work) => {
if (!work.isBookmarkable) {
obj.unavaliable.push(work.id);
}
else if (option.filterExcludeDownloaded && pixivHistory.has(work.id)) {
obj.invalid.push(work.id);
}
else if (!isValidIllustType(work.illustType, option)) {
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) {
logger.info('getFollowLatestGenerator: got duplicate works');
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 profile = await api.getUserAllProfile(userId);
let illustIds = [];
let mangaIds = [];
if ((filterOption.filterIllusts || filterOption.filterUgoria) && typeof profile.illusts === 'object') {
illustIds.push(...Object.keys(profile.illusts).reverse());
}
if (filterOption.filterManga && typeof profile.manga === 'object') {
mangaIds.push(...Object.keys(profile.manga).reverse());
}
if (filterOption.filterExcludeDownloaded) {
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).reverse(), 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).reverse(), 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('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('Fail to parse global data.');
const globalData = JSON.parse(globalDataText[1]);
let ugoiraMeta;
if (illustData.illustType === IllustType.ugoira) {
ugoiraMeta = await api.getUgoriaMeta(illustId);
}
return {
illustData,
globalData,
ugoiraMeta
};
}
async function getAjaxArtworkData(illustId) {
const illustData = await api.getArtworkDetail(illustId);
let ugoiraMeta;
if (illustData.illustType === IllustType.ugoira) {
ugoiraMeta = await api.getUgoriaMeta(illustId);
}
return {
illustData,
ugoiraMeta
};
}
function getDownloadSource(artworkData, seletedPage) {
const { illustData, ugoiraMeta } = artworkData;
const { illustType, userName, userId, illustTitle, illustId, tags, pageCount, createDate } = illustData;
const pathInfo = {
user: replaceInvalidChar(unescapeHtml(userName)) || 'userId-' + userId,
title: replaceInvalidChar(unescapeHtml(illustTitle)) || 'illustId-' + illustId,
tagStr: replaceInvalidChar(unescapeHtml(tags.tags.reduce(makeTagsStr, ''))),
illustId,
userId,
createDate,
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) {
const ugoriaFormat = config.get('ugoriaFormat');
pathInfo.ext = '.' + ugoriaFormat;
if (ugoriaFormat !== 'zip') {
metas.push({
kind: 'convert',
format: ugoriaFormat,
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,
getAjaxArtworkData
};
function handleDownload(pdlBtn, illustId) {
if (isShouldGetDirHandle())
updateDirHandle();
let pageCount;
let pageComplete = 0;
let shouldDownloadPage;
let downloading = true;
const pageAttr = pdlBtn.getAttribute('should-download');
if (pageAttr) {
shouldDownloadPage = Number(pageAttr);
}
const onProgress = (progress = 0, type = null) => {
if (pageCount > 1 || !downloading)
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 || !downloading)
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)
logger.error(err);
pdlBtn.classList.remove('pdl-complete');
pdlBtn.classList.add('pdl-error');
})
.finally(() => {
downloading = false;
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 createThumbnailsBtn(nodes) {
let isSelfBookmark = false;
const inBookmarkPage = regexp.bookmarkPage.exec(location.pathname);
if (inBookmarkPage) {
inBookmarkPage[1] === getSelfId() && (isSelfBookmark = true);
}
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 (isSelfBookmark)
attrs.classList.push('self-bookmark');
e.appendChild(createPdlBtn(attrs));
}
}
});
}
function fixPixivPreviewer(nodes) {
const isPpSearchPage = regexp.searchPage.test(location.pathname);
if (!isPpSearchPage)
return;
nodes.forEach((node) => {
const pdlEle = node.querySelector('.pdl-btn');
if (!pdlEle)
return false;
pdlEle.remove();
});
}
function getFilterOption() {
return {
filterExcludeDownloaded: config.get('filterExcludeDownloaded'),
filterIllusts: config.get('filterIllusts'),
filterManga: config.get('filterManga'),
filterUgoria: config.get('filterUgoria')
};
}
function downloadAndRetry(chunksGenerators) {
useDownloadBar(chunksGenerators).then((failed) => {
if (failed instanceof Array && failed.length) {
const gen = async function* () {
yield {
avaliable: failed,
unavaliable: [],
invalid: []
};
};
console.log('[Pixiv Downloader] Retry...');
useDownloadBar({ total: failed.length, generator: gen() });
}
});
}
function downloadWorks(evt) {
evt.preventDefault();
evt.stopPropagation();
if (isDownloading)
return;
const btn = evt.target;
const userId = btn.getAttribute('pdl-userid');
const filterOption = getFilterOption();
if (isShouldGetDirHandle())
updateDirHandle();
const ids = parser.getAllWorksGenerator(userId, filterOption);
downloadAndRetry(ids);
}
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');
if (isShouldGetDirHandle())
updateDirHandle();
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);
}
downloadAndRetry(idsGenerators);
}
function downloadFollowLatest(evt) {
evt.preventDefault();
evt.stopPropagation();
if (isDownloading)
return;
const btn = evt.target;
const mode = location.pathname.includes('r18') ? 'r18' : 'all';
const filterOption = getFilterOption();
let idsGenerators;
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);
}
downloadAndRetry(idsGenerators);
}
function downloadSearchResult(evt) {
evt.preventDefault();
evt.stopPropagation();
if (isDownloading)
return;
const pdlNodes = document.querySelectorAll('section ul li button.pdl-btn-sub');
if (!pdlNodes.length)
return;
let ids = Array.prototype.map.call(pdlNodes, (node) => node.getAttribute('pdl-id'));
if (getFilterOption().filterExcludeDownloaded) {
ids = ids.filter((id) => !pixivHistory.has(id));
}
const idsGenerators = {
total: ids.length,
generator: (async function* () {
yield {
avaliable: ids,
unavaliable: [],
invalid: []
};
})()
};
downloadAndRetry(idsGenerators);
}
const dlBarRef = {
filter: {
filterExcludeDownloaded: undefined,
filterIllusts: undefined,
filterManga: undefined,
filterUgoria: undefined
},
statusBar: undefined,
abortBtn: undefined
};
function updateStatus(str) {
dlBarRef.statusBar && (dlBarRef.statusBar.textContent = str);
}
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 = config.get(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');
config.set(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', 'filterExcludeDownloaded', t('downloadBar.filter.exclude_downloaded')));
wrapper.appendChild(createFilterEl('pdl-filter-illusts', 'filterIllusts', t('downloadBar.filter.illusts')));
wrapper.appendChild(createFilterEl('pdl-filter-manga', 'filterManga', t('downloadBar.filter.manga')));
wrapper.appendChild(createFilterEl('pdl-filter-ugoria', 'filterUgoria', t('downloadBar.filter.ugoria')));
return wrapper;
}
function createExcludeDownloadedFilter() {
const wrapper = document.createElement('div');
wrapper.classList.add('pdl-filter-wrap');
wrapper.appendChild(createFilterEl('pdl-filter-exclude_downloaded', 'filterExcludeDownloaded', t('downloadBar.filter.exclude_downloaded')));
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']
}, t('downloadBar.button.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']
}, t('downloadBar.button.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']
}, t('downloadBar.button.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']
}, t('downloadBar.button.bookmarks'), { addEvent: false }));
dlBar.appendChild(createPdlBtn({
attrs: {
'pdl-userid': userId,
category: 'bookmarks',
rest: 'show'
},
classList: [...baseClasses, 'pdl-btn-all']
}, t('downloadBar.button.bookmarks_public'), { addEvent: false }));
dlBar.appendChild(createPdlBtn({
attrs: {
'pdl-userid': userId,
category: 'bookmarks',
rest: 'hide'
},
classList: [...baseClasses, 'pdl-btn-all']
}, t('downloadBar.button.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 updateFollowLatestDownloadBarBtnText(prevDlBtn, prevDlAllBtn) {
if (location.pathname.includes('r18') && prevDlBtn.textContent !== t('downloadBar.button.r18_one_page')) {
prevDlBtn.textContent = t('downloadBar.button.r18_one_page');
prevDlAllBtn.textContent = t('downloadBar.button.r18');
}
else if (!location.pathname.includes('r18') && prevDlBtn.textContent !== t('downloadBar.button.all_one_page')) {
prevDlBtn.textContent = t('downloadBar.button.all_one_page');
prevDlAllBtn.textContent = t('downloadBar.button.all');
}
}
function createFollowLatestDownloadBar() {
const prevDlBtn = document.querySelector('.pdl-btn-all');
if (prevDlBtn) {
const prevDlAllBtn = document.querySelector('.pdl-dl-all');
updateFollowLatestDownloadBarBtnText(prevDlBtn, prevDlAllBtn);
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']
}, t('downloadBar.button.stop'), { addEvent: false }));
const dlBtn = createPdlBtn({
attrs: { 'pdl-userid': '' },
classList: [...baseClasses, 'pdl-btn-all']
}, t('downloadBar.button.works'), { addEvent: false });
dlBtn.addEventListener('click', downloadFollowLatest);
dlBar.appendChild(dlBtn);
const dlAllBtn = createPdlBtn({
attrs: { 'pdl-userid': '' },
classList: [...baseClasses, 'pdl-btn-all', 'pdl-dl-all']
}, t('downloadBar.button.works'), { addEvent: false });
dlAllBtn.addEventListener('click', downloadFollowLatest);
dlBar.appendChild(dlAllBtn);
navBar.insertBefore(dlBar, modeSwitch);
}
function createSearchDownloadbar() {
if (document.querySelector('.pdl-dlbar'))
return;
const sections = document.querySelectorAll('section');
const worksSection = sections[sections.length - 1];
const styleRefEle = document.querySelector('nav a:not([aria-current])');
if (!worksSection || !styleRefEle)
return;
const dlBarContainer = worksSection.firstElementChild.firstElementChild;
const dlBar = document.createElement('div');
dlBar.classList.add('pdl-dlbar');
dlBar.classList.add('pdl-dlbar-search');
const statusBar = document.createElement('div');
statusBar.classList.add('pdl-dlbar-status_bar');
dlBarRef.statusBar = dlBar.appendChild(statusBar);
const baseClasses = styleRefEle.classList;
dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
attrs: { 'pdl-userid': '' },
classList: [...baseClasses, 'pdl-stop', 'pdl-hide']
}, t('downloadBar.button.stop'), { addEvent: false }));
const dlBtn = createPdlBtn({
attrs: { 'pdl-userid': '' },
classList: [...baseClasses, 'pdl-btn-all']
}, t('downloadBar.button.all_one_page'), { addEvent: false });
dlBtn.addEventListener('click', downloadSearchResult);
const filter = createExcludeDownloadedFilter();
dlBarContainer.parentElement.insertBefore(filter, dlBarContainer);
dlBar.appendChild(dlBtn);
dlBarContainer.appendChild(dlBar);
}
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');
}
function onProgressCB(progressData) {
if (typeof progressData === 'string') {
updateStatus(progressData);
}
else {
logger.info('Update progress by', progressData.illustId, ', completed: ', progressData.completed);
updateStatus(`Downloading: ${progressData.completed} / ${progressData.avaliable}`);
}
}
async function downloadByIds(total, idsGenerators, signal, onProgress) {
signal.throwIfAborted();
const failed = [];
const unavaliable = [];
const invalid = [];
const tasks = [];
let completed = 0;
let tooManyRequests = false;
let wakeTooManyRequest;
let wakeInterval;
let resolve;
let reject;
const done = new Promise((r, j) => {
resolve = r;
reject = j;
});
signal.addEventListener('abort', () => {
if (tasks.length) {
downloader.del(tasks);
tasks.length = 0;
}
wakeTooManyRequest?.();
wakeInterval?.();
reject(signal.aborted ? signal.reason : 'unexpected generator error');
}, { once: true });
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) {
for await (const ids of idsGenerator) {
logger.info('Got ids:', ids);
signal.throwIfAborted();
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) {
signal.throwIfAborted();
if (tooManyRequests) {
onProgress('Too many requests, wait 30s');
const { wake, sleep } = wakeableSleep(30000);
wakeTooManyRequest = wake;
await sleep;
signal.throwIfAborted();
tooManyRequests = false;
onProgress('Downloading...');
}
parser
.getAjaxArtworkData(id)
.then((data) => {
signal.throwIfAborted();
const sources = parser.getDownloadSource(data);
tasks.push(sources[0].taskId);
return downloader.add(sources);
})
.then((taskId) => {
pixivHistory.add(id);
if (!signal.aborted) {
tasks.splice(tasks.indexOf(taskId), 1);
completed++;
afterEach(id);
}
}, (reason) => {
if (!signal.aborted) {
reason && logger.error(reason);
if (reason instanceof RequestError && reason.response.status === 429) {
tooManyRequests = true;
}
if (reason instanceof JsonDataError) {
unavaliable.push(id);
}
else {
failed.push(id);
}
afterEach(id);
}
});
const { wake, sleep } = wakeableSleep(1000);
wakeInterval = wake;
await sleep;
}
}
else {
afterEach('no avaliable id');
}
}
}
}
catch (error) {
if (!signal.aborted) {
done.catch((reason) => {
logger.info('catch unexpected abort: ', reason);
});
signal.dispatchEvent(new Event('abort'));
throw error;
}
}
return done;
}
let isDownloading = false;
async function useDownloadBar(chunksGenerators) {
if (!dlBarRef.abortBtn)
return;
let total = 0;
let failedResult;
const idsGenerators = [];
!Array.isArray(chunksGenerators) && (chunksGenerators = [chunksGenerators]);
isDownloading = true;
changeDlbarDisplay();
try {
await Promise.all(chunksGenerators).then((gens) => {
gens.forEach((val) => {
total += val.total;
idsGenerators.push(val.generator);
});
});
}
catch (error) {
logger.error(error);
updateStatus('Network error, see console');
changeDlbarDisplay();
isDownloading = false;
return;
}
if (total === 0) {
updateStatus('No works');
}
else {
try {
logger.info('Total works:', total);
const controller = new AbortController();
const signal = controller.signal;
!signal.throwIfAborted &&
(signal.throwIfAborted = function () {
if (this.aborted) {
throw this.reason;
}
});
if (!('reason' in signal)) {
const abort = controller.abort;
controller.abort = function (reason) {
this.signal.reason = reason ? reason : new DOMException('signal is aborted without reason');
abort.apply(this);
};
}
dlBarRef.abortBtn?.addEventListener('click', () => {
controller.abort();
}, { once: true });
const { failed, unavaliable } = await downloadByIds(total, idsGenerators, signal, 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 {
console.log('[Pixiv Downloader] Download complete');
updateStatus('Complete');
}
}
catch (error) {
if (error instanceof DOMException) {
updateStatus('Stop');
}
else {
updateStatus('Error, see console');
logger.error(error);
}
}
}
changeDlbarDisplay();
isDownloading = false;
return failedResult;
}
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 logger.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 createToolbarBtn(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 createWorkScrollBtn(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(() => {
logger.info('Start to find modal.');
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 onPageTypeChange() {
const pathname = location.pathname;
let param;
switch (true) {
case !!(param = regexp.artworksPage.exec(pathname)): {
const id = param[1];
createToolbarBtn(id);
createWorkScrollBtn(id);
createPresentationBtn(id);
createPreviewModalBtn();
break;
}
case !!(param = regexp.userPage.exec(pathname)): {
const id = param[1];
createDownloadBar(id);
const matchTag = regexp.userPageTags.exec(pathname);
if (matchTag) {
createTagsBtn(id, matchTag[1]);
}
break;
}
case regexp.followLatest.test(pathname): {
createFollowLatestDownloadBar();
break;
}
case regexp.searchPage.test(pathname): {
createSearchDownloadbar();
break;
}
default:
removeDownloadBar();
break;
}
}
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) {
createThumbnailsBtn(document.querySelectorAll('a'));
firstRun = false;
}
else {
fixPixivPreviewer(addedNodes);
const thunmnails = addedNodes.reduce((prev, current) => {
return prev.concat(Array.from(current.querySelectorAll('a')));
}, []);
createThumbnailsBtn(thunmnails);
}
onPageTypeChange();
}
function createModal(args, option = { closeOnClickModal: false }) {
const modalHtml = `
<div class="pdl-modal">
<div class="pdl-dialog">
<button class="pdl-dialog-close"></button>
<header class="pdl-dialog-header"></header>
<div class="pdl-dialog-content"></div>
<footer class="pdl-dialog-footer"></footer>
</div>
</div>`;
const fragment = stringToFragment(modalHtml);
const modal = fragment.querySelector('.pdl-modal');
const dialog = modal.querySelector('.pdl-dialog');
const closeBtn = modal.querySelector('.pdl-dialog-close');
const header = modal.querySelector('.pdl-dialog-header');
const content = modal.querySelector('.pdl-dialog-content');
const footer = modal.querySelector('.pdl-dialog-footer');
if (option.closeOnClickModal) {
dialog.addEventListener('click', (e) => {
e.stopPropagation();
});
modal.addEventListener('click', () => {
modal.remove();
});
}
closeBtn.addEventListener('click', () => {
modal.remove();
});
args.header && header.appendChild(args.header);
args.footer && footer.appendChild(args.footer);
content.appendChild(args.content);
return fragment;
}
function showUpgradeMsg() {
const headerHtml = `<h3>Pixiv Downloader ${config.get('version')}</h3>`;
const contentHtml = `
<div class="pdl-changelog">
<ul>
<li>新增:现在可以批量下载单页搜索结果中的作品,如<a style="color: #0096fa; text-decoration: underline" href="https://www.pixiv.net/tags/miku%2010000users/artworks?mode=safe&s_mode=s_tag">此页</a>。</li>
</ul>
</div>`;
const footerHtml = `
<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>
<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"
>${t('modals.upgradeMsg.feedback')}
</a>`;
document.body.appendChild(createModal({
header: stringToFragment(headerHtml),
content: stringToFragment(contentHtml),
footer: stringToFragment(footerHtml)
}));
}
function createTabUgoria() {
const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.ugoria.tab_title')}</div>`;
const paneHtml = `
<div class="pdl-tab-pane">
<div id="pdl-setting-ugoria">
<p class="option-header">${t('modals.setting.ugoria.format_label')}</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>
</div>`;
const tab = stringToFragment(tabHtml);
const pane = stringToFragment(paneHtml);
const ugoriaFormat = config.get('ugoriaFormat');
pane.querySelectorAll('.pdl-ugoria-format-item input[type="radio"]').forEach((el) => {
if (ugoriaFormat === el.value)
el.checked = true;
el.addEventListener('change', (ev) => {
config.set('ugoriaFormat', ev.currentTarget.value);
});
});
return {
tab,
pane
};
}
function createTabFilename() {
const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.filename.tab_title')}</div>`;
const paneHtml = `
<div class="pdl-tab-pane">
<div id="pdl-setting-filename">
<div>
<div class="pdl-input-wrap">
<label for="pdlfolder">${t('modals.setting.filename.input.folder_label')}</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">${t('modals.setting.filename.input.filename_label')}</label>
<input type="text" id="pdlfilename" placeholder="${t('modals.setting.filename.input.folder_placeholder')}" 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">${t('modals.setting.filename.input.tag_label')}</span>
<div class="tags-content">
<div class="tags-item">
<input class="pdl-option-tag" type="radio" name="lang" id="lang_ja" value="ja" />
<label for="lang_ja">日本語(default)</label>
</div>
<div class="tags-item">
<input class="pdl-option-tag" type="radio" name="lang" id="lang_zh" value="zh" />
<label for="lang_zh">简中</label>
</div>
<div class="tags-item">
<input class="pdl-option-tag" type="radio" name="lang" id="lang_zh_tw" value="zh_tw" />
<label for="lang_zh_tw">繁中</label>
</div>
<div class="tags-item">
<input class="pdl-option-tag" 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">${t('modals.setting.filename.tips.filename_pattern')}</p>
<p style="font-size: 14px; margin: 0.5em 0">${t('modals.setting.filename.tips.empty_folder')}</p>
<p style="font-size: 14px; margin: 0.5em 0">${t('modals.setting.filename.tips.tag_translation')}</p>
<hr />
<div ${env.isFileSystemAccessAvaliable() ? '' : 'class="pdl-unavailable"'}>
<div style="display: flex; justify-content: space-between; align-items: center; margin: 12px 0; gap: 12px">
<label class="pdl-options" style="padding: 0.6em 4px">
<span style="font-weight: 700">${t('modals.setting.filename.input.fsa_label')}</span>
<input id="pdl-options-file-system-access" type="checkbox" class="pdl-checkbox" style="margin-left: 8px" />
</label>
<hr class="vertical" />
<div class="pdl-input-wrap" style="flex: 1; margin: 0">
<input id="pdl-fsa-show-directory" type="text" placeholder="${t('modals.setting.filename.input.fsa_placeholder')}" style="font-size: 14px; padding: 8px 0.5em; line-height: 1.15" disabled/>
</div>
<button id="pdl-fsa-change-directory" class="pdl-dialog-button primary">${t('modals.setting.filename.button.fsa_change_dir')}</button>
</div>
<div class="tags-option">
<span class="tags-title">${t('modals.setting.filename.input.filename_conflict_label')}</span>
<div class="tags-content">
<div class="tags-item">
<input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_uniquify" value="uniquify"/>
<label for="action_uniquify">${t('modals.setting.filename.input.filename_conflict_option_uniquify')}</label>
</div>
<div class="tags-item">
<input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_overwrite" value="overwrite"/>
<label for="action_overwrite">${t('modals.setting.filename.input.filename_conflict_option_overwrite')}</label>
</div>
<div class="tags-item">
<input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_prompt" value="prompt"/>
<label for="action_prompt">${t('modals.setting.filename.input.filename_conflict_option_prompt')}</label>
</div>
</div>
</div>
</div>
</div>
</div>`;
const tab = stringToFragment(tabHtml);
const pane = stringToFragment(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');
const filenamePattern = config.get('filenamePattern');
const folderPattern = config.get('folderPattern');
if (!folder || !filename)
throw new Error('[Error]Can not create modal.');
filename.value = filenamePattern;
if (!env.isSupportSubpath()) {
folder.setAttribute('disabled', '');
folder.value = '';
}
else {
folder.value = folderPattern;
}
folder.placeholder = env.isViolentmonkey()
? t('modals.setting.filename.input.folder_placeholder_vm')
: !env.isSupportSubpath()
? t('modals.setting.filename.input.folder_placeholder_need_api')
: t('modals.setting.filename.input.folder_placeholder');
folder.addEventListener('input', () => {
folderReset?.removeAttribute('disabled');
folderUpdate?.removeAttribute('disabled');
});
folderReset?.addEventListener('click', () => {
folder.value = config.get('folderPattern');
folderReset?.setAttribute('disabled', '');
folderUpdate?.setAttribute('disabled', '');
});
folderUpdate?.addEventListener('click', () => {
const folderPattern = folder.value
.split('/')
.map(replaceInvalidChar)
.filter((path) => !!path)
.join('/');
config.set('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 = config.get('filenamePattern');
filenameReset?.setAttribute('disabled', '');
filenameUpdate?.setAttribute('disabled', '');
});
filenameUpdate?.addEventListener('click', () => {
const filenamePattern = replaceInvalidChar(filename.value);
if (filenamePattern === '')
return filenameReset?.click();
config.set('filenamePattern', filenamePattern);
filename.value = filenamePattern;
filenameReset?.setAttribute('disabled', '');
filenameUpdate?.setAttribute('disabled', '');
});
const tagLang = config.get('tagLang');
pane.querySelectorAll('.tags-content .tags-item input.pdl-option-tag').forEach((input) => {
if (tagLang === input.value)
input.checked = true;
input.addEventListener('change', (ev) => {
config.set('tagLang', ev.currentTarget.value);
});
});
if (env.isFileSystemAccessAvaliable()) {
const enableFSA = pane.querySelector('#pdl-options-file-system-access');
const showDir = pane.querySelector('#pdl-fsa-show-directory');
const changeDirBtn = pane.querySelector('#pdl-fsa-change-directory');
const actionInput = pane.querySelectorAll('.tags-content .tags-item input.pdl-option-conflict');
const interactElems = [changeDirBtn, ...actionInput];
const isUseFSA = config.get('useFileSystemAccess');
const conflictAction = config.get('fileSystemFilenameConflictAction');
if (isUseFSA) {
folder.placeholder = t('modals.setting.filename.input.folder_placeholder');
folder.removeAttribute('disabled');
folder.value = folderPattern;
}
enableFSA.checked = isUseFSA;
if (!isUseFSA) {
interactElems.forEach((el) => el.setAttribute('disabled', ''));
}
enableFSA.addEventListener('change', (ev) => {
const isEnabled = ev.target.checked;
config.set('useFileSystemAccess', isEnabled);
if (isEnabled) {
folder.placeholder = t('modals.setting.filename.input.folder_placeholder');
if (folder.hasAttribute('disabled')) {
folder.removeAttribute('disabled');
folder.value = config.get('folderPattern');
}
interactElems.forEach((el) => el.removeAttribute('disabled'));
}
else {
if (env.isViolentmonkey()) {
folder.placeholder = t('modals.setting.filename.input.folder_placeholder_vm');
folder.setAttribute('disabled', '');
folder.value = '';
}
else if (!env.isSupportSubpath()) {
folder.placeholder = t('modals.setting.filename.input.folder_placeholder_need_api');
folder.setAttribute('disabled', '');
folder.value = '';
}
interactElems.forEach((el) => el.setAttribute('disabled', ''));
}
});
showDir.value = getCurrentDirName();
changeDirBtn.addEventListener('click', async () => {
await updateDirHandle();
showDir.value = getCurrentDirName();
});
actionInput.forEach((input) => {
if (conflictAction === input.value)
input.checked = true;
input.addEventListener('change', (ev) => {
config.set('fileSystemFilenameConflictAction', ev.currentTarget.value);
});
});
}
return {
tab,
pane
};
}
function createTabAdjustButtonPosition() {
const style = getComputedStyle(document.documentElement);
const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.button_pos.tab_title')}</div>`;
const paneHtml = `
<div class="pdl-tab-pane">
<div class="pdl-adjust-button">
<div class="pdl-adjust-content">
<datalist id="pdl-adjust-tickmarks">
<option value="0"></option>
<option value="25"></option>
<option value="50"></option>
<option value="75"></option>
<option value="100"></option>
</datalist>
<div class="pdl-adjust-item">
<p class="pdl-adjust-title">${t('modals.setting.button_pos.self_bookmark_title')}</p>
<div class="pdl-adjust-select">
<span>${t('modals.setting.button_pos.input.horizon_label')}</span
><input
id="pdl-btn-self-bookmark-left"
type="range"
max="100"
min="0"
step="1"
list="pdl-adjust-tickmarks"
value="${style.getPropertyValue('--pdl-btn-self-bookmark-left')}"
/>
</div>
<div class="pdl-adjust-select">
<span>${t('modals.setting.button_pos.input.vertical_label')}</span
><input
id="pdl-btn-self-bookmark-top"
type="range"
max="100"
min="0"
step="1"
list="pdl-adjust-tickmarks"
value="${style.getPropertyValue('--pdl-btn-self-bookmark-top')}"
/>
</div>
</div>
<div class="pdl-adjust-item">
<p class="pdl-adjust-title">${t('modals.setting.button_pos.preview_title')}</p>
<div class="pdl-adjust-select">
<span>${t('modals.setting.button_pos.input.horizon_label')}</span
><input
id="pdl-btn-left"
type="range"
max="100"
min="0"
step="1"
list="pdl-adjust-tickmarks"
value="${style.getPropertyValue('--pdl-btn-left')}"
/>
</div>
<div class="pdl-adjust-select">
<span>${t('modals.setting.button_pos.input.vertical_label')}</span
><input
id="pdl-btn-top"
type="range"
max="100"
min="0"
step="1"
list="pdl-adjust-tickmarks"
value="${style.getPropertyValue('--pdl-btn-top')}"
/>
</div>
</div>
</div>
<div class="pdl-adjust-preview">
<div class="pdl-thumbnail-sample">
<button class="pdl-btn pdl-btn-sub"></button>
<button class="pdl-btn pdl-btn-sub self-bookmark pdl-complete"></button>
</div>
</div>
</div>
</div>`;
const tab = stringToFragment(tabHtml);
const pane = stringToFragment(paneHtml);
pane.querySelectorAll('.pdl-adjust-select input[type="range"]').forEach((el) => {
el.addEventListener('input', (ev) => {
const el = ev.target;
const val = el.value;
document.documentElement.style.setProperty('--' + el.id, val);
});
el.addEventListener('change', (ev) => {
const el = ev.target;
const key = el.id;
config.set(key, el.value);
});
});
return {
tab,
pane
};
}
function createTabHistory() {
const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.history.tab_title')}</div>`;
const paneHtml = `
<div class="pdl-tab-pane">
<div id="pdl-setting-history">
<div>
<button id="pdl-export" class="btn-history pdl-dialog-button primary">
${t('modals.setting.history.button.export')}
</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">
${t('modals.setting.history.button.import')}
</button>
</div>
<div>
<input type="file" id="pdl-merge" accept=".txt" style="display: none" />
<button id="pdl-merge-btn" class="btn-history pdl-dialog-button primary">
${t('modals.setting.history.button.merge')}
</button>
</div>
<div>
<button id="pdl-clear-history" class="btn-history pdl-dialog-button primary">
${t('modals.setting.history.button.clear')}
</button>
</div>
</div>
</div>`;
const tab = stringToFragment(tabHtml);
const pane = stringToFragment(paneHtml);
const importFile = pane.querySelector('#pdl-import');
importFile?.addEventListener('change', (evt) => {
const file = evt.currentTarget.files?.[0];
if (!file)
return;
pixivHistory.replace(file);
});
const mergeFile = pane.querySelector('#pdl-merge');
mergeFile?.addEventListener('change', (evt) => {
const file = evt.currentTarget.files?.[0];
if (!file)
return;
pixivHistory.merge(file);
});
const importBtn = pane.querySelector('#pdl-import-btn');
importBtn?.addEventListener('click', () => importFile?.click());
const mergeBtn = pane.querySelector('#pdl-merge-btn');
mergeBtn?.addEventListener('click', () => mergeFile?.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
};
}
var css = ".pdl-popup-button {\n position: fixed;\n z-index: 1;\n right: 28px;\n bottom: 100px;\n padding: 12px;\n border-radius: 50%;\n border: none;\n cursor: pointer;\n color: rgb(255, 255, 255);\n background-color: rgba(0, 150, 250, 0.5);\n transition: opacity 0.3s ease 0s;\n opacity: 0.32;\n line-height: 0;\n margin: 0;\n}\n\n.pdl-popup-button:hover {\n opacity: 1;\n}\n\n.pdl-popup-button svg {\n width: 24px;\n height: 24px;\n fill: currentColor;\n}\n";
n(css,{});
function showPopupBtn() {
if (document.querySelector('.pdl-popup-button'))
return;
const btn = document.createElement('button');
btn.className = 'pdl-popup-button';
btn.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12,15.5A3.5,3.5 0 0,1 8.5,12A3.5,3.5 0 0,1 12,8.5A3.5,3.5 0 0,1 15.5,12A3.5,3.5 0 0,1 12,15.5M19.43,12.97C19.47,12.65 19.5,12.33 19.5,12C19.5,11.67 19.47,11.34 19.43,11L21.54,9.37C21.73,9.22 21.78,8.95 21.66,8.73L19.66,5.27C19.54,5.05 19.27,4.96 19.05,5.05L16.56,6.05C16.04,5.66 15.5,5.32 14.87,5.07L14.5,2.42C14.46,2.18 14.25,2 14,2H10C9.75,2 9.54,2.18 9.5,2.42L9.13,5.07C8.5,5.32 7.96,5.66 7.44,6.05L4.95,5.05C4.73,4.96 4.46,5.05 4.34,5.27L2.34,8.73C2.21,8.95 2.27,9.22 2.46,9.37L4.57,11C4.53,11.34 4.5,11.67 4.5,12C4.5,12.33 4.53,12.65 4.57,12.97L2.46,14.63C2.27,14.78 2.21,15.05 2.34,15.27L4.34,18.73C4.46,18.95 4.73,19.03 4.95,18.95L7.44,17.94C7.96,18.34 8.5,18.68 9.13,18.93L9.5,21.58C9.54,21.82 9.75,22 10,22H14C14.25,22 14.46,21.82 14.5,21.58L14.87,18.93C15.5,18.67 16.04,18.34 16.56,17.94L19.05,18.95C19.27,19.03 19.54,18.95 19.66,18.73L21.66,15.27C21.78,15.05 21.73,14.78 21.54,14.63L19.43,12.97Z" /></svg>`;
btn.addEventListener('click', () => {
showSettings();
});
document.body.appendChild(btn);
}
function removePopupBtn() {
document.querySelector('button.pdl-popup-button')?.remove();
}
function createTabOthers() {
const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.others.tab_title')}</div>`;
const paneHtml = `
<div class="pdl-tab-pane">
<div id="pdl-setting-others">
<div>
<label class="pdl-options">
<input id="pdl-options-bundle-illusts" type="checkbox" class="pdl-checkbox" />
<span>${t('modals.setting.others.input.bundle_illusts')}</span>
</label>
</div>
<hr />
<div>
<label class="pdl-options">
<input id="pdl-options-bundle-manga" type="checkbox" class="pdl-checkbox" />
<span>${t('modals.setting.others.input.bundle_manga')}</span>
</label>
</div>
<hr />
<div>
<label class="pdl-options">
<input id="pdl-options-add-bookmark" type="checkbox" class="pdl-checkbox" />
<span>${t('modals.setting.others.input.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>${t('modals.setting.others.input.add_bookmark_with_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>${t('modals.setting.others.input.add_bookmark_private_r18')}</span>
</label>
</div>
<hr />
<div>
<label class="pdl-options">
<input id="pdl-options-show-popup-button" type="checkbox" class="pdl-checkbox" />
<span>${t('modals.setting.others.input.show_popup_button')}</span>
</label>
</div>
</div>
</div>`;
const tab = stringToFragment(tabHtml);
const pane = stringToFragment(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'
},
{ selector: '#pdl-options-show-popup-button', settingKey: 'showPopupButton' }
].forEach(({ selector, settingKey }) => {
const optionEl = pane.querySelector(selector);
if (!optionEl)
return;
optionEl.checked = config.get(settingKey);
optionEl.addEventListener('change', (ev) => {
config.set(settingKey, ev.currentTarget.checked);
});
});
pane.querySelector('#pdl-options-show-popup-button').addEventListener('change', (ev) => {
if (ev.currentTarget.checked) {
showPopupBtn();
}
else {
removePopupBtn();
}
});
return {
tab,
pane
};
}
function createTabFeedback() {
const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.feedback.tab_title')}</div>`;
const paneHtml = `
<div class="pdl-tab-pane">
<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"
>${t('modals.setting.feedback.tips.feedback')}</a
>
</p>
</div>
</div>`;
return {
tab: stringToFragment(tabHtml),
pane: stringToFragment(paneHtml)
};
}
function showSettings() {
if (document.querySelector('.pdl-modal'))
return;
const contentHtml = `
<div>
<div class="pdl-tabs-nav">
<div class="pdl-tabs__active-bar"></div>
</div>
<div class="pdl-tabs-content"></div>
</div>`;
const modal = createModal({
content: stringToFragment(contentHtml)
});
const tabsNav = modal.querySelector('.pdl-tabs-nav');
const tabContent = modal.querySelector('.pdl-tabs-content');
[
createTabFilename(),
createTabUgoria(),
createTabHistory(),
createTabAdjustButtonPosition(),
createTabOthers(),
createTabFeedback()
].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(t('gm_menu.setting'), showSettings, 's');
if (config.get('showMsg')) {
showUpgradeMsg();
config.set('showMsg', false);
}
if (config.get('showPopupButton')) {
showPopupBtn();
}
['pdl-btn-self-bookmark-left', 'pdl-btn-self-bookmark-top', 'pdl-btn-left', 'pdl-btn-top'].forEach((key) => {
let val;
if ((val = config.get(key)) !== undefined) {
document.documentElement.style.setProperty('--' + key, val);
}
});
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'));
}
}
}
});
})(workerChunk, GIF, JSZip, dayjs);