您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
一键下载Pixiv各页面原图。支持多图下载,动图下载,按作品标签下载,画师作品批量下载。动图支持格式转换:Gif | Apng | Webp | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。
当前为
// ==UserScript== // @name Pixiv Downloader // @namespace https://greasyfork.org/zh-CN/scripts/432150 // @version 0.9.2 // @description:en Download the original images of Pixiv pages with one click. Supports:multiple illustrations, ugoira(animation), and batch downloads of artists' work. Ugoira support format conversion: Gif | Apng | Webp | Webm. The downloaded images will be saved in a separate folder named after the artist (you need to adjust the tampermonkey "Download" setting to "Browser API"). A record of downloaded images is kept. // @description 一键下载Pixiv各页面原图。支持多图下载,动图下载,按作品标签下载,画师作品批量下载。动图支持格式转换:Gif | Apng | Webp | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。 // @description:zh-TW 一鍵下載Pixiv各頁面原圖。支持多圖下載,動圖下載,按作品標籤下載,畫師作品批次下載。動圖支持格式轉換:Gif | Apng | Webp | Webm。下載的圖片將保存到以畫師名命名的單獨文件夾(需要調整tampermonkey“下載”設置為“瀏覽器API”)。保留已下載圖片的紀錄。 // @author ruaruarua // @match https://www.pixiv.net/* // @icon https://www.pixiv.net/favicon.ico // @noframes // @grant unsafeWindow // @grant GM_xmlhttpRequest // @grant GM_download // @grant GM_setValue // @grant GM_getValue // @grant GM_info // @grant GM_registerMenuCommand // @connect i.pximg.net // @require https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/gif.js/0.2.0/gif.js // @require https://cdn.jsdelivr.net/npm/dayjs@1/dayjs.min.js // @require https://greasyfork.org/scripts/455256-toanimatedwebp/code/toAnimatedWebp.js?version=1120088 // ==/UserScript== (function (workerChunk, GIF, JSZip, dayjs) { 'use strict'; function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; } var workerChunk__default = /*#__PURE__*/_interopDefaultLegacy(workerChunk); var GIF__default = /*#__PURE__*/_interopDefaultLegacy(GIF); var JSZip__default = /*#__PURE__*/_interopDefaultLegacy(JSZip); var dayjs__default = /*#__PURE__*/_interopDefaultLegacy(dayjs); var e=[],t$1=[];function n(n,r){if(n&&"undefined"!=typeof document){var a,s=!0===r.prepend?"prepend":"append",d=!0===r.singleTag,i="string"==typeof r.container?document.querySelector(r.container):document.getElementsByTagName("head")[0];if(d){var u=e.indexOf(i);-1===u&&(u=e.push(i)-1,t$1[u]={}),a=t$1[u]&&t$1[u][s]?t$1[u][s]:t$1[u][s]=c();}else a=c();65279===n.charCodeAt(0)&&(n=n.substring(1)),a.styleSheet?a.styleSheet.cssText+=n:a.appendChild(document.createTextNode(n));}function c(){var e=document.createElement("style");if(e.setAttribute("type","text/css"),r.attributes)for(var t=Object.keys(r.attributes),n=0;n<t.length;n++)e.setAttribute(t[n],r.attributes[t[n]]);var a="prepend"===s?"afterbegin":"beforeend";return i.insertAdjacentElement(a,e),e}} var css$4 = "@property --pdl-progress {\n syntax: '<percentage>';\n inherits: true;\n initial-value: 0%;\n}\n\n@keyframes pdl_loading {\n 100% {\n transform: translate(-50%, -50%) rotate(360deg);\n }\n}\n\n:root {\n --pdl-btn-top: 100;\n --pdl-btn-left: 0;\n --pdl-btn-self-bookmark-top: 75;\n --pdl-btn-self-bookmark-left: 100;\n}\n\n.pdl-btn {\n position: relative;\n border-radius: 4px;\n background: no-repeat center/85%;\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%233C3C3C' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n color: #01b468;\n display: inline-block;\n font-size: 13px;\n font-weight: bold;\n height: 32px;\n line-height: 32px;\n margin: 0;\n overflow: hidden;\n padding: 0;\n border: none;\n text-decoration: none !important;\n text-align: center;\n text-overflow: ellipsis;\n user-select: none;\n white-space: nowrap;\n width: 32px;\n z-index: 10;\n cursor: pointer;\n}\n\n.pdl-btn-main {\n margin: 0 0 0 10px;\n}\n\n.pdl-btn-sub {\n position: absolute;\n background-color: rgba(255, 255, 255, 0.5);\n top: calc((100% - 32px) * var(--pdl-btn-top) / 100);\n left: calc((100% - 32px) * var(--pdl-btn-left) / 100);\n}\n\n.pdl-btn-sub.artworks {\n position: sticky;\n top: 40px;\n left: 0px;\n}\n\n.pdl-btn-sub.presentation {\n position: fixed;\n top: 50px;\n right: 20px;\n left: auto;\n border-radius: 8px;\n}\n\n.pdl-btn-sub.self-bookmark {\n top: calc((100% - 32px) * var(--pdl-btn-self-bookmark-top) / 100);\n left: calc((100% - 32px) * var(--pdl-btn-self-bookmark-left) / 100);\n}\n\n.pdl-error {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23EA0000' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\") !important;\n}\n\n.pdl-complete {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%2301B468' d='M256 8C119.033 8 8 119.033 8 256s111.033 248 248 248 248-111.033 248-248S392.967 8 256 8zm0 48c110.532 0 200 89.451 200 200 0 110.532-89.451 200-200 200-110.532 0-200-89.451-200-200 0-110.532 89.451-200 200-200m140.204 130.267l-22.536-22.718c-4.667-4.705-12.265-4.736-16.97-.068L215.346 303.697l-59.792-60.277c-4.667-4.705-12.265-4.736-16.97-.069l-22.719 22.536c-4.705 4.667-4.736 12.265-.068 16.971l90.781 91.516c4.667 4.705 12.265 4.736 16.97.068l172.589-171.204c4.704-4.668 4.734-12.266.067-16.971z'%3E%3C/path%3E %3C/svg%3E\") !important;\n}\n\n.pdl-progress {\n background-image: none !important;\n cursor: default !important;\n}\n\n.pdl-progress:after {\n content: '';\n display: inline-block;\n position: absolute;\n top: 50%;\n left: 50%;\n width: 27px;\n height: 27px;\n transform: translate(-50%, -50%);\n -webkit-mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);\n mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);\n border-radius: 50%;\n}\n\n.pdl-progress:not(:empty):after {\n background: conic-gradient(#01b468 0, #01b468 var(--pdl-progress), transparent var(--pdl-progress), transparent);\n transition: --pdl-progress 0.2s ease;\n}\n\n.pdl-progress:empty:after {\n background: conic-gradient(#01b468 0, #01b468 25%, #01b46833 25%, #01b46833);\n animation: 1.5s infinite linear pdl_loading;\n}\n\n.pdl-btn.pdl-tag {\n height: auto;\n border-top-right-radius: 4px;\n border-bottom-right-radius: 4px;\n left: -1px;\n background-color: rgba(0, 0, 0, 0.12);\n transition: background-image 0.5s;\n}\n\n.pdl-btn.pdl-tag.pdl-tag-hide,\n.pdl-btn.pdl-modal-tag.pdl-tag-hide {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3C/svg%3E\");\n pointer-events: none;\n}\n\n.pdl-btn.pdl-modal-tag {\n position: absolute;\n right: 65px;\n top: 6px;\n background-origin: content-box;\n border-radius: 4px;\n padding: 5px;\n width: 42px;\n height: 50px;\n background-color: rgba(0, 0, 0, 0.04);\n transition: 0.25s background-color;\n}\n\n.pdl-btn.pdl-modal-tag:not(.pdl-tag-hide):hover {\n background-color: rgba(0, 0, 0, 0.12);\n}\n\n.pdl-wrap-artworks {\n position: absolute;\n right: 8px;\n top: 0px;\n bottom: 0px;\n margin-top: 40px;\n}\n"; n(css$4,{}); var css$3 = ".pdl-modal * {\n font-family: 'win-bug-omega, system-ui, -apple-system, \"Segoe UI\", Roboto, Ubuntu, Cantarell, \"Noto Sans\", \"Hiragino Kaku Gothic ProN\", Meiryo, sans-serif';\n line-height: 1.15;\n}\n\n.pdl-modal {\n position: fixed;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n display: flex;\n z-index: 99;\n background-color: rgba(0, 0, 0, 0.32);\n user-select: none;\n}\n\n.pdl-dialog {\n position: relative;\n background-color: #fff;\n border-radius: 24px;\n margin: auto;\n padding: 20px 40px 30px 40px;\n width: 600px;\n font-size: 16px;\n}\n\n.pdl-dialog-header > h3 {\n font-weight: bold;\n font-size: 1.17em;\n margin: 1em 0;\n}\n\n.pdl-dialog-close {\n position: absolute;\n top: 10px;\n right: 10px;\n margin: 0;\n padding: 0;\n width: 25px;\n height: 25px;\n border: none;\n cursor: pointer;\n border-radius: 50%;\n background-color: transparent;\n transform: rotate(45deg);\n transition: 0.25s background-color;\n background: linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/18px 2px no-repeat,\n linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/2px 18px no-repeat;\n}\n\n.pdl-dialog-close:hover {\n background-color: rgba(0, 0, 0, 0.05);\n}\n\n.pdl-dialog-content hr {\n height: 0 !important;\n margin: 0;\n border: none;\n border-top: 1px solid rgba(0, 0, 0, 0.1);\n}\n\n.pdl-dialog-content hr.sub {\n margin-inline-start: 1.5em;\n}\n\n.pdl-dialog-content hr.vertical {\n height: 1.15em !important;\n border: none;\n border-left: 1px solid rgba(0, 0, 0, 0.1);\n}\n\n/* button */\n.pdl-dialog-button {\n font-size: 16px;\n background-color: #fff;\n border: 1px solid rgb(125, 125, 125);\n border-radius: 5px;\n padding: 0.5em 1.5em;\n cursor: pointer;\n transition: 0.2s opacity;\n line-height: 1.15;\n}\n.pdl-dialog-button:hover {\n opacity: 0.7;\n}\n\n.pdl-dialog-button.primary {\n color: #fff;\n background-color: #0096fa;\n border-color: #0096fa;\n}\n\n.pdl-dialog-button.icon {\n padding: 0.5em 0.8em;\n}\n\n.pdl-dialog-button[disabled] {\n cursor: not-allowed !important;\n color: #c0c4cc;\n background-color: #fff;\n border-color: #e4e7ed;\n opacity: 1 !important;\n}\n\n.pdl-dialog-button[disabled]:hover {\n opacity: 1 !important;\n}\n\n.pdl-dialog-button.primary[disabled] {\n color: #fff;\n background-color: #a0cfff;\n border-color: #a0cfff;\n}\n\n/* tabs */\n.pdl-tabs-nav {\n display: flex;\n align-items: center;\n border-bottom: 1px solid #dcdfe6;\n position: relative;\n}\n\n.pdl-tabs-nav .pdl-tab-item {\n padding: 0px 16px;\n line-height: 2.5;\n cursor: pointer;\n transition: 0.2s color;\n}\n\n.pdl-tabs-nav .pdl-tab-item:nth-child(2) {\n padding-left: 0px;\n}\n\n.pdl-tabs-nav .pdl-tab-item:last-child {\n padding-right: 0px;\n}\n\n.pdl-tab-item:hover,\n.pdl-tab-item.active {\n color: #0096fa;\n}\n\n.pdl-tab-item.active {\n font-weight: 700;\n}\n\n.pdl-tabs-content {\n padding: 16px;\n min-height: 200px;\n}\n\n.pdl-tabs-nav .pdl-tabs__active-bar {\n position: absolute;\n bottom: 0;\n left: 0;\n height: 2px;\n background-color: #0096fa;\n z-index: 1;\n transition: width 0.2s, transform 0.2s;\n}\n\n/* filename */\n\n#pdl-setting-filename a {\n color: #0096fa;\n text-decoration: underline;\n}\n#pdl-setting-filename .tags-option label,\n#pdl-setting-filename .tags-option input {\n cursor: pointer;\n}\n\n#pdl-setting-filename .tags-content {\n flex: 1;\n display: flex;\n gap: 20px;\n}\n\n#pdl-setting-filename .pdl-input-wrap,\n#pdl-setting-filename .tags-option {\n display: flex;\n align-items: center;\n margin: 12px 0;\n gap: 12px;\n}\n\n#pdl-setting-filename .pdl-input-wrap label,\n#pdl-setting-filename .tags-option .tags-title {\n cursor: default;\n font-weight: 700;\n width: 7em;\n}\n\n#pdl-setting-filename .pdl-input-wrap input[type='text'] {\n height: auto;\n padding: 0.5em;\n font-size: 16px;\n line-height: 1.5;\n flex: 1;\n border: 1px solid #333;\n}\n\n#pdl-setting-filename .pdl-input-wrap input[type='text']:focus {\n background-color: #fff !important;\n}\n\n#pdl-setting-filename .pdl-input-wrap button {\n line-height: 1.5;\n}\n\n#pdl-setting-filename .pdl-options {\n display: flex;\n align-items: center;\n justify-content: space-between;\n cursor: pointer;\n padding: 0.6em 0;\n}\n\n#pdl-setting-filename .pdl-options:hover {\n background-color: rgba(0, 0, 0, 0.05);\n}\n\n#pdl-setting-filename .pdl-unavailable {\n pointer-events: none !important;\n opacity: 0.5 !important;\n cursor: not-allowed !important;\n}\n\n/* ugoria */\n#pdl-setting-ugoria p.option-header {\n font-weight: 700;\n}\n\n#pdl-setting-ugoria #pdl-ugoria-format-wrap {\n display: flex;\n justify-content: space-between;\n flex-wrap: nowrap;\n margin: 1.5em 1em;\n}\n\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item,\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item label,\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item input {\n cursor: pointer;\n}\n\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item label {\n padding-left: 4px;\n}\n\n/* history */\n#pdl-setting-history div {\n text-align: center;\n margin: 1em 0;\n}\n\n#pdl-setting-history .btn-history {\n width: 80%;\n}\n\n/* others setting */\n#pdl-setting-others .pdl-options {\n display: flex;\n align-items: center;\n gap: 20px;\n cursor: pointer;\n transition: 0.2s background-color;\n padding: 1em 0.5em;\n border-radius: 4px;\n}\n\n#pdl-setting-others .pdl-options:hover {\n background-color: rgba(0, 0, 0, 0.05);\n}\n\n#pdl-setting-others .pdl-options.sub-option {\n padding: 0.5em;\n padding-inline-start: 2em;\n}\n\n/* donate */\n#pdl-setting-donate {\n text-align: center;\n}\n\n#pdl-setting-donate p {\n margin: 0.5em 0;\n}\n\n/* upgrade msg */\n.pdl-changelog h4 {\n margin: 1em 0;\n font-weight: bold !important;\n}\n\n.pdl-changelog ul {\n padding-inline-start: 40px !important;\n}\n\n.pdl-changelog li {\n line-height: 2;\n list-style-type: disc !important;\n}\n\n.pdl-changelog p {\n margin: 0.5em 0;\n}\n\n/* adjust button position */\n.pdl-adjust-button {\n display: flex;\n justify-content: space-between;\n gap: 32px;\n}\n\n.pdl-adjust-content {\n flex: 1;\n}\n\n.pdl-adjust-content .pdl-adjust-item {\n margin-bottom: 1em;\n}\n\n.pdl-adjust-content .pdl-adjust-item .pdl-adjust-title {\n font-weight: 700;\n margin-bottom: 0.8em;\n}\n\n.pdl-adjust-content .pdl-adjust-item .pdl-adjust-select {\n display: flex;\n align-items: center;\n margin: 0.6em 0;\n padding: 0 0.4em;\n gap: 20px;\n}\n\n.pdl-adjust-select input[type='range'] {\n flex: 1;\n}\n\n.pdl-adjust-preview {\n align-self: center;\n}\n\n.pdl-adjust-preview .pdl-thumbnail-sample {\n position: relative;\n width: 184px;\n height: 184px;\n background-color: rgba(0, 150, 250, 0.15);\n border-radius: 4px;\n}\n"; n(css$3,{}); var css$2 = ".pdl-dlbar-status_bar {\n flex-grow: 1;\n height: 46px;\n line-height: 46px;\n padding-right: 8px;\n text-align: right;\n font-weight: bold;\n font-size: 16px;\n color: rgb(133, 133, 133);\n cursor: default;\n white-space: nowrap;\n}\n\n.pdl-btn-all.pdl-btn-all,\n.pdl-stop.pdl-stop {\n background-color: transparent;\n border: none;\n padding: 0 10px;\n}\n\n.pdl-btn-all.pdl-btn-all:hover,\n.pdl-stop.pdl-stop:hover {\n color: rgb(31, 31, 31);\n}\n\n.pdl-btn-all::before,\n.pdl-stop::before {\n content: '';\n height: 24px;\n width: 24px;\n transition: background-image 0.2s ease 0s;\n background: no-repeat center/85%;\n}\n\n.pdl-btn-all::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-stop::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-btn-all:hover::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-stop:hover::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-hide {\n display: none !important;\n}\n\n.pdl-filter-wrap {\n display: flex;\n justify-content: flex-end;\n gap: 12px;\n margin: 4px 0;\n font-weight: bold;\n font-size: 14px;\n line-height: 14px;\n color: rgb(133, 133, 133);\n transition: color 0.2s ease 0s;\n}\n\n.pdl-filter-wrap.unavailable {\n pointer-events: none !important;\n opacity: 0.5 !important;\n}\n\n.pdl-filter-wrap .pdl-filter:hover {\n color: rgb(31, 31, 31);\n}\n\n.pdl-filter-wrap label {\n padding-left: 8px;\n cursor: pointer;\n}\n\n.pdl-checkbox.pdl-checkbox {\n vertical-align: top;\n appearance: none;\n position: relative;\n box-sizing: border-box;\n width: 28px;\n border: 2px solid transparent;\n cursor: pointer;\n border-radius: 14px;\n height: 14px;\n background-color: rgba(133, 133, 133);\n transition: background-color 0.2s ease 0s, box-shadow 0.2s ease 0s;\n}\n\n.pdl-checkbox:hover {\n background-color: rgba(31, 31, 31);\n}\n\n.pdl-checkbox::after {\n content: '';\n position: absolute;\n display: block;\n top: 0px;\n left: 0px;\n width: 10px;\n height: 10px;\n transform: translateX(0px);\n background-color: rgb(255, 255, 255);\n border-radius: 10px;\n transition: transform 0.2s ease 0s;\n}\n\n.pdl-checkbox:checked {\n background-color: rgb(0, 150, 250);\n}\n\n.pdl-checkbox:checked::after {\n transform: translateX(14px);\n}\n\n.pdl-dlbar {\n display: flex;\n flex-grow: 1;\n}\n\n.pdl-dlbar-follow_latest {\n padding: 0 8px;\n}\n"; n(css$2,{}); var css$1 = "[data-theme='dark'] .pdl-btn-all::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-btn-main,\n[data-theme='dark'] .pdl-btn-all:hover::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23D6D6D6' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-stop::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-stop:hover::before {\n background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23D6D6D6' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-checkbox:not(:checked):hover {\n background-color: rgba(155, 155, 155);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-tag {\n background-color: rgba(255, 255, 255, 0.4);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-modal-tag {\n background-color: rgba(255, 255, 255, 0.4);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-modal-tag:hover {\n background-color: rgba(255, 255, 255, 0.6);\n}\n\n[data-theme='dark'] .pdl-wrap .pdl-filter:hover,\n[data-theme='dark'] .pdl-stop.pdl-stop:hover,\n[data-theme='dark'] .pdl-btn-all.pdl-btn-all:hover {\n color: rgb(214, 214, 214);\n}\n\n/* modal */\n[data-theme='dark'] .pdl-dialog {\n background-color: rgb(31, 31, 31);\n}\n\n[data-theme='dark'] .pdl-dialog-footer button {\n background-color: rgb(245, 245, 245);\n}\n\n[data-theme='dark'] .pdl-dialog-content hr {\n border-top: 1px solid rgba(255, 255, 255, 0.3);\n}\n\n/* others setting */\n[data-theme='dark'] #pdl-setting-others .pdl-options:hover {\n background-color: rgba(255, 255, 255, 0.1);\n}\n"; n(css$1,{}); const regexp = { preloadData: /"meta-preload-data" content='(.*?)'>/, globalData: /"meta-global-data" content='(.*?)'>/, artworksPage: /artworks\/(\d+)$/, userPage: /users\/(\d+)/, bookmarkPage: /users\/(\d+)\/bookmarks\/artworks/, userPageTags: /users\/\d+\/(artworks|illustrations|manga|bookmarks(?!artworks))/, 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 langZh = { downloadBar: { button: { stop: '停止', works: '作品', bookmarks: '收藏', bookmarks_public: '公开', bookmarks_private: '不公开', all_one_page: '全部(单页)', all: '全部(批量)', r18_one_page: 'R-18(单页)', r18: 'R-18(批量)' }, filter: { exclude_downloaded: '排除已下载图片', illusts: '插画', manga: '漫画', ugoria: '动图' } }, modals: { upgradeMsg: { feedback: '有问题or想建议?这里反馈' }, setting: { filename: { tab_title: '文件名', input: { folder_label: '文件夹名:', folder_placeholder: '我不想保存到子文件夹', folder_placeholder_vm: 'Violentmonkey不支持', folder_placeholder_need_api: '需要Browser Api', filename_label: '文件名:', filename_placeholder: '你的名字?', tag_label: '标签语言:', fsa_label: '使用FileSystemAccess API', fsa_placeholder: '根文件夹名', filename_conflict_label: '文件名重复时:', filename_conflict_option_uniquify: '重命名', filename_conflict_option_overwrite: '覆盖', filename_conflict_option_prompt: '提示' }, tips: { filename_pattern: '{artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}:作品ID, {page}:页码, {tags}:作品标签,{date} / {date(占位符)}: 创建时间', empty_folder: '如果不想保存到画师目录,文件夹名留空即可。', tag_translation: '请注意:标签翻译不一定是你选择的语言,部分<a href="https://crowdin.com/project/pixiv-tags" target="_blank">无对应语言翻译的标签</a>仍可能是其他语言。' }, button: { fsa_change_dir: '更改' } }, ugoria: { tab_title: '动图', format_label: '动图格式:' }, history: { tab_title: '历史记录', button: { import: '导入记录(替换)', merge: '导入记录(合并)', export: '导出记录', clear: '清除记录' }, tips: { clear: '真的要清除历史记录吗?' } }, button_pos: { tab_title: '按钮', self_bookmark_title: '预览图(我的收藏)', preview_title: '预览图', input: { horizon_label: '水平:', vertical_label: '垂直:' } }, others: { tab_title: '其它', input: { bundle_illusts: '将多页插图打包为.zip压缩包', bundle_manga: '将多页漫画作品打包为.zip压缩包', add_bookmark: '下载单个作品时收藏作品', add_bookmark_with_tags: '收藏时添加作品标签', add_bookmark_private_r18: '将R-18作品收藏到不公开类别', show_popup_button: '显示设置按钮' } }, feedback: { tab_title: '反馈 / 赞赏', tips: { feedback: '有问题or想建议?这里反馈' } } } }, gm_menu: { setting: '设置' } }; const langEn = { downloadBar: { button: { stop: 'Stop', works: 'Works', bookmarks: 'Bookmarks', bookmarks_public: 'Public', bookmarks_private: 'Private', all_one_page: 'All (one page)', all: 'All', r18_one_page: 'R-18 (one page)', r18: 'R-18' }, filter: { exclude_downloaded: 'Exclude downloaded', illusts: 'Illustrations', manga: 'Manga', ugoria: 'Ugoria' } }, modals: { upgradeMsg: { feedback: 'Feedback' }, setting: { filename: { tab_title: 'Filename', input: { folder_label: 'FileName:', folder_placeholder: "I don't need subfolder", folder_placeholder_vm: "VM doesn't support", folder_placeholder_need_api: 'Need Browser Api', filename_label: 'FileName:', filename_placeholder: 'Your Name?', tag_label: 'Tags language:', fsa_label: 'FileSystemAccess API', fsa_placeholder: 'Root directory', filename_conflict_label: 'Conflict Action: ', filename_conflict_option_uniquify: 'Uniquify', filename_conflict_option_overwrite: 'Overwrite', filename_conflict_option_prompt: 'Prompt' }, tips: { filename_pattern: '{artist}, {artistID}, {title}, {id}, {page}, {tags}, {date} / {date(format)}', empty_folder: "If you don't need a subfolder, just leave the folder name blank", tag_translation: 'Note: Tags language may not be the language you selected, <a href="https://crowdin.com/project/pixiv-tags" target="_blank">some tags without translations</a> may still be in other languages.' }, button: { fsa_change_dir: 'Change' } }, ugoria: { tab_title: 'Ugoria', format_label: 'Ugoria Format:' }, history: { tab_title: 'History', button: { import: 'Import (Replace)', merge: 'Import (Merge)', export: 'Export', clear: 'Clear' }, tips: { clear: 'Do you really want to clear history?' } }, button_pos: { tab_title: 'Button', self_bookmark_title: 'Thumbnail(My bookmarks)', preview_title: 'Thumbnail', input: { horizon_label: 'X:', vertical_label: 'Y:' } }, others: { tab_title: 'Others', input: { bundle_illusts: 'Pack multi-page illustrations into a .zip archive', bundle_manga: 'Pack manga into a .zip archive', add_bookmark: 'Bookmark work when downloading a single work', add_bookmark_with_tags: 'Add works tags', add_bookmark_private_r18: 'Bookmark R-18 works to private category', show_popup_button: 'Show setting button' } }, feedback: { tab_title: 'Feedback', tips: { feedback: 'Feedback' } } } }, gm_menu: { setting: '设置' } }; const messages = { 'zh-cn': langZh, 'zh-tw': langZh, zh: langZh, en: langEn }; const curLang = document.documentElement.getAttribute('lang')?.toLowerCase() || 'en'; const defaultLang = 'en'; function t(key) { const lang = (curLang in messages ? curLang : defaultLang); const paths = key.split('.'); let last = messages[lang]; for (let i = 0; i < paths.length; i++) { const value = last[paths[i]]; if (value === undefined || value === null) return null; last = value; } return last; } unsafeWindow.t = t; function createHistory() { let records = (function getHistory() { const storage = localStorage.pixivDownloader || '[]'; return new Set(JSON.parse(storage)); })(); function readHistoryFile(file, cb) { if (file.type === 'text/plain') { const reader = new FileReader(); reader.readAsText(file); reader.onload = (readEvt) => { const text = readEvt.target?.result; try { if (typeof text !== 'string') throw new Error('Invalid file'); const history = JSON.parse(text); if (!(history instanceof Array)) throw new Error('Invalid file'); cb(history); location.reload(); } catch (error) { alert(error.message); } }; } else { alert('Invalid file'); } } return { add(pixivId) { if (records.has(pixivId)) return; records.add(pixivId); localStorage.setItem(`pdlTemp-${pixivId}`, ''); }, has(pixivId) { return records.has(pixivId); }, getAll() { return [...records]; }, updateHistory() { const validKeys = Object.keys(localStorage).filter((key) => /(?<=^pdlTemp-)\d+$/.test(key)); if (!validKeys.length) return; validKeys.forEach((key) => { const [id] = /(?<=^pdlTemp-)\d+$/.exec(key); records.add(id); localStorage.removeItem(key); }); this.saveHistory(); }, saveHistory(historyArr) { if (historyArr) { if (historyArr.length && !historyArr.every((id) => typeof id === 'string')) { throw new Error('Invalid id type'); } this.updateHistory(); localStorage.pixivDownloader = JSON.stringify(historyArr); } else { localStorage.pixivDownloader = JSON.stringify([...records]); } }, clearHistory() { const isConfirm = confirm(t('modals.setting.history.tips.clear')); if (!isConfirm) return; this.updateHistory(); records = new Set(); localStorage.pixivDownloader = '[]'; location.reload(); }, replace(file) { readHistoryFile(file, this.saveHistory.bind(this)); }, merge(file) { readHistoryFile(file, (historyArr) => { if (!historyArr.length) throw new Error('No id found'); if (!historyArr.every((id) => typeof id === 'string')) { throw new Error('Invalid id type'); } historyArr.forEach((id) => records.add(id)); this.saveHistory(); }); } }; } const pixivHistory = createHistory(); function getSelfId() { return document.querySelector('#qualtrics_user-id')?.textContent ?? ''; } function getIllustId(node) { const isLinkToArtworksPage = regexp.artworksPage.exec(node.href); if (isLinkToArtworksPage) { if (node.getAttribute('data-gtm-value') || node.classList.contains('gtm-illust-recommend-node-node') || node.classList.contains('gtm-discover-user-recommend-node') || node.classList.contains('work')) { return isLinkToArtworksPage[1]; } } else { const isActivityThumb = regexp.activityHref.exec(node.href); if (isActivityThumb && node.classList.contains('work')) { return isActivityThumb[1]; } } return ''; } function getLogger() { const methods = ['info', 'warn', 'error']; const style = ['color: green;', 'color: orange;', 'color: red;']; const logLevel = 2 ; const namePrefix = '[Pixiv Downlaoder] '; function log(level, args) { if (logLevel <= level) console[methods[level]]('%c[Pixiv Downloader]', style[level], ...args); } return { info(...args) { log(0 , args); }, warn(...args) { log(1 , args); }, error(...args) { log(2 , args); }, time(label) { console.time(namePrefix + label); }, timeLog(label) { console.timeLog(namePrefix + label); }, timeEnd(label) { console.timeEnd(namePrefix + label); } }; } const logger = getLogger(); var IllustType; (function (IllustType) { IllustType[IllustType["illusts"] = 0] = "illusts"; IllustType[IllustType["manga"] = 1] = "manga"; IllustType[IllustType["ugoira"] = 2] = "ugoira"; })(IllustType || (IllustType = {})); var BookmarkRestrict; (function (BookmarkRestrict) { BookmarkRestrict[BookmarkRestrict["public"] = 0] = "public"; BookmarkRestrict[BookmarkRestrict["private"] = 1] = "private"; })(BookmarkRestrict || (BookmarkRestrict = {})); function sleep(delay) { return new Promise((resolve) => { setTimeout(resolve, delay); }); } function wakeableSleep(delay) { let wake = () => void {}; const sleep = new Promise((r) => { setTimeout(r, delay); wake = r; }); return { wake, sleep }; } function replaceInvalidChar(str) { if (typeof str !== 'string') throw new TypeError('expect string but got ' + typeof str); if (!str) return ''; return str .replace(/\p{C}/gu, '') .replace(/\\/g, '\') .replace(/\//g, '/') .replace(/:/g, ':') .replace(/\*/g, '*') .replace(/\?/g, '?') .replace(/\|/g, '|') .replace(/"/g, '"') .replace(/</g, '﹤') .replace(/>/g, '﹥') .trim() .replace(/^\.|\.$/g, '.'); } function unescapeHtml(str) { if (typeof str !== 'string') throw new TypeError('expect string but got ' + typeof str); if (!str) return ''; const el = document.createElement('p'); el.innerHTML = str; return el.innerText; } function stringToFragment(string) { const renderer = document.createElement('template'); renderer.innerHTML = string; return renderer.content; } function loadConfig() { const defaultConfig = Object.freeze({ version: "0.9.2", ugoriaFormat: 'zip', folderPattern: 'pixiv/{artist}', filenamePattern: '{artist}_{title}_{id}_p{page}', tagLang: 'ja', showMsg: true, filterExcludeDownloaded: false, filterIllusts: true, filterManga: true, filterUgoria: true, bundleIllusts: false, bundleManga: false, addBookmark: false, addBookmarkWithTags: false, privateR18: false, useFileSystemAccess: false, fileSystemFilenameConflictAction: 'uniquify', showPopupButton: true }); const config = (() => { if (!localStorage.pdlSetting) return {}; let config; try { config = JSON.parse(localStorage.pdlSetting); } catch (error) { console.log(error); return {}; } if (config.version !== defaultConfig.version) { config.version = defaultConfig.version; config.showMsg = true; } return config; })(); return { get(key) { return config[key] ?? defaultConfig[key]; }, set(key, value) { if (config[key] !== value) { config[key] = value; localStorage.pdlSetting = JSON.stringify(config); logger.info('Config set:', key, value); } } }; } const config = loadConfig(); const handleWorker = ` let webpApi = {}; Module.onRuntimeInitialized = () => { webpApi = { init: Module.cwrap('init', '', ['number', 'number', 'number']), createBuffer: Module.cwrap('createBuffer', 'number', ['number']), addFrame: Module.cwrap('addFrame', 'number', ['number', 'number', 'number']), generate: Module.cwrap('generate', 'number', []), freeResult: Module.cwrap('freeResult', '', []), getResultPointer: Module.cwrap('getResultPointer', 'number', []), getResultSize: Module.cwrap('getResultSize', 'number', []), }; postMessage('ok'); }; onmessage = (evt) => { const { data, delays, lossless = 1, quality = 75, method = 4} = evt.data; webpApi.init(lossless, quality, method); data.forEach((u8a, idx) => { const pointer = webpApi.createBuffer(u8a.length); Module.HEAPU8.set(u8a, pointer); webpApi.addFrame(pointer, u8a.length, delays[idx]); postMessage(idx); }); webpApi.generate(); const resultPointer = webpApi.getResultPointer(); const resultSize = webpApi.getResultSize(); const result = new Uint8Array(Module.HEAP8.buffer, resultPointer, resultSize); postMessage(result); webpApi.freeResult(); };`; class RequestError extends Error { response; constructor(message, response) { super(message); this.name = 'RequestError'; this.response = response; } } class CancelError extends Error { constructor() { super('User aborted'); this.name = 'CancelError'; } } class JsonDataError extends Error { constructor(msg) { super(msg); this.name = 'JsonDataError'; } } function createService() { async function _requestJson(url, init) { logger.info('fetch url:', url); const res = await fetch(url, init); if (!res.ok) throw new RequestError('Request ' + url + ' failed with status code ' + res.status, res); const data = await res.json(); if (data.error) throw new JsonDataError(data.message); return data.body; } async function _getDeps(url) { return fetch(url).then((res) => { if (res.ok) return res.text(); throw new RequestError(`Fetch dependency ${url} failed with status code ${res.status}`, res); }); } return { async getJson(url) { let json; let retry = 0; const MAX_RETRY = 3; do { try { json = await _requestJson(url); } catch (error) { logger.error(error); retry++; if (retry === MAX_RETRY) throw error; sleep(3000); } } while (!json); return json; }, async getArtworkHtml(illustId) { logger.info('Fetch illust:', illustId); let params = ''; const tagLang = config.get('tagLang'); if (tagLang !== 'ja') params = '?lang=' + tagLang; const res = await fetch('https://www.pixiv.net/artworks/' + illustId + params); if (!res.ok) throw new RequestError('Request failed with status code ' + res.status, res); return await res.text(); }, async getGifWS() { let gifWS; if (!(gifWS = await GM_getValue('gifWS'))) { gifWS = await _getDeps(depsUrls.gifWorker); GM_setValue('gifWS', gifWS); } return gifWS; }, async getApngWS() { let apngWS; if (!(apngWS = await GM_getValue('apngWS'))) { const [pako, upng] = await Promise.all([_getDeps(depsUrls.pako), _getDeps(depsUrls.upng)]); const upngScript = upng.replace('window.UPNG', 'UPNG').replace('window.pako', 'pako'); const workerEvt = `onmessage = (evt) => { const {data, width, height, delay } = evt.data; const png = UPNG.encode(data, width, height, 0, delay, {loop: 0}); if (!png) console.error('Convert Apng failed.'); postMessage(png); };`; apngWS = workerEvt + pako + upngScript; GM_setValue('apngWS', apngWS); } return apngWS; }, getWebpWS() { return workerChunk__default["default"] + handleWorker; }, addBookmark(illustId, token, tags = [], restrict = BookmarkRestrict.public) { return _requestJson('/ajax/illusts/bookmarks/add', { method: 'POST', headers: { accept: 'application/json', 'content-type': 'application/json; charset=utf-8', 'x-csrf-token': token }, body: JSON.stringify({ illust_id: illustId, restrict, comment: '', tags }) }); }, getFollowLatestWorks(page, mode = 'all') { return _requestJson(`/ajax/follow_latest/illust?p=${page}&mode=${mode}&lang=jp`); }, getUserAllProfile(userId) { return _requestJson('/ajax/user/' + userId + '/profile/all'); }, getUgoriaMeta(illustId) { return _requestJson('/ajax/illust/' + illustId + '/ugoira_meta'); }, getArtworkDetail(illustId) { let params = ''; const tagLang = config.get('tagLang'); if (tagLang !== 'ja') params = '?lang=' + tagLang; return _requestJson('/ajax/illust/' + illustId + params); } }; } const api = createService(); function addBookmark(pdlBtn, illustId, token, tags) { if (!config.get('addBookmark')) return; api .addBookmark(illustId, token, config.get('addBookmarkWithTags') ? tags : [], config.get('privateR18') && tags.includes('R-18') ? BookmarkRestrict.private : BookmarkRestrict.public) .then(() => { const bookmarkBtnRef = findBookmarkBtn(pdlBtn); if (!bookmarkBtnRef) return; switch (bookmarkBtnRef.kind) { case "main" : { const pathBorder = bookmarkBtnRef.button.querySelector('svg g path'); pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)'); break; } case "sub" : { const pathBorder = bookmarkBtnRef.button.querySelector('path'); pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)'); break; } case "rank" : { bookmarkBtnRef.button.style.backgroundColor = 'rgb(255, 64, 96)'; break; } } }) .catch((reason) => { logger.error(reason.message); }); } function findBookmarkBtn(pdlBtn) { const bookmarkBtnRef = {}; if (pdlBtn.classList.contains('pdl-btn-sub')) { const btn = pdlBtn.parentElement?.nextElementSibling?.querySelector('button[type="button"]'); if (btn) { bookmarkBtnRef.kind = "sub" ; bookmarkBtnRef.button = btn; } else { const btn = pdlBtn.parentElement?.querySelector('div._one-click-bookmark'); if (btn) { bookmarkBtnRef.kind = "rank" ; bookmarkBtnRef.button = btn; } } } else if (pdlBtn.classList.contains('pdl-btn-main')) { const btn = pdlBtn.parentElement?.parentElement?.querySelector('button.gtm-main-bookmark'); if (btn) { bookmarkBtnRef.kind = "main" ; bookmarkBtnRef.button = btn; } } else { return logger.error(new Error('Can not find bookmark button.')); } return bookmarkBtnRef; } const env = { isFirefox() { return navigator.userAgent.includes('Firefox'); }, isViolentmonkey() { return GM_info.scriptHandler === 'Violentmonkey'; }, isTampermonkey() { return GM_info.scriptHandler === 'Tampermonkey'; }, isBlobDlAvaliable() { return !this.isFirefox() || (this.isFirefox() && this.isTampermonkey() && parseFloat(GM_info.version ?? '') < 4.18); }, isSupportSubpath() { return this.isBrowserDownloadMode(); }, isBrowserDownloadMode() { return GM_info.downloadMode === 'browser'; }, isConflictActionEnable() { return this.isTampermonkey() && parseFloat(GM_info.version ?? '') >= 4.18 && this.isBrowserDownloadMode(); }, isConflictActionPromptEnable() { return !this.isFirefox() && this.isConflictActionEnable(); }, isFileSystemAccessAvaliable() { return (typeof unsafeWindow.showDirectoryPicker === 'function' && typeof unsafeWindow.showSaveFilePicker === 'function'); } }; function createCompressor() { const zip = new JSZip__default["default"](); return { add(id, name, data) { zip.folder(id)?.file(name, data); }, bundle(id) { const folder = zip.folder(id); if (!folder) throw new TypeError('no such folder:' + id); return folder.generateAsync({ type: 'blob' }); }, remove(ids) { if (typeof ids === 'string') { zip.remove(ids); } else { const dirs = zip.filter((_, file) => file.dir).map((dir) => dir.name); const dirsToDel = ids.filter((id) => dirs.some((dir) => dir.includes(id))); dirsToDel.forEach((dir) => zip.remove(dir)); logger.info('Compressor: Remove', zip); } }, fileCount(id) { let count = 0; zip.folder(id)?.forEach(() => count++); return count; }, async unzip(data) { const id = Math.random().toString(36); let folder = zip.folder(id); if (!folder) throw TypeError('Can not get new root folder'); const filesPromises = []; folder = await folder.loadAsync(data); folder.forEach((_, file) => { filesPromises.push(file.async('blob')); }); const files = await Promise.all(filesPromises); zip.remove(id); return files; } }; } const compressor = createCompressor(); function createConverter() { const freeApngWorkers = []; const freeWebpWorkers = []; const MAX_CONVERT = 2; let isStop = false; let queue = []; let active = []; const cachedQueue = { gif: [], png: [] }; const depsUrl = { gif: '', png: '', webp: URL.createObjectURL(new Blob([api.getWebpWS()], { type: 'text/javascript' })) }; let LoadStatus; (function (LoadStatus) { LoadStatus[LoadStatus["unloaded"] = 0] = "unloaded"; LoadStatus[LoadStatus["loading"] = 1] = "loading"; LoadStatus[LoadStatus["loaded"] = 2] = "loaded"; })(LoadStatus || (LoadStatus = {})); const depsStatus = { gif: { loaded: LoadStatus.unloaded, load() { logger.info('开始加载gif依赖'); this.loaded = LoadStatus.loading; return api.getGifWS().then((str) => { depsUrl.gif = URL.createObjectURL(new Blob([str], { type: 'text/javascript' })); this.loaded = LoadStatus.loaded; logger.info('加载gif依赖完成'); }); } }, png: { loaded: LoadStatus.unloaded, load() { logger.info('开始加载png依赖'); this.loaded = LoadStatus.loading; return api.getApngWS().then((str) => { depsUrl.png = URL.createObjectURL(new Blob([str], { type: 'text/javascript' })); this.loaded = LoadStatus.loaded; logger.info('加载png依赖完成'); }); } } }; const convertTo = { webp: (frames, convertMeta) => { return new Promise((resolve, reject) => { let worker; let reuse = false; logger.time(convertMeta.id); if (freeWebpWorkers.length) { logger.info('Reuse webp workers.'); worker = freeWebpWorkers.shift(); reuse = true; } else { worker = new Worker(depsUrl.webp); } convertMeta.abort = () => { logger.timeEnd(convertMeta.id); logger.warn('Convert stop manually.' + convertMeta.id); reject(new CancelError()); convertMeta.isAborted = true; worker.terminate(); }; const workerLoad = new Promise((onLoaded) => { if (reuse) return onLoaded(); worker.onmessage = (evt) => { if (evt.data === 'ok') { logger.info('Webp worker loaded.'); onLoaded(); } }; }); const delays = convertMeta.source.framesInfo.map((frameInfo) => { return Number(frameInfo.delay); }); const data = []; let completed = 0; frames.forEach((frame, idx) => { const canvas = document.createElement('canvas'); const width = (canvas.width = frame.naturalWidth); const height = (canvas.height = frame.naturalHeight); const context = canvas.getContext('2d'); if (!context) return; context.drawImage(frame, 0, 0, width, height); data.push(new Promise((onFulfilled, onRejected) => { canvas.toBlob((blob) => { if (!blob) return onRejected(new TypeError('Convert failed when invoke canvas.toBlob() ' + idx)); blob.arrayBuffer().then((buffer) => { const u8a = new Uint8Array(buffer); onFulfilled(u8a); convertMeta.onProgress?.((++completed / frames.length) * 0.5, 'webp'); }); }, 'image/webp', 1); })); }); workerLoad .then(() => Promise.all(data)) .then((u8arrs) => { if (convertMeta.isAborted) return; logger.timeLog(convertMeta.id); worker.onmessage = (evt) => { const data = evt.data; if (typeof data !== 'object') { convertMeta.onProgress?.(0.5 + (evt.data / frames.length) * 0.5, 'webp'); } else { logger.timeEnd(convertMeta.id); freeWebpWorkers.push(worker); resolve(new Blob([evt.data], { type: 'image/webp' })); } }; worker.postMessage({ data: u8arrs, delays }); }, (reason) => { logger.timeLog(convertMeta.id); reject(reason); }); }); }, gif: (frames, convertMeta) => { return new Promise((resolve, reject) => { const gif = new GIF__default["default"]({ workers: 2, quality: 10, workerScript: depsUrl.gif }); convertMeta.abort = () => { gif.abort(); }; logger.info('Start convert:', convertMeta.id); logger.time(convertMeta.id); frames.forEach((frame, i) => { gif.addFrame(frame, { delay: convertMeta.source.framesInfo[i].delay }); }); gif.on('progress', (progress) => { if (typeof convertMeta.onProgress === 'function') convertMeta.onProgress(progress, 'gif'); }); gif.on('finished', (gifBlob) => { logger.timeEnd(convertMeta.id); resolve(gifBlob); }); gif.on('abort', () => { logger.timeEnd(convertMeta.id); logger.warn('Convert stop manually. ' + convertMeta.id); convertMeta.isAborted = true; reject(new CancelError()); }); gif.render(); }); }, png: (frames, convertMeta) => { return new Promise((resolve, reject) => { const canvas = document.createElement('canvas'); const width = (canvas.width = frames[0].naturalWidth); const height = (canvas.height = frames[0].naturalHeight); const context = canvas.getContext('2d', { willReadFrequently: true }); if (!context) return reject(new TypeError('Can not get canvas context')); const data = []; const delay = convertMeta.source.framesInfo.map((frameInfo) => { return Number(frameInfo.delay); }); logger.info('Start convert:', convertMeta.id); logger.time(convertMeta.id); for (const frame of frames) { if (convertMeta.isAborted) { logger.timeEnd(convertMeta.id); logger.warn('Convert stop manually. ' + convertMeta.id); return reject(new CancelError()); } context.clearRect(0, 0, width, height); context.drawImage(frame, 0, 0, width, height); data.push(context.getImageData(0, 0, width, height).data); } logger.timeLog(convertMeta.id); let worker; if (freeApngWorkers.length) { worker = freeApngWorkers.shift(); logger.info('Reuse apng workers.'); } else { worker = new Worker(depsUrl.png); } convertMeta.abort = () => { logger.timeEnd(convertMeta.id); logger.warn('Convert stop manually. ' + convertMeta.id); reject(new CancelError()); convertMeta.isAborted = true; worker.terminate(); }; worker.onmessage = function (e) { freeApngWorkers.push(worker); logger.timeEnd(convertMeta.id); if (!e.data) { return reject(new TypeError('Failed to get png data. ' + convertMeta.id)); } const pngBlob = new Blob([e.data], { type: 'image/png' }); resolve(pngBlob); }; const cfg = { data, width, height, delay }; worker.postMessage(cfg); }); }, webm: (frames, convertMeta) => { return new Promise((resolve, reject) => { const canvas = document.createElement('canvas'); const width = (canvas.width = frames[0].naturalWidth); const height = (canvas.height = frames[0].naturalHeight); const context = canvas.getContext('2d'); if (!context) return reject(new TypeError('Can not get canvas context')); const stream = canvas.captureStream(); const recorder = new MediaRecorder(stream, { mimeType: 'video/webm', videoBitsPerSecond: 80000000 }); const delay = convertMeta.source.framesInfo.map((frame) => { return Number(frame.delay); }); const data = []; let frame = 0; const displayFrame = () => { context.clearRect(0, 0, width, height); context.drawImage(frames[frame], 0, 0); if (convertMeta.isAborted) { return recorder.stop(); } setTimeout(() => { if (typeof convertMeta.onProgress === 'function') convertMeta.onProgress((frame + 1) / frames.length, 'webm'); if (frame === frames.length - 1) { return recorder.stop(); } else { frame++; } displayFrame(); }, delay[frame]); }; recorder.ondataavailable = (event) => { if (event.data && event.data.size) { data.push(event.data); } }; recorder.onstop = () => { if (convertMeta.isAborted) { logger.warn('Convert stop manually.' + convertMeta.id); return reject(new CancelError()); } resolve(new Blob(data, { type: 'video/webm' })); }; displayFrame(); recorder.start(); }); } }; const convert = (convertMeta) => { const { id, source, resolve, reject } = convertMeta; let frames; active.push(convertMeta); if (typeof convertMeta.onProgress === 'function') convertMeta.onProgress(0, 'zip'); compressor .unzip(source.data) .then((files) => { const imagePromises = files.map((blob) => { return new Promise((resolve) => { const image = new Image(); image.onload = () => { resolve(image); }; image.src = URL.createObjectURL(blob); }); }); return Promise.all(imagePromises); }) .then((imgEles) => { frames = imgEles; if (convertMeta.isAborted) { logger.warn('Convert stop manually.' + id); throw new CancelError(); } return convertTo[source.format](frames, convertMeta); }) .then(resolve, reject) .finally(() => { frames.forEach((frame) => URL.revokeObjectURL(frame.src)); active.splice(active.indexOf(convertMeta), 1); if (queue.length) convert(queue.shift()); }); }; return { add: (convertSource, handler) => { logger.info('Converter add', convertSource.id); return new Promise((resolve, reject) => { const meta = { id: convertSource.id, isAborted: false, source: convertSource, onProgress: handler?.onProgress, resolve, reject, abort() { this.isAborted = true; } }; const format = convertSource.format; if ((format === 'gif' || format === 'png') && depsStatus[format].loaded !== LoadStatus.loaded) { switch (depsStatus[format].loaded) { case LoadStatus.unloaded: cachedQueue[format].push(meta); depsStatus[format].load().then(() => { logger.info(`添加${cachedQueue[format].length}个任务`); queue.push(...cachedQueue[format]); cachedQueue[format].length = 0; while (active.length < MAX_CONVERT && queue.length && !isStop) { convert(queue.shift()); } }, (reason) => { cachedQueue[format].forEach((meta) => { meta.reject(reason); }); depsStatus[format].loaded = LoadStatus.unloaded; cachedQueue[format].length = 0; }); break; case LoadStatus.loading: cachedQueue[format].push(meta); break; default: throw new RangeError('Invalid deps status.'); } } else { queue.push(meta); while (active.length < MAX_CONVERT && queue.length && !isStop) { convert(queue.shift()); } } }); }, del: (taskIds) => { if (!taskIds.length) return; logger.info('Converter del, active:', active.map((meta) => meta.id), 'queue:', queue.map((meta) => meta.id)); isStop = true; active = active.filter((convertMeta) => { if (taskIds.includes(convertMeta.id)) { convertMeta.abort(); } else { return true; } }); queue = queue.filter((convertMeta) => !taskIds.includes(convertMeta.id)); isStop = false; while (active.length < MAX_CONVERT && queue.length) { convert(queue.shift()); } } }; } const converter = createConverter(); const updateDirHandleChannel = new BroadcastChannel('update_dir_channel'); updateDirHandleChannel.onmessage = (evt) => { const data = evt.data; switch (data.kind) { case 1 : dirHandleStatus = 1 ; logger.info('正在选择目录'); break; case 0 : logger.warn('取消更新dirHandle'); if (dirHandle) { dirHandleStatus = 2 ; processCachedSave(); } else { dirHandleStatus = 0 ; rejectCachedSave(); } break; case 2 : dirHandleStatus = 2 ; dirHandle = data.handle; logger.info('更新dirHandle', dirHandle); processCachedSave(); break; case 'request': if (dirHandle) { updateDirHandleChannel.postMessage({ kind: 'response', handle: dirHandle }); logger.info('响应请求dirHandle'); } break; case 'response': if (!dirHandle) { if (dirHandleStatus === 0 ) dirHandleStatus = 2 ; dirHandle = data.handle; logger.info('首次获取dirHandle', dirHandle); } break; default: throw new Error('Invalid data kind.'); } }; updateDirHandleChannel.postMessage({ kind: 'request' }); async function getDirHandleRecursive(dirs) { let handler = dirHandle; if (typeof dirs === 'string') { if (dirs.indexOf('/') === -1) return await handler.getDirectoryHandle(dirs, { create: true }); dirs = dirs.split('/').filter((dir) => !!dir); } for await (const dir of dirs) { handler = await handler.getDirectoryHandle(dir, { create: true }); } return handler; } const duplicateFilenameCached = {}; async function getFilenameHandle(dirHandle, filename) { const conflictAction = config.get('fileSystemFilenameConflictAction'); if (conflictAction === 'overwrite') return await dirHandle.getFileHandle(filename, { create: true }); if (!(filename in duplicateFilenameCached)) { duplicateFilenameCached[filename] = []; try { await dirHandle.getFileHandle(filename); logger.warn('存在同名文件', filename); } catch (error) { return await dirHandle.getFileHandle(filename, { create: true }); } } const extIndex = filename.lastIndexOf('.'); const ext = filename.slice(extIndex + 1); const name = filename.slice(0, extIndex); if (conflictAction === 'prompt') { return await unsafeWindow.showSaveFilePicker({ suggestedName: filename, types: [{ description: 'Image file', accept: { ['image/' + ext]: ['.' + ext] } }] }); } else { for (let suffix = 1; suffix < 1000; suffix++) { const newName = `${name} (${suffix}).${ext}`; try { await dirHandle.getFileHandle(newName); } catch (error) { if (duplicateFilenameCached[filename].includes(newName)) { continue; } else { duplicateFilenameCached[filename].push(newName); } logger.info('使用文件名:', newName); return await dirHandle.getFileHandle(newName, { create: true }); } } throw new RangeError('Oops, you have too many duplicate files.'); } } function clearFilenameCached(duplicateName, actualName) { if (!(duplicateName in duplicateFilenameCached)) return; const usedNameArr = duplicateFilenameCached[duplicateName]; logger.info('清理重名文件名', usedNameArr, actualName); if (usedNameArr.length === 0) { delete duplicateFilenameCached[duplicateName]; return; } const index = usedNameArr.indexOf(actualName); if (index === -1) return; usedNameArr.splice(index, 1); if (usedNameArr.length === 0) delete duplicateFilenameCached[duplicateName]; } async function updateDirHandle() { try { dirHandleStatus = 1 ; updateDirHandleChannel.postMessage({ kind: 1 }); dirHandle = await unsafeWindow.showDirectoryPicker({ id: 'pdl', mode: 'readwrite' }); logger.info('更新dirHandle', dirHandle); dirHandleStatus = 2 ; updateDirHandleChannel.postMessage({ kind: 2 , handle: dirHandle }); processCachedSave(); return true; } catch (error) { logger.warn(error); updateDirHandleChannel.postMessage({ kind: 0 }); if (dirHandle) { dirHandleStatus = 2 ; processCachedSave(); } else { dirHandleStatus = 0 ; rejectCachedSave(); } return false; } } let dirHandleStatus = 0 ; let dirHandle; const cachedSaveProcess = []; async function saveWithFileSystemAccess(blob, downloadMeta) { try { if (downloadMeta.state === 0 ) return; if (dirHandleStatus === 1 ) { cachedSaveProcess.push([blob, downloadMeta]); return; } if (dirHandleStatus === 0 ) { const isSuccess = await updateDirHandle(); if (!isSuccess) throw new TypeError('Failed to get dir handle.'); } let currenDirHandle; let filename; const path = downloadMeta.source.path; const index = path.lastIndexOf('/'); if (index === -1) { filename = path; currenDirHandle = dirHandle; } else { filename = path.slice(index + 1); currenDirHandle = await getDirHandleRecursive(path.slice(0, index)); } const fileHandle = await getFilenameHandle(currenDirHandle, filename); const writableStream = await fileHandle.createWritable(); await writableStream.write(blob); await writableStream.close(); clearFilenameCached(filename, fileHandle.name); downloadMeta.resolve(downloadMeta.taskId); logger.info('Download complete:', downloadMeta.source.path); } catch (error) { downloadMeta.reject(error); logger.error(error); } downloadMeta.state = 2 ; } function processCachedSave() { cachedSaveProcess.forEach((args) => saveWithFileSystemAccess(...args)); logger.info(`执行${cachedSaveProcess.length}个缓存任务`); cachedSaveProcess.length = 0; } function rejectCachedSave() { cachedSaveProcess.forEach(([, downloadMeta]) => downloadMeta.reject(new CancelError())); logger.info(`取消${cachedSaveProcess.length}个缓存任务`); cachedSaveProcess.length = 0; } function getCurrentDirName() { return dirHandle?.name ?? ''; } function isShouldGetDirHandle() { return isUseFileSystemAccess() && dirHandleStatus === 0 ; } function isUseFileSystemAccess() { return env.isFileSystemAccessAvaliable() && config.get('useFileSystemAccess'); } const _saveWithoutSubpath = (blob, downloadMeta) => { const dlEle = document.createElement('a'); dlEle.href = URL.createObjectURL(blob); dlEle.download = downloadMeta.source.path; dlEle.click(); URL.revokeObjectURL(dlEle.href); downloadMeta.state = 2 ; downloadMeta.resolve(downloadMeta.taskId); }; const _saveWithSubpath = (blob, downloadMeta) => { const imgUrl = URL.createObjectURL(blob); const request = { url: imgUrl, name: downloadMeta.source.path, onerror: (error) => { if (downloadMeta.state !== 0 ) { downloadMeta.reject(new Error(`Download error when saving ${downloadMeta.source.path} because ${error.error} ${error.details ?? ''} `)); } URL.revokeObjectURL(imgUrl); }, onload: () => { if (typeof downloadMeta.onLoad === 'function') downloadMeta.onLoad(); URL.revokeObjectURL(imgUrl); downloadMeta.state = 2 ; downloadMeta.resolve(downloadMeta.taskId); logger.info('Download complete:', downloadMeta.source.path); } }; downloadMeta.abort = GM_download(request).abort; }; function createDownloader() { const MAX_DOWNLOAD = 5; const MAX_RETRY = 3; const INTERVAL = 500; const TIMEOUT = 20000; let isStop = false; let queue = []; let active = []; let save; if (env.isBlobDlAvaliable() && env.isSupportSubpath()) { save = _saveWithSubpath; } else { logger.warn('Download function not full support:', GM_info.scriptHandler, GM_info.version); save = _saveWithoutSubpath; } const download = (downloadMeta) => { const { taskId, source } = downloadMeta; logger.info('Start download:', source.path); active.push(downloadMeta); let fileSaveFn = save; const isUseFSA = isUseFileSystemAccess(); if (isUseFSA) { fileSaveFn = saveWithFileSystemAccess; } let abortObj; const errorHandler = errorHandlerFactory(downloadMeta); if ((!env.isBlobDlAvaliable() || (env.isViolentmonkey() && !isUseFSA)) && !('kind' in source)) { abortObj = GM_download({ url: source.src, name: source.path, headers: { referer: 'https://www.pixiv.net' }, ontimeout: errorHandler, onerror: errorHandler, onload: async () => { logger.info('Download complete:', taskId, source.path); if (typeof downloadMeta.onLoad === 'function') downloadMeta.onLoad(); downloadMeta.resolve(taskId); await sleep(INTERVAL); active.splice(active.indexOf(downloadMeta), 1); if (queue.length && !isStop) download(queue.shift()); } }); } else { abortObj = GM_xmlhttpRequest({ url: source.src, timeout: TIMEOUT, method: 'GET', headers: { referer: 'https://www.pixiv.net' }, responseType: 'blob', ontimeout: errorHandler, onerror: errorHandler, onprogress: (e) => { if (e.lengthComputable && typeof downloadMeta.onProgress === 'function') { downloadMeta.onProgress(e.loaded / e.total); } }, onload: async (e) => { logger.info('Xhr complete:', source.path); if (downloadMeta.state === 0 ) return logger.warn('Download was canceled.', taskId, source.path); if (!('kind' in source)) { fileSaveFn(e.response, downloadMeta); } else if (source.kind === 'convert') { const convertSource = { id: taskId, data: e.response, format: source.format, framesInfo: source.ugoiraMeta?.frames }; converter.add(convertSource, { onProgress: downloadMeta.onProgress }).then((blob) => { fileSaveFn(blob, downloadMeta); }, downloadMeta.reject); } else if (source.kind === 'bundle') { compressor.add(taskId, source.filename, e.response); if (compressor.fileCount(taskId) === source.pageCount) { compressor.bundle(taskId).then((blob) => { fileSaveFn(blob, downloadMeta); compressor.remove(taskId); }); } else { downloadMeta.resolve(taskId); if (typeof downloadMeta.onLoad === 'function') downloadMeta.onLoad(); } } await sleep(INTERVAL); active.splice(active.indexOf(downloadMeta), 1); if (queue.length && !isStop) download(queue.shift()); } }); } downloadMeta.abort = abortObj.abort; }; function errorHandlerFactory(downloadMeta) { return function (error) { const { taskId, source, state } = downloadMeta; if (state === 0 ) return; if (error) { logger.error('Download ' + taskId + ' error', error.error ? ' with reason: ' + error.error : '', 'details' in error ? error.details : error); if ('status' in error && error.status === 429) { downloadMeta.reject(new RequestError('Too many request', error)); active.splice(active.indexOf(downloadMeta), 1); return; } } else { logger.warn('Download timeout:', source.src); } downloadMeta.retry++; if (downloadMeta.retry > MAX_RETRY) { downloadMeta.reject(new Error('Download failed: ' + source.src)); active.splice(active.indexOf(downloadMeta), 1); if (queue.length && !isStop) download(queue.shift()); } else { logger.info('Download retry:', downloadMeta.retry, source.src); download(downloadMeta); } }; } return { add(metas, handler = {}) { logger.info('Downloader add:', metas); if (metas.length < 1) return Promise.resolve(''); const promises = []; metas.forEach((source) => { promises.push(new Promise((resolve, reject) => { const downloadMeta = { taskId: source.taskId, source, state: 1 , retry: 0, onProgress: handler.onProgress, onLoad: handler.onLoad, resolve, reject }; queue.push(downloadMeta); })); }); while (active.length < MAX_DOWNLOAD && queue.length && !isStop) { download(queue.shift()); } return Promise.all(promises).then(([taskId]) => taskId); }, del(taskIds) { if (!taskIds.length) return; isStop = true; logger.info('Downloader delete. active:', active.length, 'queue', queue.length); active = active.filter((downloadMeta) => { if (taskIds.includes(downloadMeta.taskId) && downloadMeta.state !== 2 ) { downloadMeta.abort?.(); downloadMeta.state = 0 ; downloadMeta.reject(new CancelError()); logger.warn('Download abort manually.', downloadMeta.source.path); } else { return true; } }); converter.del(taskIds); compressor.remove(taskIds); queue = queue.filter((downloadMeta) => !taskIds.includes(downloadMeta.taskId)); isStop = false; while (active.length < MAX_DOWNLOAD && queue.length) { download(queue.shift()); } } }; } const downloader = createDownloader(); const needBundle = (type) => { return ((type === IllustType.manga && config.get('bundleManga')) || (type === IllustType.illusts && config.get('bundleIllusts'))); }; const getFilePath = ({ user, userId, title, tagStr, illustId, createDate, page, ext }, option = { needBundle: false, needConvert: false }) => { let pathPattern; const folderPattern = config.get('folderPattern'); const filenamePattern = config.get('filenamePattern'); const isUseFSA = isUseFileSystemAccess(); const shouldAddFolder = !!folderPattern && !option.needBundle && ((env.isSupportSubpath() && (!option.needConvert || env.isBlobDlAvaliable())) || isUseFSA); if (shouldAddFolder) { pathPattern = folderPattern + '/' + filenamePattern; } else { pathPattern = filenamePattern; } if (option.needBundle && !filenamePattern.includes('{page}')) { pathPattern += '_{page}'; } function replaceDate(match, p1) { const format = p1 || 'YYYY-MM-DD'; return dayjs__default["default"](createDate).format(format); } return (pathPattern .replaceAll(/\{date\((.*?)\)\}|\{date\}/g, replaceDate) .replaceAll('{artist}', user) .replaceAll('{artistID}', userId) .replaceAll('{title}', title) .replaceAll('{tags}', tagStr) .replaceAll('{page}', String(page)) .replaceAll('{id}', illustId) + ext); }; const makeTagsStr = (prev, cur, index, tagsArr) => { const tag = config.get('tagLang') === 'ja' ? cur.tag : cur.translation?.['en'] || cur.tag; if (index < tagsArr.length - 1) { return prev + tag + '_'; } else { return prev + tag; } }; function isValidIllustType(illustType, option) { switch (illustType) { case IllustType.illusts: if (option.filterIllusts) return true; break; case IllustType.manga: if (option.filterManga) return true; break; case IllustType.ugoira: if (option.filterUgoria) return true; break; default: throw new Error('Invalid filter type'); } return false; } function filterWorks(works, option) { const obj = { unavaliable: [], avaliable: [], invalid: [] }; works.forEach((work) => { if (!work.isBookmarkable) { obj.unavaliable.push(work.id); } else if (option.filterExcludeDownloaded && pixivHistory.has(work.id)) { obj.invalid.push(work.id); } else if (!isValidIllustType(work.illustType, option)) { obj.invalid.push(work.id); } else { obj.avaliable.push(work.id); } }); return obj; } async function getFollowLatestGenerator(filterOption, mode, page) { const MAX_PAGE = 34; const MAX_ILLUSTS_PER_PAGE = 60; let lastId; let total; let data; let cache; function findLastId(ids) { return Math.min(...ids.map((id) => Number(id))); } if (page === undefined) { data = await api.getFollowLatestWorks(1, mode); const ids = data.page.ids; total = ids.length; lastId = findLastId(ids); if (total === MAX_ILLUSTS_PER_PAGE) { const secondPageData = await api.getFollowLatestWorks(2, mode); const secondIds = secondPageData.page.ids; const secondLastId = findLastId(secondIds); if (secondLastId < lastId) { lastId = secondLastId; cache = secondPageData; total += secondIds.length; } } } else { data = await api.getFollowLatestWorks(page, mode); total = data.page.ids.length; } async function* generateIds() { yield filterWorks(data.thumbnails.illust, filterOption); if (page === undefined) { if (total === MAX_ILLUSTS_PER_PAGE) return; if (total < MAX_ILLUSTS_PER_PAGE * 2) { yield filterWorks(cache.thumbnails.illust, filterOption); return; } let currentPage = 3; while (currentPage <= MAX_PAGE) { const data = await api.getFollowLatestWorks(currentPage, mode); const ids = data.page.ids; const pageLastId = findLastId(ids); if (pageLastId >= lastId) { logger.info('getFollowLatestGenerator: got duplicate works'); yield filterWorks(cache.thumbnails.illust, filterOption); break; } lastId = pageLastId; total += ids.length; yield { ...filterWorks(cache.thumbnails.illust, filterOption), total }; cache = data; currentPage++; await sleep(3000); } } } return { total, generator: generateIds() }; } async function getChunksGenerator(userId, category, tag, rest, filterOption) { const OFFSET = 48; let requestUrl; if (category === 'bookmarks') { requestUrl = `https://www.pixiv.net/ajax/user/${userId}/illusts/bookmarks?tag=${tag}&offset=0&limit=${OFFSET}&rest=${rest}&lang=ja`; } else { requestUrl = `https://www.pixiv.net/ajax/user/${userId}/${category}/tag?tag=${tag}&offset=0&limit=${OFFSET}&lang=ja`; } let head = 0; const firstPageData = await api.getJson(requestUrl); const total = firstPageData.total; async function* generateIds() { yield filterWorks(firstPageData.works, filterOption); head += OFFSET; while (head < total) { const data = await api.getJson(requestUrl.replace('offset=0', 'offset=' + head)); head += OFFSET; await sleep(3000); yield filterWorks(data.works, filterOption); } } return { total, generator: generateIds() }; } async function getAllWorksGenerator(userId, filterOption) { const profile = await api.getUserAllProfile(userId); let illustIds = []; let mangaIds = []; if ((filterOption.filterIllusts || filterOption.filterUgoria) && typeof profile.illusts === 'object') { illustIds.push(...Object.keys(profile.illusts).reverse()); } if (filterOption.filterManga && typeof profile.manga === 'object') { mangaIds.push(...Object.keys(profile.manga).reverse()); } if (filterOption.filterExcludeDownloaded) { illustIds = illustIds.filter((id) => !pixivHistory.has(id)); mangaIds = mangaIds.filter((id) => !pixivHistory.has(id)); } async function* generateIds() { const OFFSET = 48; const baseUrl = 'https://www.pixiv.net/ajax/user/' + userId + '/profile/illusts'; let workCategory = 'illust'; while (illustIds.length > 0) { let searchStr = '?'; const chunk = illustIds.splice(0, OFFSET); searchStr += chunk.map((id) => 'ids[]=' + id).join('&') + `&work_category=${workCategory}&is_first_page=0&lang=ja`; const data = await api.getJson(baseUrl + searchStr); await sleep(3000); yield filterWorks(Object.values(data.works).reverse(), filterOption); } workCategory = 'manga'; while (mangaIds.length > 0) { let searchStr = '?'; const chunk = mangaIds.splice(0, OFFSET); searchStr += chunk.map((id) => 'ids[]=' + id).join('&') + `&work_category=${workCategory}&is_first_page=0&lang=ja`; const data = await api.getJson(baseUrl + searchStr); await sleep(3000); yield filterWorks(Object.values(data.works).reverse(), filterOption); } } return { total: illustIds.length + mangaIds.length, generator: generateIds() }; } async function getArtworkData(illustId) { const htmlText = await api.getArtworkHtml(illustId); const preloadDataText = htmlText.match(regexp.preloadData); if (!preloadDataText) throw new Error('Fail to parse preload data.'); const preloadData = JSON.parse(preloadDataText[1]); const illustData = preloadData.illust[illustId]; const globalDataText = htmlText.match(regexp.globalData); if (!globalDataText) throw new Error('Fail to parse global data.'); const globalData = JSON.parse(globalDataText[1]); let ugoiraMeta; if (illustData.illustType === IllustType.ugoira) { ugoiraMeta = await api.getUgoriaMeta(illustId); } return { illustData, globalData, ugoiraMeta }; } async function getAjaxArtworkData(illustId) { const illustData = await api.getArtworkDetail(illustId); let ugoiraMeta; if (illustData.illustType === IllustType.ugoira) { ugoiraMeta = await api.getUgoriaMeta(illustId); } return { illustData, ugoiraMeta }; } function getDownloadSource(artworkData, seletedPage) { const { illustData, ugoiraMeta } = artworkData; const { illustType, userName, userId, illustTitle, illustId, tags, pageCount, createDate } = illustData; const pathInfo = { user: replaceInvalidChar(unescapeHtml(userName)) || 'userId-' + userId, title: replaceInvalidChar(unescapeHtml(illustTitle)) || 'illustId-' + illustId, tagStr: replaceInvalidChar(unescapeHtml(tags.tags.reduce(makeTagsStr, ''))), illustId, userId, createDate, ext: '', page: 0 }; const metas = []; const taskId = illustId + '_' + Math.random().toString(36).slice(2); if (illustType === IllustType.illusts || illustType === IllustType.manga) { const firstImgSrc = illustData.urls.original; const srcPrefix = firstImgSrc.slice(0, firstImgSrc.indexOf('_') + 2); const extendName = firstImgSrc.slice(-4); pathInfo.ext = extendName; if (pageCount > 1 && seletedPage === undefined) { if (needBundle(illustType)) { const path = getFilePath({ ...pathInfo, ext: '.zip', page: pageCount }); for (let i = 0; i < pageCount; i++) { pathInfo.page = i; metas.push({ kind: 'bundle', taskId, path, src: srcPrefix + i + extendName, filename: getFilePath(pathInfo, { needBundle: true }), pageCount }); } } else { for (let i = 0; i < pageCount; i++) { pathInfo.page = i; metas.push({ taskId, path: getFilePath(pathInfo), src: srcPrefix + i + extendName }); } } } else { let src = firstImgSrc; if (seletedPage !== undefined) { src = srcPrefix + seletedPage + extendName; pathInfo.page = seletedPage; } metas.push({ taskId, path: getFilePath(pathInfo), src }); } } else if (illustType === IllustType.ugoira && ugoiraMeta) { const ugoriaFormat = config.get('ugoriaFormat'); pathInfo.ext = '.' + ugoriaFormat; if (ugoriaFormat !== 'zip') { metas.push({ kind: 'convert', format: ugoriaFormat, ugoiraMeta, taskId, src: ugoiraMeta.originalSrc, path: getFilePath(pathInfo, { needConvert: true }) }); } else { metas.push({ taskId, src: ugoiraMeta.originalSrc, path: getFilePath(pathInfo) }); } } return metas; } const parser = { getChunksGenerator, getAllWorksGenerator, getFollowLatestGenerator, getArtworkData, getDownloadSource, getAjaxArtworkData }; function handleDownload(pdlBtn, illustId) { if (isShouldGetDirHandle()) updateDirHandle(); let pageCount; let pageComplete = 0; let shouldDownloadPage; let downloading = true; const pageAttr = pdlBtn.getAttribute('should-download'); if (pageAttr) { shouldDownloadPage = Number(pageAttr); } const onProgress = (progress = 0, type = null) => { if (pageCount > 1 || !downloading) return; progress = Math.floor(progress * 100); switch (type) { case null: pdlBtn.style.setProperty('--pdl-progress', progress + '%'); case 'gif': case 'webm': case 'webp': pdlBtn.textContent = String(progress); break; case 'zip': pdlBtn.textContent = ''; break; } }; const onLoad = function () { if (pageCount < 2 || !downloading) return; const progress = Math.floor((++pageComplete / pageCount) * 100); pdlBtn.textContent = String(progress); pdlBtn.style.setProperty('--pdl-progress', progress + '%'); }; pdlBtn.classList.add('pdl-progress'); parser .getArtworkData(illustId) .then((artworkData) => { const { illustData, globalData } = artworkData; const { illustId, tags, bookmarkData } = illustData; if (!bookmarkData) { const { token } = globalData; const tagsArr = tags.tags.map((item) => item.tag); addBookmark(pdlBtn, illustId, token, tagsArr); } return parser.getDownloadSource(artworkData, shouldDownloadPage); }) .then((sources) => { pageCount = sources.length; return downloader.add(sources, { onLoad, onProgress }); }) .then(() => { pixivHistory.add(illustId); pdlBtn.classList.remove('pdl-error'); pdlBtn.classList.add('pdl-complete'); }) .catch((err) => { if (err) logger.error(err); pdlBtn.classList.remove('pdl-complete'); pdlBtn.classList.add('pdl-error'); }) .finally(() => { downloading = false; pdlBtn.innerHTML = ''; pdlBtn.style.removeProperty('--pdl-progress'); pdlBtn.classList.remove('pdl-progress'); }); } function createPdlBtn(attributes, textContent = '', { addEvent } = { addEvent: true }) { const ele = document.createElement('button'); ele.textContent = textContent; if (!attributes) return ele; const { attrs, classList } = attributes; if (classList && classList.length > 0) { for (const cla of classList) { ele.classList.add(cla); } } if (attrs) { for (const key in attrs) { ele.setAttribute(key, attrs[key]); } } if (addEvent) { ele.addEventListener('click', (evt) => { evt.preventDefault(); evt.stopPropagation(); const ele = evt.currentTarget; if (!ele.classList.contains('pdl-progress')) { handleDownload(ele, ele.getAttribute('pdl-id')); } }); } return ele; } function createThumbnailsBtn(nodes) { let isSelfBookmark = false; const inBookmarkPage = regexp.bookmarkPage.exec(location.pathname); if (inBookmarkPage) { inBookmarkPage[1] === getSelfId() && (isSelfBookmark = true); } nodes.forEach((e) => { if (e.childElementCount !== 0 && !e.querySelector('.pdl-btn-sub')) { const illustId = getIllustId(e); if (illustId) { const attrs = { attrs: { 'pdl-id': illustId }, classList: ['pdl-btn', 'pdl-btn-sub'] }; if (pixivHistory.has(illustId)) attrs.classList.push('pdl-complete'); if (isSelfBookmark) attrs.classList.push('self-bookmark'); e.appendChild(createPdlBtn(attrs)); } } }); } function fixPixivPreviewer(nodes) { const isPpSearchPage = regexp.ppSearchPage.test(location.pathname); if (!isPpSearchPage) return; nodes.forEach((node) => { const pdlEle = node.querySelector('.pdl-btn'); if (!pdlEle) return false; pdlEle.remove(); }); } function getFilterOption() { return { filterExcludeDownloaded: config.get('filterExcludeDownloaded'), filterIllusts: config.get('filterIllusts'), filterManga: config.get('filterManga'), filterUgoria: config.get('filterUgoria') }; } function downloadAndRetry(chunksGenerators) { useDownloadBar(chunksGenerators).then((failed) => { if (failed instanceof Array && failed.length) { const gen = async function* () { yield { avaliable: failed, unavaliable: [], invalid: [] }; }; console.log('[Pixiv Downloader] Retry...'); useDownloadBar(Promise.resolve({ total: failed.length, generator: gen() })); } }); } function downloadWorks(evt) { evt.preventDefault(); evt.stopPropagation(); if (isDownloading) return; const btn = evt.target; const userId = btn.getAttribute('pdl-userid'); const filterOption = getFilterOption(); if (isShouldGetDirHandle()) updateDirHandle(); const ids = parser.getAllWorksGenerator(userId, filterOption); downloadAndRetry(ids); } async function downloadBookmarksOrTags(evt) { evt.preventDefault(); evt.stopPropagation(); if (isDownloading) return; const btn = evt.target; const userId = btn.getAttribute('pdl-userid'); const category = btn.getAttribute('category'); const tag = btn.getAttribute('tag') || ''; const rest = (btn.getAttribute('rest') || 'show'); if (isShouldGetDirHandle()) updateDirHandle(); const filterOption = getFilterOption(); let idsGenerators; if (rest === 'all') { const idsShowPromise = parser.getChunksGenerator(userId, 'bookmarks', '', 'show', filterOption); const idsHidePromise = parser.getChunksGenerator(userId, 'bookmarks', '', 'hide', filterOption); idsGenerators = [idsShowPromise, idsHidePromise]; } else { idsGenerators = parser.getChunksGenerator(userId, category, tag, rest, filterOption); } downloadAndRetry(idsGenerators); } function downloadFollowLatest(evt) { evt.preventDefault(); evt.stopPropagation(); if (isDownloading) return; const btn = evt.target; const mode = location.pathname.includes('r18') ? 'r18' : 'all'; const filterOption = getFilterOption(); let idsGenerators; if (btn.classList.contains('pdl-dl-all')) { idsGenerators = parser.getFollowLatestGenerator(filterOption, mode); } else { const params = new URLSearchParams(location.search); const page = Number(params.get('p')) || 1; idsGenerators = parser.getFollowLatestGenerator(filterOption, mode, page); } downloadAndRetry(idsGenerators); } const dlBarRef = { filter: { filterExcludeDownloaded: undefined, filterIllusts: undefined, filterManga: undefined, filterUgoria: undefined }, statusBar: undefined, abortBtn: undefined }; function updateStatus(str) { dlBarRef.statusBar && (dlBarRef.statusBar.textContent = str); } function createFilterEl(id, filterType, text) { const checkbox = document.createElement('input'); const label = document.createElement('label'); checkbox.id = id; checkbox.type = 'checkbox'; checkbox.classList.add('pdl-checkbox'); checkbox.setAttribute('category', String(filterType)); checkbox.checked = config.get(filterType); label.setAttribute('for', id); label.setAttribute('category', String(filterType)); label.textContent = text; checkbox.addEventListener('change', (evt) => { const checkbox = evt.currentTarget; const category = checkbox.getAttribute('category'); config.set(category, checkbox.checked); }); dlBarRef.filter[filterType] = checkbox; const wrap = document.createElement('div'); wrap.classList.add('pdl-filter'); wrap.appendChild(checkbox); wrap.appendChild(label); return wrap; } function createFilter() { const wrapper = document.createElement('div'); wrapper.classList.add('pdl-filter-wrap'); wrapper.appendChild(createFilterEl('pdl-filter-exclude_downloaded', 'filterExcludeDownloaded', t('downloadBar.filter.exclude_downloaded'))); wrapper.appendChild(createFilterEl('pdl-filter-illusts', 'filterIllusts', t('downloadBar.filter.illusts'))); wrapper.appendChild(createFilterEl('pdl-filter-manga', 'filterManga', t('downloadBar.filter.manga'))); wrapper.appendChild(createFilterEl('pdl-filter-ugoria', 'filterUgoria', t('downloadBar.filter.ugoria'))); return wrapper; } function createDownloadBar(userId) { const nav = document.querySelector('nav[class~="sc-192ftwf-0"]'); if (!nav) return; const dlBtn = nav.querySelector('.pdl-btn-all'); if (dlBtn) { if (dlBtn.getAttribute('pdl-userid') === userId) return; removeDownloadBar(); } const dlBar = document.createElement('div'); dlBar.classList.add('pdl-dlbar'); const statusBar = document.createElement('div'); statusBar.classList.add('pdl-dlbar-status_bar'); dlBarRef.statusBar = dlBar.appendChild(statusBar); const baseClasses = nav.querySelector('a:not([aria-current])').classList; dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({ attrs: { 'pdl-userId': userId }, classList: [...baseClasses, 'pdl-stop', 'pdl-hide'] }, t('downloadBar.button.stop'), { addEvent: false })); if (userId !== getSelfId()) { const hasWorks = ["a[href$='illustrations']", "a[href$='manga']"].some((selector) => !!nav.querySelector(selector)); if (hasWorks) { const el = createPdlBtn({ attrs: { 'pdl-userid': userId }, classList: [...baseClasses, 'pdl-btn-all'] }, t('downloadBar.button.works'), { addEvent: false }); el.addEventListener('click', downloadWorks); dlBar.appendChild(el); } if (nav.querySelector("a[href*='bookmarks']")) { const el = createPdlBtn({ attrs: { 'pdl-userid': userId, category: 'bookmarks' }, classList: [...baseClasses, 'pdl-btn-all'] }, t('downloadBar.button.bookmarks'), { addEvent: false }); el.addEventListener('click', downloadBookmarksOrTags); dlBar.appendChild(el); } } else { if (nav.querySelector("a[href*='bookmarks']")) { dlBar.appendChild(createPdlBtn({ attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'all' }, classList: [...baseClasses, 'pdl-btn-all'] }, t('downloadBar.button.bookmarks'), { addEvent: false })); dlBar.appendChild(createPdlBtn({ attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'show' }, classList: [...baseClasses, 'pdl-btn-all'] }, t('downloadBar.button.bookmarks_public'), { addEvent: false })); dlBar.appendChild(createPdlBtn({ attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'hide' }, classList: [...baseClasses, 'pdl-btn-all'] }, t('downloadBar.button.bookmarks_private'), { addEvent: false })); dlBar.querySelectorAll('.pdl-btn-all').forEach((node) => { node.addEventListener('click', downloadBookmarksOrTags); }); } } const filter = createFilter(); nav.parentElement.insertBefore(filter, nav); nav.appendChild(dlBar); } function removeDownloadBar() { const dlBarWrap = document.querySelector('.pdl-dlbar'); if (dlBarWrap) { dlBarWrap.remove(); document.querySelector('.pdl-filter-wrap')?.remove(); } } function updateFollowLatestDownloadBarBtnText(prevDlBtn, prevDlAllBtn) { if (location.pathname.includes('r18') && prevDlBtn.textContent !== t('downloadBar.button.r18_one_page')) { prevDlBtn.textContent = t('downloadBar.button.r18_one_page'); prevDlAllBtn.textContent = t('downloadBar.button.r18'); } else if (!location.pathname.includes('r18') && prevDlBtn.textContent !== t('downloadBar.button.all_one_page')) { prevDlBtn.textContent = t('downloadBar.button.all_one_page'); prevDlAllBtn.textContent = t('downloadBar.button.all'); } } function createFollowLatestDownloadBar() { const prevDlBtn = document.querySelector('.pdl-btn-all'); if (prevDlBtn) { const prevDlAllBtn = document.querySelector('.pdl-dl-all'); updateFollowLatestDownloadBarBtnText(prevDlBtn, prevDlAllBtn); return; } const nav = document.querySelector('nav'); if (!nav || nav.parentElement.childElementCount === 1) return; const navBar = nav.parentElement; const modeSwitch = nav.nextElementSibling; const filter = createFilter(); navBar.parentElement.insertBefore(filter, navBar); const dlBar = document.createElement('div'); dlBar.classList.add('pdl-dlbar'); dlBar.classList.add('pdl-dlbar-follow_latest'); const statusBar = document.createElement('div'); statusBar.classList.add('pdl-dlbar-status_bar'); dlBarRef.statusBar = dlBar.appendChild(statusBar); const baseClasses = nav.querySelector('a:not([aria-current])').classList; dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({ attrs: { 'pdl-userid': '' }, classList: [...baseClasses, 'pdl-stop', 'pdl-hide'] }, t('downloadBar.button.stop'), { addEvent: false })); const dlBtn = createPdlBtn({ attrs: { 'pdl-userid': '' }, classList: [...baseClasses, 'pdl-btn-all'] }, t('downloadBar.button.works'), { addEvent: false }); dlBtn.addEventListener('click', downloadFollowLatest); dlBar.appendChild(dlBtn); const dlAllBtn = createPdlBtn({ attrs: { 'pdl-userid': '' }, classList: [...baseClasses, 'pdl-btn-all', 'pdl-dl-all'] }, t('downloadBar.button.works'), { addEvent: false }); dlAllBtn.addEventListener('click', downloadFollowLatest); dlBar.appendChild(dlAllBtn); navBar.insertBefore(dlBar, modeSwitch); } function changeDlbarDisplay() { document.querySelectorAll('.pdl-dlbar .pdl-btn-all').forEach((ele) => { ele.classList.toggle('pdl-hide'); }); document.querySelector('.pdl-dlbar .pdl-stop')?.classList.toggle('pdl-hide'); document.querySelectorAll('.pdl-tag').forEach((ele) => { ele.classList.toggle('pdl-tag-hide'); }); document.querySelector('.pdl-filter-wrap')?.classList.toggle('unavailable'); } function onProgressCB(progressData) { if (typeof progressData === 'string') { updateStatus(progressData); } else { logger.info('Update progress by', progressData.illustId, ', completed: ', progressData.completed); updateStatus(`Downloading: ${progressData.completed} / ${progressData.avaliable}`); } } async function downloadByIds(total, idsGenerators, signal, onProgress) { signal.throwIfAborted(); const failed = []; const unavaliable = []; const invalid = []; const tasks = []; let completed = 0; let tooManyRequests = false; let wakeTooManyRequest; let wakeInterval; let resolve; let reject; const done = new Promise((r, j) => { resolve = r; reject = j; }); signal.addEventListener('abort', () => { if (tasks.length) { downloader.del(tasks); tasks.length = 0; } wakeTooManyRequest?.(); wakeInterval?.(); reject(signal.aborted ? signal.reason : 'unexpected generator error'); }, { once: true }); const afterEach = (illustId) => { const avaliable = total - failed.length - unavaliable.length - invalid.length; onProgress({ illustId, avaliable, completed }); if (completed === avaliable) { resolve({ failed, unavaliable }); } }; onProgress('Downloading...'); try { for (const idsGenerator of idsGenerators) { for await (const ids of idsGenerator) { logger.info('Got ids:', ids); signal.throwIfAborted(); if (ids.unavaliable.length) { unavaliable.push(...ids.unavaliable); } if (ids.invalid.length) { invalid.push(...ids.invalid); } if (typeof ids.total === 'number' && !Number.isNaN(ids.total)) { total = ids.total; } if (ids.avaliable.length) { for (const id of ids.avaliable) { signal.throwIfAborted(); if (tooManyRequests) { onProgress('Too many requests, wait 30s'); const { wake, sleep } = wakeableSleep(30000); wakeTooManyRequest = wake; await sleep; signal.throwIfAborted(); tooManyRequests = false; onProgress('Downloading...'); } parser .getAjaxArtworkData(id) .then((data) => { signal.throwIfAborted(); const sources = parser.getDownloadSource(data); tasks.push(sources[0].taskId); return downloader.add(sources); }) .then((taskId) => { pixivHistory.add(id); if (!signal.aborted) { tasks.splice(tasks.indexOf(taskId), 1); completed++; afterEach(id); } }, (reason) => { if (!signal.aborted) { reason && logger.error(reason); if (reason instanceof RequestError && reason.response.status === 429) { tooManyRequests = true; } if (reason instanceof JsonDataError) { unavaliable.push(id); } else { failed.push(id); } afterEach(id); } }); const { wake, sleep } = wakeableSleep(1000); wakeInterval = wake; await sleep; } } else { afterEach('no avaliable id'); } } } } catch (error) { if (!signal.aborted) { done.catch((reason) => { logger.info('catch unexpected abort: ', reason); }); signal.dispatchEvent(new Event('abort')); throw error; } } return done; } let isDownloading = false; async function useDownloadBar(chunksGenerators) { if (!dlBarRef.abortBtn) return; let total = 0; let failedResult; const idsGenerators = []; !Array.isArray(chunksGenerators) && (chunksGenerators = [chunksGenerators]); isDownloading = true; changeDlbarDisplay(); try { await Promise.all(chunksGenerators).then((gens) => { gens.forEach((val) => { total += val.total; idsGenerators.push(val.generator); }); }); } catch (error) { logger.error(error); updateStatus('Network error, see console'); changeDlbarDisplay(); isDownloading = false; return; } if (total === 0) { updateStatus('No works'); } else { try { logger.info('Total works:', total); const controller = new AbortController(); const signal = controller.signal; !signal.throwIfAborted && (signal.throwIfAborted = function () { if (this.aborted) { throw this.reason; } }); if (!('reason' in signal)) { const abort = controller.abort; controller.abort = function (reason) { this.signal.reason = reason ? reason : new DOMException('signal is aborted without reason'); abort.apply(this); }; } dlBarRef.abortBtn?.addEventListener('click', () => { controller.abort(); }, { once: true }); const { failed, unavaliable } = await downloadByIds(total, idsGenerators, signal, onProgressCB); if (failed.length || unavaliable.length) { updateStatus(`Failed: ${failed.length + unavaliable.length}. See console.`); console.log('[Pixiv Downloader] Failed: ', failed.join(', ')); console.log('[Pixiv Downloader] Unavaliable: ', unavaliable.join(', ')); if (failed.length) failedResult = failed; } else { console.log('[Pixiv Downloader] Download complete'); updateStatus('Complete'); } } catch (error) { if (error instanceof DOMException) { updateStatus('Stop'); } else { updateStatus('Error, see console'); logger.error(error); } } } changeDlbarDisplay(); isDownloading = false; return failedResult; } function createTagsBtn(userId, category) { const tagsEles = document.querySelectorAll('section> div:nth-child(2) > div > div'); if (!tagsEles.length) return; let cate; if (category === 'illustrations' || category === 'artworks') { cate = 'illusts'; } else { cate = category; } let rest = 'show'; if (userId === getSelfId() && category === 'bookmarks' && location.search.includes('rest=hide')) rest = 'hide'; tagsEles.forEach((ele) => { const tagBtn = ele.querySelector('.pdl-btn'); if (tagBtn) { const btnRest = tagBtn.getAttribute('rest'); if (rest !== btnRest) tagBtn.setAttribute('rest', rest); return; } let tag; const tagLink = ele.querySelector('a'); if (!tagLink) return; if (tagLink.getAttribute('status') !== 'active') { if (rest === 'hide') { tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1, tagLink.href.lastIndexOf('?')); } else { tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1); } } else { const tagTextEles = ele.querySelectorAll('div[title]'); if (!tagTextEles.length) return logger.info('No Tags Element found.'); tag = tagTextEles[tagTextEles.length - 1].getAttribute('title').slice(1); } const attrs = { attrs: { 'pdl-userId': userId, category: cate, tag, rest }, classList: ['pdl-btn', 'pdl-tag'] }; if (isDownloading) attrs.classList.push('pdl-tag-hide'); const dlBtn = createPdlBtn(attrs, '', { addEvent: false }); if (!(tagLink.href.includes('bookmarks') && tagLink.getAttribute('status') !== 'active')) { dlBtn.style.backgroundColor = tagLink.getAttribute('color') + '80'; } dlBtn.addEventListener('click', downloadBookmarksOrTags); ele.appendChild(dlBtn); }); let modalTagsEles; let modal; if (category === 'bookmarks') { modal = document.querySelector('div[role="presentation"]'); if (!modal) return; modalTagsEles = modal.querySelectorAll('a'); } else { const charcoalTokens = document.querySelectorAll('.charcoal-token'); modal = charcoalTokens[charcoalTokens.length - 1]; if (!modal) return; modalTagsEles = modal.querySelectorAll('a'); } if (!regexp.userPageTags.exec(modalTagsEles[0]?.href)) return; modalTagsEles.forEach((ele) => { if (ele.querySelector('.pdl-btn')) return; let tag; if (rest === 'hide') { tag = ele.href.slice(ele.href.lastIndexOf('/') + 1, ele.href.lastIndexOf('?')); } else { tag = ele.href.slice(ele.href.lastIndexOf('/') + 1); } const attrs = { attrs: { 'pdl-userId': userId, category: cate, tag, rest }, classList: ['pdl-btn', 'pdl-modal-tag'] }; if (isDownloading) attrs.classList.push('pdl-tag-hide'); const dlBtn = createPdlBtn(attrs, '', { addEvent: false }); dlBtn.addEventListener('click', (evt) => { modal.querySelector('svg').parentElement.click(); downloadBookmarksOrTags(evt); }); ele.appendChild(dlBtn); }); } function createToolbarBtn(id) { if (document.querySelector('.pdl-btn-main')) return; const handleBar = document.querySelector('main section section'); if (handleBar) { const pdlBtnWrap = handleBar.lastElementChild.cloneNode(); const attrs = { attrs: { 'pdl-id': id }, classList: ['pdl-btn', 'pdl-btn-main'] }; if (pixivHistory.has(id)) attrs.classList.push('pdl-complete'); pdlBtnWrap.appendChild(createPdlBtn(attrs)); handleBar.appendChild(pdlBtnWrap); } } function createWorkScrollBtn(id) { const works = document.querySelectorAll("[role='presentation'] > a"); if (works.length < 2) return; const containers = Array.from(works).map((node) => node.parentElement.parentElement); if (containers[0].querySelector('.pdl-btn')) return; containers.forEach((node, idx) => { const wrapper = document.createElement('div'); wrapper.classList.add('pdl-wrap-artworks'); const attrs = { attrs: { 'pdl-id': id, 'should-download': String(idx) }, classList: ['pdl-btn', 'pdl-btn-sub', 'artworks'] }; wrapper.appendChild(createPdlBtn(attrs)); node.appendChild(wrapper); }); } const createPresentationBtn = (() => { let observer, btn; function cb(mutationList) { const newImg = mutationList[1]['addedNodes'][0]; const [pageNum] = regexp.originSrcPageNum.exec(newImg.src) ?? []; if (!pageNum) throw new Error('[Error]Invalid Image Element.'); btn?.setAttribute('should-download', String(pageNum)); } return (id) => { const containers = document.querySelector("body > [role='presentation'] > div"); if (!containers) { if (observer) { observer.disconnect(); observer = null; btn = null; } return; } if (containers.querySelector('.pdl-btn')) return; const img = containers.querySelector('img'); if (!img) return; const isOriginImg = regexp.originSrcPageNum.exec(img.src); if (!isOriginImg) return; const [pageNum] = isOriginImg; const attrs = { attrs: { 'pdl-id': id, 'should-download': pageNum }, classList: ['pdl-btn', 'pdl-btn-sub', 'presentation'] }; btn = createPdlBtn(attrs); containers.appendChild(btn); if (!img.parentElement) return; observer = new MutationObserver(cb); observer.observe(img.parentElement, { childList: true, subtree: true }); }; })(); function createPreviewModalBtn() { const illustModalBtn = document.querySelectorAll('.gtm-manga-viewer-preview-modal-open'); const mangaModalBtn = document.querySelectorAll('.gtm-manga-viewer-open-preview'); const mangaViewerModalBtn = document.querySelectorAll('.gtm-manga-viewer-close-icon')?.[1]; if (!illustModalBtn.length && !mangaModalBtn.length) return; const btns = [...illustModalBtn, ...mangaModalBtn]; if (mangaViewerModalBtn) btns.push(mangaViewerModalBtn); btns.forEach((node) => { node.addEventListener('click', handleModalClick); }); } function handleModalClick() { const timer = setInterval(() => { logger.info('Start to find modal.'); const ulList = document.querySelectorAll('ul'); const previewList = ulList[ulList.length - 1]; if (getComputedStyle(previewList).display !== 'grid') return; clearInterval(timer); const [, id] = regexp.artworksPage.exec(location.pathname) ?? []; previewList.childNodes.forEach((node, idx) => { node.style.position = 'relative'; const attrs = { attrs: { 'pdl-id': id, 'should-download': String(idx) }, classList: ['pdl-btn', 'pdl-btn-sub'] }; node.appendChild(createPdlBtn(attrs)); }); }, 300); } let firstRun = true; function observerCallback(records) { const addedNodes = []; records.forEach((record) => { if (!record.addedNodes.length) return; record.addedNodes.forEach((node) => { if (node.nodeType === Node.ELEMENT_NODE && node.tagName !== 'BUTTON' && node.tagName !== 'IMG') { addedNodes.push(node); } }); }); if (!addedNodes.length) return; if (firstRun) { createThumbnailsBtn(document.querySelectorAll('a')); firstRun = false; } else { fixPixivPreviewer(addedNodes); const thunmnails = addedNodes.reduce((prev, current) => { return prev.concat(Array.from(current.querySelectorAll('a'))); }, []); createThumbnailsBtn(thunmnails); } 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]; createToolbarBtn(id); createWorkScrollBtn(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(args, option = { closeOnClickModal: false }) { const modalHtml = ` <div class="pdl-modal"> <div class="pdl-dialog"> <button class="pdl-dialog-close"></button> <header class="pdl-dialog-header"></header> <div class="pdl-dialog-content"></div> <footer class="pdl-dialog-footer"></footer> </div> </div>`; const fragment = stringToFragment(modalHtml); const modal = fragment.querySelector('.pdl-modal'); const dialog = modal.querySelector('.pdl-dialog'); const closeBtn = modal.querySelector('.pdl-dialog-close'); const header = modal.querySelector('.pdl-dialog-header'); const content = modal.querySelector('.pdl-dialog-content'); const footer = modal.querySelector('.pdl-dialog-footer'); if (option.closeOnClickModal) { dialog.addEventListener('click', (e) => { e.stopPropagation(); }); modal.addEventListener('click', () => { modal.remove(); }); } closeBtn.addEventListener('click', () => { modal.remove(); }); args.header && header.appendChild(args.header); args.footer && footer.appendChild(args.footer); content.appendChild(args.content); return fragment; } function showUpgradeMsg() { const headerHtml = `<h3>Pixiv Downloader ${config.get('version')}</h3>`; const contentHtml = ` <div class="pdl-changelog"> <ul> <li> <p>文件名现在可以使用 {date} 显示图片发布时间。</p> <p>默认格式为YYYY-MM-DD,你也可以自定义格式,如: </p> <p>{date(MMM D, YYYY)}, {date(h:mm A)}, 更多可用占位符<a target="_blank" style="color: #0096fa; text-decoration: underline" href="https://dayjs.gitee.io/docs/zh-CN/display/format">见此</a>。</p> </li> <li>修复了部分低版本浏览器无法批量下载的问题。</li> </ul> </div>`; const footerHtml = ` <style> .pdl-dialog-footer { position: relative; font-size: 12px; } </style> <details style="margin-top: 1.5em"> <summary style="display: inline-block; list-style: none; cursor: pointer; color: #0096fa; text-decoration: underline"> 脚本还行?请我喝杯可乐吧! </summary> ${creditCode} <p style="text-align: center">愿你每天都能找到对的色图,就像我每天都能喝到香草味可乐</p> </details> <a target="_blank" style="position: absolute; right: 0px; top: 0px; color: #0096fa; text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback" >${t('modals.upgradeMsg.feedback')} </a>`; document.body.appendChild(createModal({ header: stringToFragment(headerHtml), content: stringToFragment(contentHtml), footer: stringToFragment(footerHtml) })); } function createTabUgoria() { const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.ugoria.tab_title')}</div>`; const paneHtml = ` <div class="pdl-tab-pane"> <div id="pdl-setting-ugoria"> <p class="option-header">${t('modals.setting.ugoria.format_label')}</p> <div id="pdl-ugoria-format-wrap"> <div class="pdl-ugoria-format-item"> <input type="radio" id="pdl-ugoria-zip" value="zip" name="format" /><label for="pdl-ugoria-zip">Zip</label> </div> <div class="pdl-ugoria-format-item"> <input type="radio" id="pdl-ugoria-gif" value="gif" name="format" /><label for="pdl-ugoria-gif">Gif</label> </div> <div class="pdl-ugoria-format-item"> <input type="radio" id="pdl-ugoria-apng" value="png" name="format" /><label for="pdl-ugoria-apng">Png</label> </div> <div class="pdl-ugoria-format-item"> <input type="radio" id="pdl-ugoria-webm" value="webm" name="format" /><label for="pdl-ugoria-webm">Webm</label> </div> <div class="pdl-ugoria-format-item"> <input type="radio" id="pdl-ugoria-webp" value="webp" name="format" /><label for="pdl-ugoria-webp">Webp</label> </div> </div> </div> </div>`; const tab = stringToFragment(tabHtml); const pane = stringToFragment(paneHtml); const ugoriaFormat = config.get('ugoriaFormat'); pane.querySelectorAll('.pdl-ugoria-format-item input[type="radio"]').forEach((el) => { if (ugoriaFormat === el.value) el.checked = true; el.addEventListener('change', (ev) => { config.set('ugoriaFormat', ev.currentTarget.value); }); }); return { tab, pane }; } function createTabFilename() { const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.filename.tab_title')}</div>`; const paneHtml = ` <div class="pdl-tab-pane"> <div id="pdl-setting-filename"> <div> <div class="pdl-input-wrap"> <label for="pdlfolder">${t('modals.setting.filename.input.folder_label')}</label> <input type="text" id="pdlfolder" maxlength="100" /> <button id="pdl-filename-folder-reset" class="pdl-dialog-button icon" disabled>↺</button> <button id="pdl-filename-folder-confirm" class="pdl-dialog-button icon primary" disabled>✓</button> </div> <div class="pdl-input-wrap"> <label for="pdlfilename">${t('modals.setting.filename.input.filename_label')}</label> <input type="text" id="pdlfilename" placeholder="${t('modals.setting.filename.input.folder_placeholder')}" required maxlength="100" /> <button id="pdl-filename-filename-reset" class="pdl-dialog-button icon" disabled>↺</button> <button id="pdl-filename-filename-confirm" class="pdl-dialog-button icon primary" disabled>✓</button> </div> </div> <div class="tags-option"> <span class="tags-title">${t('modals.setting.filename.input.tag_label')}</span> <div class="tags-content"> <div class="tags-item"> <input class="pdl-option-tag" type="radio" name="lang" id="lang_ja" value="ja" /> <label for="lang_ja">日本語(default)</label> </div> <div class="tags-item"> <input class="pdl-option-tag" type="radio" name="lang" id="lang_zh" value="zh" /> <label for="lang_zh">简中</label> </div> <div class="tags-item"> <input class="pdl-option-tag" type="radio" name="lang" id="lang_zh_tw" value="zh_tw" /> <label for="lang_zh_tw">繁中</label> </div> <div class="tags-item"> <input class="pdl-option-tag" type="radio" name="lang" id="lang_en" value="en" /> <label for="lang_en">English</label> </div> </div> </div> <p style="font-size: 14px; margin: 0.5em 0">${t('modals.setting.filename.tips.filename_pattern')}</p> <p style="font-size: 14px; margin: 0.5em 0">${t('modals.setting.filename.tips.empty_folder')}</p> <p style="font-size: 14px; margin: 0.5em 0">${t('modals.setting.filename.tips.tag_translation')}</p> <hr /> <div ${env.isFileSystemAccessAvaliable() ? '' : 'class="pdl-unavailable"'}> <div style="display: flex; justify-content: space-between; align-items: center; margin: 12px 0; gap: 12px"> <label class="pdl-options" style="padding: 0.6em 4px"> <span style="font-weight: 700">${t('modals.setting.filename.input.fsa_label')}</span> <input id="pdl-options-file-system-access" type="checkbox" class="pdl-checkbox" style="margin-left: 8px" /> </label> <hr class="vertical" /> <div class="pdl-input-wrap" style="flex: 1; margin: 0"> <input id="pdl-fsa-show-directory" type="text" placeholder="${t('modals.setting.filename.input.fsa_placeholder')}" style="font-size: 14px; padding: 8px 0.5em; line-height: 1.15" disabled/> </div> <button id="pdl-fsa-change-directory" class="pdl-dialog-button primary">${t('modals.setting.filename.button.fsa_change_dir')}</button> </div> <div class="tags-option"> <span class="tags-title">${t('modals.setting.filename.input.filename_conflict_label')}</span> <div class="tags-content"> <div class="tags-item"> <input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_uniquify" value="uniquify"/> <label for="action_uniquify">${t('modals.setting.filename.input.filename_conflict_option_uniquify')}</label> </div> <div class="tags-item"> <input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_overwrite" value="overwrite"/> <label for="action_overwrite">${t('modals.setting.filename.input.filename_conflict_option_overwrite')}</label> </div> <div class="tags-item"> <input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_prompt" value="prompt"/> <label for="action_prompt">${t('modals.setting.filename.input.filename_conflict_option_prompt')}</label> </div> </div> </div> </div> </div> </div>`; const tab = stringToFragment(tabHtml); const pane = stringToFragment(paneHtml); const folder = pane.querySelector('#pdlfolder'); const folderReset = pane.querySelector('#pdl-filename-folder-reset'); const folderUpdate = pane.querySelector('#pdl-filename-folder-confirm'); const filename = pane.querySelector('#pdlfilename'); const filenameReset = pane.querySelector('#pdl-filename-filename-reset'); const filenameUpdate = pane.querySelector('#pdl-filename-filename-confirm'); const filenamePattern = config.get('filenamePattern'); const folderPattern = config.get('folderPattern'); if (!folder || !filename) throw new Error('[Error]Can not create modal.'); filename.value = filenamePattern; if (!env.isSupportSubpath()) { folder.setAttribute('disabled', ''); folder.value = ''; } else { folder.value = folderPattern; } folder.placeholder = env.isViolentmonkey() ? t('modals.setting.filename.input.folder_placeholder_vm') : !env.isSupportSubpath() ? t('modals.setting.filename.input.folder_placeholder_need_api') : t('modals.setting.filename.input.folder_placeholder'); folder.addEventListener('input', () => { folderReset?.removeAttribute('disabled'); folderUpdate?.removeAttribute('disabled'); }); folderReset?.addEventListener('click', () => { folder.value = config.get('folderPattern'); folderReset?.setAttribute('disabled', ''); folderUpdate?.setAttribute('disabled', ''); }); folderUpdate?.addEventListener('click', () => { const folderPattern = folder.value .split('/') .map(replaceInvalidChar) .filter((path) => !!path) .join('/'); config.set('folderPattern', folderPattern); folder.value = folderPattern; folderReset?.setAttribute('disabled', ''); folderUpdate?.setAttribute('disabled', ''); }); filename.addEventListener('input', () => { filenameReset?.removeAttribute('disabled'); filenameUpdate?.removeAttribute('disabled'); }); filenameReset?.addEventListener('click', () => { filename.value = config.get('filenamePattern'); filenameReset?.setAttribute('disabled', ''); filenameUpdate?.setAttribute('disabled', ''); }); filenameUpdate?.addEventListener('click', () => { const filenamePattern = replaceInvalidChar(filename.value); if (filenamePattern === '') return filenameReset?.click(); config.set('filenamePattern', filenamePattern); filename.value = filenamePattern; filenameReset?.setAttribute('disabled', ''); filenameUpdate?.setAttribute('disabled', ''); }); const tagLang = config.get('tagLang'); pane.querySelectorAll('.tags-content .tags-item input.pdl-option-tag').forEach((input) => { if (tagLang === input.value) input.checked = true; input.addEventListener('change', (ev) => { config.set('tagLang', ev.currentTarget.value); }); }); if (env.isFileSystemAccessAvaliable()) { const enableFSA = pane.querySelector('#pdl-options-file-system-access'); const showDir = pane.querySelector('#pdl-fsa-show-directory'); const changeDirBtn = pane.querySelector('#pdl-fsa-change-directory'); const actionInput = pane.querySelectorAll('.tags-content .tags-item input.pdl-option-conflict'); const interactElems = [changeDirBtn, ...actionInput]; const isUseFSA = config.get('useFileSystemAccess'); const conflictAction = config.get('fileSystemFilenameConflictAction'); if (isUseFSA) { folder.placeholder = t('modals.setting.filename.input.folder_placeholder'); folder.removeAttribute('disabled'); folder.value = folderPattern; } enableFSA.checked = isUseFSA; if (!isUseFSA) { interactElems.forEach((el) => el.setAttribute('disabled', '')); } enableFSA.addEventListener('change', (ev) => { const isEnabled = ev.target.checked; config.set('useFileSystemAccess', isEnabled); if (isEnabled) { folder.placeholder = t('modals.setting.filename.input.folder_placeholder'); if (folder.hasAttribute('disabled')) { folder.removeAttribute('disabled'); folder.value = config.get('folderPattern'); } interactElems.forEach((el) => el.removeAttribute('disabled')); } else { if (env.isViolentmonkey()) { folder.placeholder = t('modals.setting.filename.input.folder_placeholder_vm'); folder.setAttribute('disabled', ''); folder.value = ''; } else if (!env.isSupportSubpath()) { folder.placeholder = t('modals.setting.filename.input.folder_placeholder_need_api'); folder.setAttribute('disabled', ''); folder.value = ''; } interactElems.forEach((el) => el.setAttribute('disabled', '')); } }); showDir.value = getCurrentDirName(); changeDirBtn.addEventListener('click', async () => { await updateDirHandle(); showDir.value = getCurrentDirName(); }); actionInput.forEach((input) => { if (conflictAction === input.value) input.checked = true; input.addEventListener('change', (ev) => { config.set('fileSystemFilenameConflictAction', ev.currentTarget.value); }); }); } return { tab, pane }; } function createTabAdjustButtonPosition() { const style = getComputedStyle(document.documentElement); const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.button_pos.tab_title')}</div>`; const paneHtml = ` <div class="pdl-tab-pane"> <div class="pdl-adjust-button"> <div class="pdl-adjust-content"> <datalist id="pdl-adjust-tickmarks"> <option value="0"></option> <option value="25"></option> <option value="50"></option> <option value="75"></option> <option value="100"></option> </datalist> <div class="pdl-adjust-item"> <p class="pdl-adjust-title">${t('modals.setting.button_pos.self_bookmark_title')}</p> <div class="pdl-adjust-select"> <span>${t('modals.setting.button_pos.input.horizon_label')}</span ><input id="pdl-btn-self-bookmark-left" type="range" max="100" min="0" step="1" list="pdl-adjust-tickmarks" value="${style.getPropertyValue('--pdl-btn-self-bookmark-left')}" /> </div> <div class="pdl-adjust-select"> <span>${t('modals.setting.button_pos.input.vertical_label')}</span ><input id="pdl-btn-self-bookmark-top" type="range" max="100" min="0" step="1" list="pdl-adjust-tickmarks" value="${style.getPropertyValue('--pdl-btn-self-bookmark-top')}" /> </div> </div> <div class="pdl-adjust-item"> <p class="pdl-adjust-title">${t('modals.setting.button_pos.preview_title')}</p> <div class="pdl-adjust-select"> <span>${t('modals.setting.button_pos.input.horizon_label')}</span ><input id="pdl-btn-left" type="range" max="100" min="0" step="1" list="pdl-adjust-tickmarks" value="${style.getPropertyValue('--pdl-btn-left')}" /> </div> <div class="pdl-adjust-select"> <span>${t('modals.setting.button_pos.input.vertical_label')}</span ><input id="pdl-btn-top" type="range" max="100" min="0" step="1" list="pdl-adjust-tickmarks" value="${style.getPropertyValue('--pdl-btn-top')}" /> </div> </div> </div> <div class="pdl-adjust-preview"> <div class="pdl-thumbnail-sample"> <button class="pdl-btn pdl-btn-sub"></button> <button class="pdl-btn pdl-btn-sub self-bookmark pdl-complete"></button> </div> </div> </div> </div>`; const tab = stringToFragment(tabHtml); const pane = stringToFragment(paneHtml); pane.querySelectorAll('.pdl-adjust-select input[type="range"]').forEach((el) => { el.addEventListener('input', (ev) => { const el = ev.target; const val = el.value; document.documentElement.style.setProperty('--' + el.id, val); }); el.addEventListener('change', (ev) => { const el = ev.target; const key = el.id; config.set(key, el.value); }); }); return { tab, pane }; } function createTabHistory() { const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.history.tab_title')}</div>`; const paneHtml = ` <div class="pdl-tab-pane"> <div id="pdl-setting-history"> <div> <button id="pdl-export" class="btn-history pdl-dialog-button primary"> ${t('modals.setting.history.button.export')} </button> </div> <div> <input type="file" id="pdl-import" accept=".txt" style="display: none" /> <button id="pdl-import-btn" class="btn-history pdl-dialog-button primary"> ${t('modals.setting.history.button.import')} </button> </div> <div> <input type="file" id="pdl-merge" accept=".txt" style="display: none" /> <button id="pdl-merge-btn" class="btn-history pdl-dialog-button primary"> ${t('modals.setting.history.button.merge')} </button> </div> <div> <button id="pdl-clear-history" class="btn-history pdl-dialog-button primary"> ${t('modals.setting.history.button.clear')} </button> </div> </div> </div>`; const tab = stringToFragment(tabHtml); const pane = stringToFragment(paneHtml); const importFile = pane.querySelector('#pdl-import'); importFile?.addEventListener('change', (evt) => { const file = evt.currentTarget.files?.[0]; if (!file) return; pixivHistory.replace(file); }); const mergeFile = pane.querySelector('#pdl-merge'); mergeFile?.addEventListener('change', (evt) => { const file = evt.currentTarget.files?.[0]; if (!file) return; pixivHistory.merge(file); }); const importBtn = pane.querySelector('#pdl-import-btn'); importBtn?.addEventListener('click', () => importFile?.click()); const mergeBtn = pane.querySelector('#pdl-merge-btn'); mergeBtn?.addEventListener('click', () => mergeFile?.click()); const exportBtn = pane.querySelector('#pdl-export'); exportBtn?.addEventListener('click', () => { const dlEle = document.createElement('a'); const history = JSON.stringify(pixivHistory.getAll()); dlEle.href = URL.createObjectURL(new Blob([history], { type: 'text/plain' })); dlEle.download = 'Pixiv Downloader ' + new Date().toLocaleString() + '.txt'; dlEle.click(); URL.revokeObjectURL(dlEle.href); }); const clearBtn = pane.querySelector('#pdl-clear-history'); clearBtn?.addEventListener('click', () => pixivHistory.clearHistory()); return { tab, pane }; } var css = ".pdl-popup-button {\n position: fixed;\n z-index: 1;\n right: 28px;\n bottom: 100px;\n padding: 12px;\n border-radius: 50%;\n border: none;\n cursor: pointer;\n color: rgb(255, 255, 255);\n background-color: rgba(0, 150, 250, 0.5);\n transition: opacity 0.3s ease 0s;\n opacity: 0.32;\n line-height: 0;\n margin: 0;\n}\n\n.pdl-popup-button:hover {\n opacity: 1;\n}\n\n.pdl-popup-button svg {\n width: 24px;\n height: 24px;\n fill: currentColor;\n}\n"; n(css,{}); function showPopupBtn() { if (document.querySelector('.pdl-popup-button')) return; const btn = document.createElement('button'); btn.className = 'pdl-popup-button'; btn.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12,15.5A3.5,3.5 0 0,1 8.5,12A3.5,3.5 0 0,1 12,8.5A3.5,3.5 0 0,1 15.5,12A3.5,3.5 0 0,1 12,15.5M19.43,12.97C19.47,12.65 19.5,12.33 19.5,12C19.5,11.67 19.47,11.34 19.43,11L21.54,9.37C21.73,9.22 21.78,8.95 21.66,8.73L19.66,5.27C19.54,5.05 19.27,4.96 19.05,5.05L16.56,6.05C16.04,5.66 15.5,5.32 14.87,5.07L14.5,2.42C14.46,2.18 14.25,2 14,2H10C9.75,2 9.54,2.18 9.5,2.42L9.13,5.07C8.5,5.32 7.96,5.66 7.44,6.05L4.95,5.05C4.73,4.96 4.46,5.05 4.34,5.27L2.34,8.73C2.21,8.95 2.27,9.22 2.46,9.37L4.57,11C4.53,11.34 4.5,11.67 4.5,12C4.5,12.33 4.53,12.65 4.57,12.97L2.46,14.63C2.27,14.78 2.21,15.05 2.34,15.27L4.34,18.73C4.46,18.95 4.73,19.03 4.95,18.95L7.44,17.94C7.96,18.34 8.5,18.68 9.13,18.93L9.5,21.58C9.54,21.82 9.75,22 10,22H14C14.25,22 14.46,21.82 14.5,21.58L14.87,18.93C15.5,18.67 16.04,18.34 16.56,17.94L19.05,18.95C19.27,19.03 19.54,18.95 19.66,18.73L21.66,15.27C21.78,15.05 21.73,14.78 21.54,14.63L19.43,12.97Z" /></svg>`; btn.addEventListener('click', () => { showSettings(); }); document.body.appendChild(btn); } function removePopupBtn() { document.querySelector('button.pdl-popup-button')?.remove(); } function createTabOthers() { const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.others.tab_title')}</div>`; const paneHtml = ` <div class="pdl-tab-pane"> <div id="pdl-setting-others"> <div> <label class="pdl-options"> <input id="pdl-options-bundle-illusts" type="checkbox" class="pdl-checkbox" /> <span>${t('modals.setting.others.input.bundle_illusts')}</span> </label> </div> <hr /> <div> <label class="pdl-options"> <input id="pdl-options-bundle-manga" type="checkbox" class="pdl-checkbox" /> <span>${t('modals.setting.others.input.bundle_manga')}</span> </label> </div> <hr /> <div> <label class="pdl-options"> <input id="pdl-options-add-bookmark" type="checkbox" class="pdl-checkbox" /> <span>${t('modals.setting.others.input.add_bookmark')}</span> </label> </div> <hr /> <div> <label class="pdl-options sub-option"> <input id="pdl-options-add-bookmark-tags" type="checkbox" class="pdl-checkbox" /> <span>${t('modals.setting.others.input.add_bookmark_with_tags')}</span> </label> </div> <hr class="sub" /> <div> <label class="pdl-options sub-option"> <input id="pdl-options-add-bookmark-private-r18" type="checkbox" class="pdl-checkbox" /> <span>${t('modals.setting.others.input.add_bookmark_private_r18')}</span> </label> </div> <hr /> <div> <label class="pdl-options"> <input id="pdl-options-show-popup-button" type="checkbox" class="pdl-checkbox" /> <span>${t('modals.setting.others.input.show_popup_button')}</span> </label> </div> </div> </div>`; const tab = stringToFragment(tabHtml); const pane = stringToFragment(paneHtml); [ { selector: '#pdl-options-bundle-illusts', settingKey: 'bundleIllusts' }, { selector: '#pdl-options-bundle-manga', settingKey: 'bundleManga' }, { selector: '#pdl-options-add-bookmark', settingKey: 'addBookmark' }, { selector: '#pdl-options-add-bookmark-tags', settingKey: 'addBookmarkWithTags' }, { selector: '#pdl-options-add-bookmark-private-r18', settingKey: 'privateR18' }, { selector: '#pdl-options-show-popup-button', settingKey: 'showPopupButton' } ].forEach(({ selector, settingKey }) => { const optionEl = pane.querySelector(selector); if (!optionEl) return; optionEl.checked = config.get(settingKey); optionEl.addEventListener('change', (ev) => { config.set(settingKey, ev.currentTarget.checked); }); }); pane.querySelector('#pdl-options-show-popup-button').addEventListener('change', (ev) => { if (ev.currentTarget.checked) { showPopupBtn(); } else { removePopupBtn(); } }); return { tab, pane }; } function createTabFeedback() { const tabHtml = `<div class="pdl-tab-item">${t('modals.setting.feedback.tab_title')}</div>`; const paneHtml = ` <div class="pdl-tab-pane"> <div id="pdl-setting-donate"> ${creditCode} <p>如果脚本有帮助到你,欢迎扫码请我喝杯可乐 ^_^</p> <p> <a target="_blank" style="color: #0096fa; text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback" >${t('modals.setting.feedback.tips.feedback')}</a > </p> </div> </div>`; return { tab: stringToFragment(tabHtml), pane: stringToFragment(paneHtml) }; } function showSettings() { if (document.querySelector('.pdl-modal')) return; const contentHtml = ` <div> <div class="pdl-tabs-nav"> <div class="pdl-tabs__active-bar"></div> </div> <div class="pdl-tabs-content"></div> </div>`; const modal = createModal({ content: stringToFragment(contentHtml) }); const tabsNav = modal.querySelector('.pdl-tabs-nav'); const tabContent = modal.querySelector('.pdl-tabs-content'); [ createTabFilename(), createTabUgoria(), createTabHistory(), createTabAdjustButtonPosition(), createTabOthers(), createTabFeedback() ].forEach(({ tab, pane }) => { tabsNav.appendChild(tab); tabContent.appendChild(pane); }); const panes = Array.from(tabContent.querySelectorAll('.pdl-tab-pane')); panes.forEach((el) => { el.style.setProperty('display', 'none'); }); const activeBar = tabsNav.querySelector('.pdl-tabs__active-bar'); const tabs = Array.from(modal.querySelectorAll('.pdl-tabs-nav .pdl-tab-item')); tabs.forEach((el) => { el.addEventListener('click', (ev) => { const tab = ev.currentTarget; if (!tab) return; tabs.forEach((tab) => tab.classList.remove('active')); tab.classList.add('active'); activeBar.style.width = getComputedStyle(tab).width; activeBar.style.transform = `translateX(${tab.offsetLeft + parseFloat(getComputedStyle(tab).paddingLeft)}px)`; panes.forEach((pane) => pane.style.setProperty('display', 'none')); panes[tabs.indexOf(tab)].style.removeProperty('display'); }); }); tabs[0].classList.add('active'); panes[0].style.removeProperty('display'); document.body.appendChild(modal); activeBar.style.width = getComputedStyle(tabs[0]).width; activeBar.style.transform = `translateX(${tabs[0].offsetLeft + parseFloat(getComputedStyle(tabs[0]).paddingLeft)}px)`; } pixivHistory.updateHistory(); GM_registerMenuCommand(t('gm_menu.setting'), showSettings, 's'); if (config.get('showMsg')) { showUpgradeMsg(); config.set('showMsg', false); } if (config.get('showPopupButton')) { showPopupBtn(); } ['pdl-btn-self-bookmark-left', 'pdl-btn-self-bookmark-top', 'pdl-btn-left', 'pdl-btn-top'].forEach((key) => { let val; if ((val = config.get(key)) !== undefined) { document.documentElement.style.setProperty('--' + key, val); } }); new MutationObserver(observerCallback).observe(document.body, { childList: true, subtree: true }); document.addEventListener('keydown', (e) => { if (e.ctrlKey && e.key === 'q') { const pdlMainBtn = document.querySelector('.pdl-btn-main'); if (pdlMainBtn) { e.preventDefault(); if (!e.repeat) { pdlMainBtn.dispatchEvent(new MouseEvent('click')); } } } }); })(workerChunk, GIF, JSZip, dayjs);