- // ==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);