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