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