Pixiv Downloader

一键下载Pixiv各页面原图。支持多图下载,动图下载,按作品标签下载,画师作品批量下载。动图支持格式转换:Gif | Apng | Webp | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。

Versión del día 29/05/2023. Echa un vistazo a la versión más reciente.

// ==UserScript==
// @name         Pixiv Downloader
// @namespace    https://greasyfork.org/zh-CN/scripts/432150
// @version      0.8.8
// @description:en  Download the original images of Pixiv pages with one click. Supports:multiple illustrations, ugoira(animation), and batch downloads of artists' work. Ugoira support format conversion: Gif | Apng | Webp | Webm. The downloaded images will be saved in a separate folder named after the artist (you need to adjust the tampermonkey "Download" setting to "Browser API"). A record of downloaded images is kept.
// @description  一键下载Pixiv各页面原图。支持多图下载,动图下载,按作品标签下载,画师作品批量下载。动图支持格式转换:Gif | Apng | Webp | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。
// @description:zh-TW  一鍵下載Pixiv各頁面原圖。支持多圖下載,動圖下載,按作品標籤下載,畫師作品批次下載。動圖支持格式轉換:Gif | Apng | Webp | Webm。下載的圖片將保存到以畫師名命名的單獨文件夾(需要調整tampermonkey“下載”設置為“瀏覽器API”)。保留已下載圖片的紀錄。
// @author       ruaruarua
// @match        https://www.pixiv.net/*
// @icon         https://www.pixiv.net/favicon.ico
// @noframes
// @grant        GM_xmlhttpRequest
// @grant        GM_download
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_info
// @grant        GM_registerMenuCommand
// @connect      i.pximg.net
// @require      https://cdnjs.cloudflare.com/ajax/libs/jszip/3.7.1/jszip.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/gif.js/0.2.0/gif.js
// @require      https://greasyfork.org/scripts/455256-toanimatedwebp/code/toAnimatedWebp.js?version=1120088
// ==/UserScript==
(function (workerChunk, JSZip, GIF) {
  'use strict';

  function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }

  var workerChunk__default = /*#__PURE__*/_interopDefaultLegacy(workerChunk);
  var JSZip__default = /*#__PURE__*/_interopDefaultLegacy(JSZip);
  var GIF__default = /*#__PURE__*/_interopDefaultLegacy(GIF);

  var e=[],t=[];function n(n,r){if(n&&"undefined"!=typeof document){var a,s=!0===r.prepend?"prepend":"append",d=!0===r.singleTag,i="string"==typeof r.container?document.querySelector(r.container):document.getElementsByTagName("head")[0];if(d){var u=e.indexOf(i);-1===u&&(u=e.push(i)-1,t[u]={}),a=t[u]&&t[u][s]?t[u][s]:t[u][s]=c();}else a=c();65279===n.charCodeAt(0)&&(n=n.substring(1)),a.styleSheet?a.styleSheet.cssText+=n:a.appendChild(document.createTextNode(n));}function c(){var e=document.createElement("style");if(e.setAttribute("type","text/css"),r.attributes)for(var t=Object.keys(r.attributes),n=0;n<t.length;n++)e.setAttribute(t[n],r.attributes[t[n]]);var a="prepend"===s?"afterbegin":"beforeend";return i.insertAdjacentElement(a,e),e}}

  var css$3 = "@property --pdl-progress {\n  syntax: '<percentage>';\n  inherits: true;\n  initial-value: 0%;\n}\n@keyframes pdl_loading {\n  100% {\n    transform: translate(-50%, -50%) rotate(360deg);\n  }\n}\n.pdl-btn {\n  position: relative;\n  border-top-right-radius: 8px;\n  background: no-repeat center/85%;\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%233C3C3C' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n  color: #01b468;\n  display: inline-block;\n  font-size: 13px;\n  font-weight: bold;\n  height: 32px;\n  line-height: 32px;\n  margin: 0;\n  overflow: hidden;\n  padding: 0;\n  border: none;\n  text-decoration: none !important;\n  text-align: center;\n  text-overflow: ellipsis;\n  user-select: none;\n  white-space: nowrap;\n  width: 32px;\n  z-index: 1;\n  cursor: pointer;\n}\n.pdl-btn-main {\n  margin: 0 0 0 10px;\n}\n.pdl-btn-sub {\n  bottom: 0;\n  background-color: rgba(255, 255, 255, 0.5);\n  left: 0;\n  position: absolute;\n}\n.pdl-btn-sub.artworks {\n  position: sticky;\n  top: 40px;\n  border-radius: 4px;\n}\n.pdl-btn-sub.presentation {\n  position: fixed;\n  top: 50px;\n  right: 16px;\n  border-radius: 8px;\n  left: auto;\n}\n.pdl-btn-sub-bookmark.pdl-btn-sub-bookmark {\n  left: auto;\n  right: 0;\n  bottom: 34px;\n  border-radius: 8px;\n  border-top-right-radius: 0px;\n  border-bottom-right-radius: 0px;\n}\n.pdl-error {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23EA0000' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\") !important;\n}\n.pdl-complete {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%2301B468' d='M256 8C119.033 8 8 119.033 8 256s111.033 248 248 248 248-111.033 248-248S392.967 8 256 8zm0 48c110.532 0 200 89.451 200 200 0 110.532-89.451 200-200 200-110.532 0-200-89.451-200-200 0-110.532 89.451-200 200-200m140.204 130.267l-22.536-22.718c-4.667-4.705-12.265-4.736-16.97-.068L215.346 303.697l-59.792-60.277c-4.667-4.705-12.265-4.736-16.97-.069l-22.719 22.536c-4.705 4.667-4.736 12.265-.068 16.971l90.781 91.516c4.667 4.705 12.265 4.736 16.97.068l172.589-171.204c4.704-4.668 4.734-12.266.067-16.971z'%3E%3C/path%3E %3C/svg%3E\") !important;\n}\n.pdl-progress {\n  background-image: none !important;\n  cursor: default !important;\n}\n.pdl-progress:after {\n  content: '';\n  display: inline-block;\n  position: absolute;\n  top: 50%;\n  left: 50%;\n  width: 27px;\n  height: 27px;\n  transform: translate(-50%, -50%);\n  -webkit-mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);\n  mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);\n  border-radius: 50%;\n}\n.pdl-progress:not(:empty):after {\n  background: conic-gradient(\n    #01b468 0,\n    #01b468 var(--pdl-progress),\n    transparent var(--pdl-progress),\n    transparent\n  );\n  transition: --pdl-progress 0.2s ease;\n}\n.pdl-progress:empty:after {\n  background: conic-gradient(#01b468 0, #01b468 25%, #01b46833 25%, #01b46833);\n  animation: 1.5s infinite linear pdl_loading;\n}\n\n.pdl-btn.pdl-tag {\n  height: auto;\n  border-top-right-radius: 4px;\n  border-bottom-right-radius: 4px;\n  left: -1px;\n  background-color: rgba(0, 0, 0, 0.12);\n  transition: background-image 0.5s;\n}\n\n.pdl-btn.pdl-tag.pdl-tag-hide,\n.pdl-btn.pdl-modal-tag.pdl-tag-hide {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E  %3C/svg%3E\");\n  pointer-events: none;\n}\n\n.pdl-btn.pdl-modal-tag {\n  position: absolute;\n  right: 65px;\n  top: 6px;\n  background-origin: content-box;\n  border-radius: 4px;\n  padding: 5px;\n  width: 42px;\n  height: 50px;\n  background-color: rgba(0, 0, 0, 0.04);\n  transition: 0.25s background-color;\n}\n\n.pdl-btn.pdl-modal-tag:not(.pdl-tag-hide):hover {\n  background-color: rgba(0, 0, 0, 0.12);\n}\n\n.pdl-wrap-artworks {\n  position: absolute;\n  right: 8px;\n  top: 0px;\n  bottom: 0px;\n  margin-top: 40px;\n}\n";
  n(css$3,{});

  var css$2 = ".pdl-modal * {\n  font-family: 'win-bug-omega, system-ui, -apple-system, \"Segoe UI\", Roboto, Ubuntu, Cantarell, \"Noto Sans\", \"Hiragino Kaku Gothic ProN\", Meiryo, sans-serif';\n  line-height: 1.15;\n}\n\n.pdl-modal {\n  position: fixed;\n  top: 0;\n  left: 0;\n  width: 100%;\n  height: 100%;\n  display: flex;\n  z-index: 99;\n  background-color: rgba(0, 0, 0, 0.32);\n  user-select: none;\n}\n\n.pdl-dialog {\n  position: relative;\n  background-color: #fff;\n  border-radius: 24px;\n  margin: auto;\n  padding: 20px 40px 30px 40px;\n  width: 600px;\n  font-size: 16px;\n}\n\n.pdl-dialog-header > h3 {\n  font-weight: bold;\n  font-size: 1.17em;\n  margin: 1em 0;\n}\n\n.pdl-dialog-close {\n  position: absolute;\n  top: 10px;\n  right: 10px;\n  margin: 0;\n  padding: 0;\n  width: 25px;\n  height: 25px;\n  border: none;\n  cursor: pointer;\n  border-radius: 50%;\n  background-color: transparent;\n  transform: rotate(45deg);\n  transition: 0.25s background-color;\n  background: linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%)\n      center/18px 2px no-repeat,\n    linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/2px\n      18px no-repeat;\n}\n\n.pdl-dialog-close:hover {\n  background-color: rgba(0, 0, 0, 0.05);\n}\n\n.pdl-dialog-content hr {\n  height: 0 !important;\n  margin: 0;\n  border: none;\n  border-top: 1px solid rgba(0, 0, 0, 0.1);\n}\n\n.pdl-dialog-content hr.sub {\n  margin-inline-start: 1.5em;\n}\n\n/* button */\n.pdl-dialog-button {\n  font-size: 16px;\n  background-color: #fff;\n  border: 1px solid rgb(125, 125, 125);\n  border-radius: 5px;\n  padding: 0.5em 1.5em;\n  cursor: pointer;\n  transition: 0.2s opacity;\n  line-height: 1.15;\n}\n.pdl-dialog-button:hover {\n  opacity: 0.7;\n}\n\n.pdl-dialog-button.primary {\n  color: #fff;\n  background-color: #0096fa;\n  border-color: #0096fa;\n}\n\n.pdl-dialog-button.icon {\n  padding: 0.5em 0.8em;\n}\n\n.pdl-dialog-button[disabled] {\n  cursor: not-allowed !important;\n  color: #c0c4cc;\n  background-color: #fff;\n  border-color: #e4e7ed;\n  opacity: 1 !important;\n}\n\n.pdl-dialog-button[disabled]:hover {\n  opacity: 1 !important;\n}\n\n.pdl-dialog-button.primary[disabled] {\n  color: #fff;\n  background-color: #a0cfff;\n  border-color: #a0cfff;\n}\n\n/* tabs */\n.pdl-tabs-nav {\n  display: flex;\n  align-items: center;\n  border-bottom: 1px solid #dcdfe6;\n  position: relative;\n}\n\n.pdl-tabs-nav .pdl-tab-item {\n  padding: 0px 16px;\n  line-height: 2.5;\n  cursor: pointer;\n  transition: 0.2s color;\n}\n\n.pdl-tabs-nav .pdl-tab-item:nth-child(2) {\n  padding-left: 0px;\n}\n\n.pdl-tabs-nav .pdl-tab-item:last-child {\n  padding-right: 0px;\n}\n\n.pdl-tab-item:hover,\n.pdl-tab-item.active {\n  color: #0096fa;\n}\n\n.pdl-tab-item.active {\n  font-weight: 700;\n}\n\n.pdl-tabs-content {\n  padding: 16px;\n  min-height: 200px;\n}\n\n.pdl-tabs-nav .pdl-tabs__active-bar {\n  position: absolute;\n  bottom: 0;\n  left: 0;\n  height: 2px;\n  background-color: #0096fa;\n  z-index: 1;\n  transition: width 0.2s, transform 0.2s;\n}\n\n/* filename */\n\n#pdl-setting-filename a {\n  color: #0096fa;\n  text-decoration: underline;\n}\n#pdl-setting-filename .tags-option label,\n#pdl-setting-filename .tags-option input {\n  cursor: pointer;\n}\n\n#pdl-setting-filename .tags-content {\n  flex: 1;\n  display: flex;\n  gap: 20px;\n}\n\n#pdl-setting-filename .pdl-input-wrap,\n#pdl-setting-filename .tags-option {\n  display: flex;\n  align-items: center;\n  margin: 12px 0;\n  gap: 12px;\n}\n\n#pdl-setting-filename .pdl-input-wrap label,\n#pdl-setting-filename .tags-option .tags-title {\n  cursor: default;\n  font-weight: 700;\n  width: 6em;\n}\n\n#pdl-setting-filename .pdl-input-wrap input[type='text'] {\n  height: auto;\n  padding: 0.5em;\n  font-size: 16px;\n  line-height: 1.5;\n  flex: 1;\n  border: 1px solid #333;\n}\n\n#pdl-setting-filename .pdl-input-wrap input[type='text']:focus {\n  background-color: #fff !important;\n}\n\n#pdl-setting-filename .pdl-input-wrap button {\n  line-height: 1.5;\n}\n\n/* ugoria */\n#pdl-setting-ugoria p.option-header {\n  font-weight: 700;\n}\n\n#pdl-setting-ugoria #pdl-ugoria-format-wrap {\n  display: flex;\n  justify-content: space-between;\n  flex-wrap: nowrap;\n  margin: 1.5em 1em;\n}\n\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item,\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item label,\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item input {\n  cursor: pointer;\n}\n\n#pdl-ugoria-format-wrap .pdl-ugoria-format-item label {\n  padding-left: 4px;\n}\n\n/* history */\n#pdl-setting-history div {\n  text-align: center;\n  margin: 1em 0;\n}\n\n#pdl-setting-history .btn-history {\n  width: 80%;\n}\n\n/* others setting */\n#pdl-setting-others .pdl-options {\n  display: flex;\n  align-items: center;\n  gap: 20px;\n  cursor: pointer;\n  transition: 0.2s background-color;\n  padding: 1em 0.5em;\n  border-radius: 4px;\n}\n\n#pdl-setting-others .pdl-options:hover {\n  background-color: rgba(0, 0, 0, 0.05);\n}\n\n#pdl-setting-others .pdl-options.sub-option {\n  padding: 0.5em;\n  padding-inline-start: 2em;\n}\n\n/* donate */\n#pdl-setting-donate {\n  text-align: center;\n}\n\n#pdl-setting-donate p {\n  margin: 0.5em 0;\n}\n\n/* upgrade msg */\n.pdl-changelog h4 {\n  margin: 1em 0;\n  font-weight: bold !important;\n}\n\n.pdl-changelog ul {\n  padding-inline-start: 40px !important;\n}\n\n.pdl-changelog li {\n  line-height: 2;\n  list-style-type: disc !important;\n}\n\n.pdl-changelog p {\n  margin: 0.5em 0;\n}\n";
  n(css$2,{});

  var css$1 = ".pdl-dlbar-status_bar {\n  flex-grow: 1;\n  height: 46px;\n  line-height: 46px;\n  padding-right: 8px;\n  text-align: right;\n  font-weight: bold;\n  font-size: 16px;\n  color: rgb(133, 133, 133);\n  cursor: default;\n  white-space: nowrap;\n}\n\n.pdl-btn-all.pdl-btn-all,\n.pdl-stop.pdl-stop {\n  background-color: transparent;\n  border: none;\n  padding: 0 10px;\n}\n\n.pdl-btn-all.pdl-btn-all:hover,\n.pdl-stop.pdl-stop:hover {\n  color: rgb(31, 31, 31);\n}\n\n.pdl-btn-all::before,\n.pdl-stop::before {\n  content: '';\n  height: 24px;\n  width: 24px;\n  transition: background-image 0.2s ease 0s;\n  background: no-repeat center/85%;\n}\n\n.pdl-btn-all::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-stop::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-btn-all:hover::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-stop:hover::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n.pdl-hide {\n  display: none !important;\n}\n\n.pdl-filter-wrap {\n  display: flex;\n  justify-content: flex-end;\n  gap: 12px;\n  margin: 4px 0;\n  font-weight: bold;\n  font-size: 14px;\n  line-height: 14px;\n  color: rgb(133, 133, 133);\n  transition: color 0.2s ease 0s;\n}\n\n.pdl-filter-wrap.unavailable {\n  pointer-events: none !important;\n  opacity: 0.5 !important;\n}\n\n.pdl-filter-wrap .pdl-filter:hover {\n  color: rgb(31, 31, 31);\n}\n\n.pdl-filter-wrap label {\n  padding-left: 8px;\n  cursor: pointer;\n}\n\n.pdl-checkbox.pdl-checkbox {\n  vertical-align: top;\n  appearance: none;\n  position: relative;\n  box-sizing: border-box;\n  width: 28px;\n  border: 2px solid transparent;\n  cursor: pointer;\n  border-radius: 14px;\n  height: 14px;\n  background-color: rgba(133, 133, 133);\n  transition: background-color 0.2s ease 0s, box-shadow 0.2s ease 0s;\n}\n\n.pdl-checkbox:hover {\n  background-color: rgba(31, 31, 31);\n}\n\n.pdl-checkbox::after {\n  content: '';\n  position: absolute;\n  display: block;\n  top: 0px;\n  left: 0px;\n  width: 10px;\n  height: 10px;\n  transform: translateX(0px);\n  background-color: rgb(255, 255, 255);\n  border-radius: 10px;\n  transition: transform 0.2s ease 0s;\n}\n\n.pdl-checkbox:checked {\n  background-color: rgb(0, 150, 250);\n}\n\n.pdl-checkbox:checked::after {\n  transform: translateX(14px);\n}\n\n.pdl-dlbar {\n  display: flex;\n  flex-grow: 1;\n}\n\n.pdl-dlbar-follow_latest {\n  padding: 0 8px;\n}\n";
  n(css$1,{});

  var css = "[data-theme='dark'] .pdl-btn-all::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-btn-main,\n[data-theme='dark'] .pdl-btn-all:hover::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23D6D6D6' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-stop::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-stop:hover::before {\n  background-image: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23D6D6D6' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E\");\n}\n\n[data-theme='dark'] .pdl-checkbox:not(:checked):hover {\n  background-color: rgba(155, 155, 155);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-tag {\n  background-color: rgba(255, 255, 255, 0.4);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-modal-tag {\n  background-color: rgba(255, 255, 255, 0.4);\n}\n\n[data-theme='dark'] .pdl-btn.pdl-modal-tag:hover {\n  background-color: rgba(255, 255, 255, 0.6);\n}\n\n[data-theme='dark'] .pdl-wrap .pdl-filter:hover,\n[data-theme='dark'] .pdl-stop.pdl-stop:hover,\n[data-theme='dark'] .pdl-btn-all.pdl-btn-all:hover {\n  color: rgb(214, 214, 214);\n}\n\n/* modal */\n[data-theme='dark'] .pdl-dialog {\n  background-color: rgb(31, 31, 31);\n}\n\n[data-theme='dark'] .pdl-dialog-footer button {\n  background-color: rgb(245, 245, 245);\n}\n\n[data-theme='dark'] .pdl-dialog-content hr {\n  border-top: 1px solid rgba(255, 255, 255, 0.3);\n}\n\n/* others setting */\n[data-theme='dark'] #pdl-setting-others .pdl-options:hover {\n  background-color: rgba(255, 255, 255, 0.1);\n}\n";
  n(css,{});

  var IllustType;
  (function (IllustType) {
      IllustType[IllustType["illusts"] = 0] = "illusts";
      IllustType[IllustType["manga"] = 1] = "manga";
      IllustType[IllustType["ugoira"] = 2] = "ugoira";
  })(IllustType || (IllustType = {}));
  var BookmarkRestrict;
  (function (BookmarkRestrict) {
      BookmarkRestrict[BookmarkRestrict["public"] = 0] = "public";
      BookmarkRestrict[BookmarkRestrict["private"] = 1] = "private";
  })(BookmarkRestrict || (BookmarkRestrict = {}));

  const defaultSettings = Object.freeze({
      version: '0.8.8',
      ugoriaFormat: 'zip',
      folderPattern: 'pixiv/{artist}',
      filenamePattern: '{artist}_{title}_{id}_p{page}',
      tagLang: 'ja',
      showMsg: true,
      log: false,
      filter: {
          exclude_downloaded: false,
          [IllustType.illusts]: true,
          [IllustType.manga]: true,
          [IllustType.ugoira]: true
      },
      bundleIllusts: false,
      bundleManga: false,
      addBookmark: false,
      addBookmarkWithTags: false,
      privateR18: false
  });
  const regexp = {
      preloadData: /"meta-preload-data" content='(.*?)'>/,
      globalData: /"meta-global-data" content='(.*?)'>/,
      artworksPage: /artworks\/(\d+)$/,
      userPage: /users\/(\d+)/,
      bookmarkPage: /users\/\d+\/bookmarks\/artworks/,
      userPageTags: /users\/\d+\/(artworks|illustrations|manga|bookmarks(?!artworks))/,
      ppSearchPage: /\/tags\/.*\/(artworks|illustrations|manga)/,
      suscribePage: /bookmark_new_illust/,
      activityHref: /illust_id=(\d+)/,
      originSrcPageNum: /(?<=_p)\d+/,
      followLatest: /\/bookmark_new_illust(?:_r18)?\.php/
  };
  const depsUrls = {
      gifWorker: 'https://cdnjs.cloudflare.com/ajax/libs/gif.js/0.2.0/gif.worker.js',
      pako: 'https://cdnjs.cloudflare.com/ajax/libs/pako/2.0.4/pako.min.js',
      upng: 'https://cdnjs.cloudflare.com/ajax/libs/upng-js/2.1.0/UPNG.min.js'
  };
  const creditCode = `<img style="display: block; margin: 1em auto; width: 200px"
src=""
/>`;

  const lang = document.documentElement.getAttribute('lang')?.toLowerCase() ||
      'en';
  const i18nLib = {
      en: {
          bookmarks: 'Bookmarks',
          bookmarks_public: 'Public',
          bookmarks_private: 'Private',
          stop: 'Stop',
          dlbar_category_works: 'Works',
          dlbar_filter_illusts: 'Illustrations',
          dlbar_filter_manga: 'Manga',
          dlbar_filter_ugoria: 'Ugoria',
          dlbar_filter_exclude_downloaded: 'Exclude downloaded',
          'dlbar.follow_latest.category_all.single': 'All (page)',
          'dlbar.follow_latest.category_all.all': 'All (batch)',
          'dlbar.follow_latest.category_r18.single': 'R-18 (page)',
          'dlbar.follow_latest.category_r18.all': 'R-18 (batch)',
          feedback: 'Feedback',
          gm_menu_setting: 'Settings',
          tabs_nav_filename: 'Filename',
          tags_lang: 'Tags language: ',
          tags_tips: '{artist}, {artistID}, {title}, {id}, {page}, {tags}',
          tags_tips2: 'Note: Tags language may not be the language you selected, <a href="https://crowdin.com/project/pixiv-tags" target="_blank">some tags without translations</a> may still be in other languages.',
          folder: 'Folder:',
          folder_tips: "I don't need subfolder",
          folder_tips2: "If you don't need a subfolder, just leave the folder name blank",
          folder_vm_tips: "VM doesn't support",
          folder_api_tips: 'Need Browser Api',
          filename: 'FileName:',
          filename_tips: 'Your Name?',
          tabs_nav_history: 'History',
          'modal_history_content.import_btn': 'Import',
          'modal_history_content.export_btn': 'Export',
          'modal_history_content.clear_btn': 'Clear',
          clear_history_tips: 'Do you really want to clear history?',
          tabs_nav_ugoria: 'Ugoria',
          modal_ugoria_format_title: 'Ugoria Format:',
          tabs_nav_others: 'Others',
          'modal_others_content.bundle_illusts': 'Pack multi-page illustrations into a .zip archive',
          'modal_others_content.bundle_manga': 'Pack manga into a .zip archive',
          'modal_others_content.add_bookmark': 'Bookmark work when downloading a single work',
          'modal_others_content.add_bookmark_tags': 'Add works tags',
          'modal_others_content.add_bookmark_private_r18': 'Bookmark R-18 works to private category',
          tabs_nav_donate: 'Feedback / Donate'
      },
      'zh-cn': {
          bookmarks: '收藏',
          bookmarks_public: '公开',
          bookmarks_private: '不公开',
          stop: '停止',
          dlbar_category_works: '作品',
          dlbar_filter_illusts: '插画',
          dlbar_filter_manga: '漫画',
          dlbar_filter_ugoria: '动图',
          dlbar_filter_exclude_downloaded: '排除已下载图片',
          'dlbar.follow_latest.category_all.single': '全部(单页)',
          'dlbar.follow_latest.category_all.all': '全部(批量)',
          'dlbar.follow_latest.category_r18.single': 'R-18(单页)',
          'dlbar.follow_latest.category_r18.all': 'R-18(批量)',
          feedback: '有问题or想建议?这里反馈',
          gm_menu_setting: '设置',
          tabs_nav_filename: '文件名',
          tags_lang: '标签语言:',
          tags_tips: '{artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}:作品pixiv ID, {page}:页码, {tags}:作品标签。',
          tags_tips2: '请注意:标签翻译不一定是你选择的语言,部分<a href="https://crowdin.com/project/pixiv-tags" target="_blank">无对应语言翻译的标签</a>仍可能是其他语言。',
          folder: '文件夹名:',
          folder_tips: '我不想保存到子文件夹',
          folder_tips2: '如果不想保存到画师目录,文件夹名留空即可。',
          folder_vm_tips: 'Violentmonkey不支持',
          folder_api_tips: '需要Browser Api',
          filename: '文件名:',
          filename_tips: '你的名字?',
          tabs_nav_history: '历史记录',
          'modal_history_content.import_btn': '导入记录',
          'modal_history_content.export_btn': '导出记录',
          'modal_history_content.clear_btn': '清除记录',
          clear_history_tips: '真的要清除历史记录吗?',
          tabs_nav_ugoria: '动图',
          modal_ugoria_format_title: '动图格式:',
          tabs_nav_others: '其它',
          'modal_others_content.bundle_illusts': '将多页插图打包为.zip压缩包',
          'modal_others_content.bundle_manga': '将多页漫画作品打包为.zip压缩包',
          'modal_others_content.add_bookmark': '下载单个作品时收藏作品',
          'modal_others_content.add_bookmark_tags': '收藏时添加作品标签',
          'modal_others_content.add_bookmark_private_r18': '将R-18作品收藏到不公开类别',
          tabs_nav_donate: '反馈 / 赞赏'
      },
      zh: {},
      ja: {},
      ko: {},
      'zh-tw': {}
  };
  i18nLib.en = Object.create(i18nLib['zh-cn'], Object.getOwnPropertyDescriptors(i18nLib.en));
  i18nLib.ja = Object.create(i18nLib.en);
  i18nLib.ko = Object.create(i18nLib.en);
  i18nLib['zh-tw'] = Object.create(i18nLib['zh-cn']);
  i18nLib.zh = i18nLib['zh-cn'];
  const i18n = (key) => i18nLib[lang]?.[key] || `i18n[${lang}][${key}] not found`;
  const modalHtml = {
      upgradeMsgTitle: `<h3>Pixiv Downloader ${defaultSettings.version}</h3>`,
      upgradeMsgContent: `      <div class="pdl-changelog">
  <p>可乐涨价,我的暑假结束了...</p>
  <h4>新增</h4>
  <ul>
    <li>现在可以批量下载“已关注用户的新作”。</li>
    <li>——可以选择下载单页或全部(最多34页)</li>
    <li>——点击“全部”或“R-18”按钮切换需要下载的作品类别</li>
  </ul>
  <h4>其它</h4>
  <ul>
  <li>稍微提高了webp转换速度</li>
  <li>一些其它调整</li>
</ul>
</div>`,
      modalCreditFooter: `<style>.pdl-dialog-footer {
    position: relative;
    font-size: 12px;
  }</style><details style="margin-top: 1.5em;">
  <summary style="display: inline-block; list-style: none; cursor: pointer; color: #0096fa; text-decoration: underline">脚本还行?请我喝杯可乐吧!</summary>
  ${creditCode}
  <p style="text-align: center">愿你每天都能找到对的色图,就像我每天都能喝到香草味可乐</p>
</details>`,
      modalFeedback: `<a target="_blank" style="position: absolute; right: 0px; top: 0px; color: #0096fa; text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback">${i18n('feedback')}</a>`,
      modalSettingsContent: `      <div>
  <div class="pdl-tabs-nav">
  <div class="pdl-tabs__active-bar"></div>
  </div>
  <div class="pdl-tabs-content">
  </div>
</div>`,
      modalSettingFilename: `            <div id="pdl-setting-filename">
<div>
  <div class="pdl-input-wrap">
    <label for="pdlfolder">${i18n('folder')}</label>
    <input type="text" id="pdlfolder" maxlength="100" />
    <button id="pdl-filename-folder-reset" class="pdl-dialog-button icon" disabled>↺</button>
    <button id="pdl-filename-folder-confirm" class="pdl-dialog-button icon primary" disabled>✓</button>
  </div>
  <div class="pdl-input-wrap">
    <label for="pdlfilename">${i18n('filename')}</label>
    <input
      type="text"
      id="pdlfilename"
      placeholder="${i18n('filename_tips')}"
      required
      maxlength="100"
    />
    <button id="pdl-filename-filename-reset" class="pdl-dialog-button icon" disabled>↺</button>
    <button id="pdl-filename-filename-confirm" class="pdl-dialog-button icon primary" disabled>✓</button>
  </div>
</div>
<div class="tags-option">
<span class="tags-title">${i18n('tags_lang')}</span>
<div class="tags-content">
  <div class="tags-item">
    <input type="radio" name="lang" id="lang_ja" value="ja" />
    <label for="lang_ja">日本語(default)</label>
  </div>
  <div class="tags-item">
    <input type="radio" name="lang" id="lang_zh" value="zh" />
    <label for="lang_zh">简中</label>
  </div>
  <div class="tags-item">
    <input type="radio" name="lang" id="lang_zh_tw" value="zh_tw" />
    <label for="lang_zh_tw">繁中</label>
  </div>
  <div class="tags-item">
    <input type="radio" name="lang" id="lang_en" value="en" />
    <label for="lang_en">English</label>
  </div>
</div>
</div>
<p style="font-size: 14px; margin: 0.5em 0">
  ${i18n('tags_tips')}
</p>
<p style="font-size: 14px; margin: 0.5em 0">
  ${i18n('folder_tips2')}
</p>
<p style="font-size: 14px; margin: 0.5em 0">
  ${i18n('tags_tips2')}
</p>
</div>`,
      modalSettingUgoria: `            <div id="pdl-setting-ugoria">
<p class="option-header">${i18n('modal_ugoria_format_title')}</p>
<div id="pdl-ugoria-format-wrap">
  <div class="pdl-ugoria-format-item">
    <input
      type="radio"
      id="pdl-ugoria-zip"
      value="zip"
      name="format"
    /><label for="pdl-ugoria-zip">Zip</label>
  </div>
  <div class="pdl-ugoria-format-item">
    <input
      type="radio"
      id="pdl-ugoria-gif"
      value="gif"
      name="format"
    /><label for="pdl-ugoria-gif">Gif</label>
  </div>
  <div class="pdl-ugoria-format-item">
    <input
      type="radio"
      id="pdl-ugoria-apng"
      value="png"
      name="format"
    /><label for="pdl-ugoria-apng">Png</label>
  </div>
  <div class="pdl-ugoria-format-item">
    <input
      type="radio"
      id="pdl-ugoria-webm"
      value="webm"
      name="format"
    /><label for="pdl-ugoria-webm">Webm</label>
  </div>
  <div class="pdl-ugoria-format-item">
    <input
      type="radio"
      id="pdl-ugoria-webp"
      value="webp"
      name="format"
    /><label for="pdl-ugoria-webp">Webp</label>
  </div>
</div>
</div>`,
      modalSettingHistory: `            <div id="pdl-setting-history">
<div>
  <button id="pdl-export" class="btn-history pdl-dialog-button primary">
    ${i18n('modal_history_content.export_btn')}
  </button>
</div>
<div>
  <input
    type="file"
    id="pdl-import"
    accept=".txt"
    style="display: none"
  /><button id="pdl-import-btn" class="btn-history pdl-dialog-button primary">
    ${i18n('modal_history_content.import_btn')}
  </button>
</div>
<div>
  <button id="pdl-clear-history" class="btn-history pdl-dialog-button primary">
    ${i18n('modal_history_content.clear_btn')}
  </button>
</div>
</div>`,
      modalSettingOthers: `            <div id="pdl-setting-others">
<div>
  <label class="pdl-options"
    ><input
      id="pdl-options-bundle-illusts"
      type="checkbox"
      class="pdl-checkbox"
    /><span
      >${i18n('modal_others_content.bundle_illusts')}</span
    ></label
  >
</div>
<hr />
<div>
  <label class="pdl-options"
    ><input
      id="pdl-options-bundle-manga"
      type="checkbox"
      class="pdl-checkbox"
    /><span
      >${i18n('modal_others_content.bundle_manga')}</span
    ></label
  >
</div>
<hr />
<div>
  <label class="pdl-options"
    ><input
      id="pdl-options-add-bookmark"
      type="checkbox"
      class="pdl-checkbox"
    /><span
      >${i18n('modal_others_content.add_bookmark')}</span
    ></label
  >
</div>
<hr />
<div>
  <label class="pdl-options sub-option"
    ><input
      id="pdl-options-add-bookmark-tags"
      type="checkbox"
      class="pdl-checkbox"
    /><span
      >${i18n('modal_others_content.add_bookmark_tags')}</span
    ></label
  >
</div>
<hr class="sub"/>
<div>
  <label class="pdl-options sub-option"
    ><input
      id="pdl-options-add-bookmark-private-r18"
      type="checkbox"
      class="pdl-checkbox"
    /><span
      >${i18n('modal_others_content.add_bookmark_private_r18')}</span
    ></label
  >
</div>
</div>`,
      modalSettingDonate: `            <div id="pdl-setting-donate">
${creditCode}
<p>如果脚本有帮助到你,欢迎扫码请我喝杯可乐 ^_^</p>
<p><a target="_blank" style="color: #0096fa; text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback">${i18n('feedback')}</a></p>
</div>`
  };

  function add$1(pixivId) {
      this._records.add(pixivId);
      localStorage.setItem(`pdlTemp-${pixivId}`, '');
  }
  function has(pixivId) {
      return this._records.has(pixivId);
  }
  function getHistory() {
      const storage = localStorage.pixivDownloader || '[]';
      return new Set(JSON.parse(storage));
  }
  function updateHistory() {
      Object.keys(localStorage).forEach((key) => {
          const matchResult = /pdlTemp-(\d+)/.exec(key);
          if (matchResult) {
              this._records.add(matchResult[1]);
              localStorage.removeItem(matchResult[0]);
          }
      });
      this.saveHistory();
  }
  function clearHistory() {
      const isConfirm = confirm(i18n('clear_history_tips'));
      if (!isConfirm)
          return;
      this.updateHistory();
      this._records = new Set();
      localStorage.pixivDownloader = '[]';
      location.reload();
  }
  function saveHistory(historyArr) {
      if (historyArr) {
          localStorage.pixivDownloader = JSON.stringify(historyArr);
      }
      else {
          localStorage.pixivDownloader = JSON.stringify([...this._records]);
      }
  }
  function getAll() {
      return [...this._records];
  }
  const pixivHistory = {
      _records: getHistory(),
      add: add$1,
      has,
      getAll,
      updateHistory,
      saveHistory,
      clearHistory
  };

  function debugLog(...msgs) {
  }

  function sleep(delay) {
      return new Promise((resolve) => {
          setTimeout(resolve, delay);
      });
  }
  function wakeableSleep(delay) {
      let wake = () => void {};
      const sleep = new Promise((r) => {
          setTimeout(r, delay);
          wake = r;
      });
      return {
          wake,
          sleep
      };
  }
  function getSelfId() {
      return document.querySelector('#qualtrics_user-id')
          ?.textContent;
  }
  const env = {
      isViolentmonkey: GM_info.scriptHandler === 'Violentmonkey',
      isBlobDlAvaliable: !(navigator.userAgent.includes('Firefox') &&
          GM_info.scriptHandler === 'Tampermonkey' &&
          parseFloat(GM_info.version ?? '') > 4.17),
      isSupportSubpath: GM_info.downloadMode && GM_info.downloadMode === 'browser'
  };

  function getSettings() {
      let settings;
      if (!localStorage.pdlSetting) {
          settings = { ...defaultSettings };
          saveSettings(settings);
      }
      else {
          settings = JSON.parse(localStorage.pdlSetting);
          if (settings.version !== defaultSettings.version) {
              settings.version = defaultSettings.version;
              settings.showMsg = true;
              settings = { ...defaultSettings, ...settings };
              saveSettings(settings);
          }
      }
      return settings;
  }
  function saveSettings(settingObj) {
      settingObj = settingObj || settings;
      localStorage.pdlSetting = JSON.stringify(settingObj);
  }
  function upgradeSettings(key, value) {
      if (key in settings && settings[key] !== value) {
          settings[key] = value;
          saveSettings();
      }
  }
  const settings = getSettings();

  const handleWorker = `
let webpApi = {};
Module.onRuntimeInitialized = () => {
  webpApi = {
    init: Module.cwrap('init', '', ['number', 'number', 'number']),
    createBuffer: Module.cwrap('createBuffer', 'number', ['number']),
    addFrame: Module.cwrap('addFrame', 'number', ['number', 'number', 'number']),
    generate: Module.cwrap('generate', 'number', []),
    freeResult: Module.cwrap('freeResult', '', []),
    getResultPointer: Module.cwrap('getResultPointer', 'number', []),
    getResultSize: Module.cwrap('getResultSize', 'number', []),
  };

  postMessage('ok');
};

onmessage = (evt) => {
  const { data, delays, lossless = 1, quality = 75, method = 4} = evt.data;

  webpApi.init(lossless, quality, method);
  data.forEach((u8a, idx) => {
    const pointer = webpApi.createBuffer(u8a.length);
    Module.HEAPU8.set(u8a, pointer);
    webpApi.addFrame(pointer, u8a.length, delays[idx]);
    postMessage(idx);
  });

  webpApi.generate();
  const resultPointer = webpApi.getResultPointer();
  const resultSize = webpApi.getResultSize();
  const result = new Uint8Array(Module.HEAP8.buffer, resultPointer, resultSize);
  postMessage(result);
  webpApi.freeResult();
};`;

  async function _requestJson(url, init) {
      const res = await fetch(url, init);
      if (!res.ok)
          throw new Error('[Error]fail to fetch:' + url + ', code:' + res.status);
      const data = await res.json();
      if (data.error)
          throw new Error('[Error]JSON return error: ' + data.message);
      return data.body;
  }
  async function getJson(url) {
      let json;
      let retry = 0;
      const MAX_RETRY = 3;
      do {
          try {
              debugLog('[Info]fetch url:', url);
              json = await _requestJson(url);
          }
          catch (error) {
              retry++;
              if (retry === MAX_RETRY)
                  throw error;
              sleep(3000);
          }
      } while (!json);
      return json;
  }
  async function getArtworkHtml(illustId) {
      let params = '';
      if (settings.tagLang !== 'ja')
          params = '?lang=' + settings.tagLang;
      const res = await fetch('https://www.pixiv.net/artworks/' + illustId + params);
      if (!res.ok)
          throw new Error(String(res.status));
      return await res.text();
  }
  async function _getDeps(url) {
      return fetch(url).then((res) => {
          if (res.ok)
              return res.text();
          throw new Error(`[Pixiv Downloader]Fetch dependency ${url} failed: ${res.status} ${res.statusText}`);
      });
  }
  async function getGifWS() {
      let gifWS;
      if (!(gifWS = await GM_getValue('gifWS'))) {
          gifWS = await _getDeps(depsUrls.gifWorker);
          GM_setValue('gifWS', gifWS);
      }
      return gifWS;
  }
  async function getApngWS() {
      let apngWS;
      if (!(apngWS = await GM_getValue('apngWS'))) {
          const [pako, upng] = await Promise.all([
              _getDeps(depsUrls.pako),
              _getDeps(depsUrls.upng)
          ]);
          const upngScript = upng
              .replace('window.UPNG', 'UPNG')
              .replace('window.pako', 'pako');
          const workerEvt = `onmessage = (evt) => {
      const {data, width, height, delay } = evt.data;
      const png = UPNG.encode(data, width, height, 0, delay, {loop: 0});
      if (!png) console.log('Convert Apng failed.');
      postMessage(png);
    };`;
          apngWS = workerEvt + pako + upngScript;
          GM_setValue('apngWS', apngWS);
      }
      return apngWS;
  }
  function getWebpWS() {
      return workerChunk__default["default"] + handleWorker;
  }
  function addBookmark$1(illustId, token, tags, restrict) {
      return _requestJson('/ajax/illusts/bookmarks/add', {
          method: 'POST',
          headers: {
              accept: 'application/json',
              'content-type': 'application/json; charset=utf-8',
              'x-csrf-token': token
          },
          body: JSON.stringify({
              illust_id: illustId,
              restrict,
              comment: '',
              tags
          })
      });
  }
  function getFollowLatestWorks(page, mode) {
      return _requestJson(`/ajax/follow_latest/illust?p=${page}&mode=${mode}&lang=jp`);
  }
  const api = {
      getJson,
      getArtworkHtml,
      getGifWS,
      getApngWS,
      getWebpWS,
      getFollowLatestWorks,
      addBookmark: addBookmark$1
  };

  function initialDeps() {
      return Promise.all([api.getGifWS(), api.getApngWS(), api.getWebpWS()]).then(([gif, apng, webp]) => {
          this._deps.gif = URL.createObjectURL(new Blob([gif], { type: 'text/javascript' }));
          this._deps.apng = URL.createObjectURL(new Blob([apng], { type: 'text/javascript' }));
          this._deps.webp = URL.createObjectURL(new Blob([webp], { type: 'text/javascript' }));
          return this;
      });
  }
  function _createImgElements(zip) {
      const eles = [];
      zip.forEach((_, file) => {
          eles.push(new Promise((resolve) => {
              const image = new Image();
              image.onload = () => {
                  resolve(image);
              };
              file
                  .async('blob')
                  .then((blob) => void (image.src = URL.createObjectURL(blob)));
          }));
      });
      return Promise.all(eles);
  }
  function createInstance() {
      const zip = new JSZip__default["default"]();
      const freeApngWorkers = [];
      const freeWebpWorkers = [];
      const MAX_CONVERT = 2;
      let queue = [];
      let active = [];
      let isStop = false;
      const convertTo = {
          webp: (frames, convertMeta) => {
              return new Promise((resolve, reject) => {
                  let worker;
                  let reuse = false;
                  if (freeWebpWorkers.length) {
                      worker = freeWebpWorkers.shift();
                      reuse = true;
                  }
                  else {
                      worker = new Worker(createConverter._deps.webp);
                  }
                  convertMeta.abort = () => {
                      reject('[Info]Convert stop manually, reject when convert webp. ' +
                          convertMeta.id);
                      convertMeta.isAborted = true;
                      worker.terminate();
                  };
                  const workerLoad = new Promise((resolve) => {
                      if (reuse)
                          return resolve();
                      worker.onmessage = (evt) => {
                          if (evt.data === 'ok') {
                              resolve();
                          }
                      };
                  });
                  const delays = convertMeta.source.framesInfo.map((frameInfo) => {
                      return Number(frameInfo.delay);
                  });
                  const data = [];
                  let completed = 0;
                  frames.forEach((frame, idx) => {
                      const canvas = document.createElement('canvas');
                      const width = (canvas.width = frame.naturalWidth);
                      const height = (canvas.height = frame.naturalHeight);
                      const context = canvas.getContext('2d');
                      if (!context)
                          return;
                      context.drawImage(frame, 0, 0, width, height);
                      data.push(new Promise((resolve, reject) => {
                          canvas.toBlob((blob) => {
                              if (!blob)
                                  return reject('[Pixiv Downloader]toBlob failed: ' + idx);
                              blob.arrayBuffer().then((buffer) => {
                                  const u8a = new Uint8Array(buffer);
                                  resolve(u8a);
                                  convertMeta.onProgress?.((++completed / frames.length) * 0.5, 'webp');
                              });
                          }, 'image/webp', 1);
                      }));
                  });
                  workerLoad
                      .then(() => Promise.all(data))
                      .then((u8arrs) => {
                      if (convertMeta.isAborted)
                          return;
                      worker.onmessage = (evt) => {
                          if (typeof evt.data !== 'object') {
                              debugLog('[Info]Webp convert phrase 2:', convertMeta.id, evt.data);
                              convertMeta.onProgress?.(0.5 + (evt.data / frames.length) * 0.5, 'webp');
                          }
                          else {
                              freeWebpWorkers.push(worker);
                              resolve(new Blob([evt.data], { type: 'image/webp' }));
                          }
                      };
                      worker.postMessage({ data: u8arrs, delays });
                  })
                      .catch(console.error);
              });
          },
          gif: (frames, convertMeta) => {
              return new Promise((resolve, reject) => {
                  const gif = new GIF__default["default"]({
                      workers: 2,
                      quality: 10,
                      workerScript: createConverter._deps.gif
                  });
                  convertMeta.abort = () => {
                      gif.abort();
                      convertMeta.isAborted = true;
                  };
                  debugLog('[Info]Start convert:', convertMeta.id);
                  frames.forEach((frame, i) => {
                      gif.addFrame(frame, {
                          delay: convertMeta.source.framesInfo[i].delay
                      });
                  });
                  gif.on('progress', (progress) => {
                      debugLog('[Info]Convert progress:', convertMeta.id);
                      if (typeof convertMeta.onProgress === 'function')
                          convertMeta.onProgress(progress, 'gif');
                  });
                  gif.on('finished', (gifBlob) => {
                      resolve(gifBlob);
                  });
                  gif.on('abort', () => {
                      reject('[Info]Convert stop: abort. ' + convertMeta.id);
                  });
                  gif.render();
              });
          },
          png: (frames, convertMeta) => {
              return new Promise((resolve, reject) => {
                  const canvas = document.createElement('canvas');
                  const width = (canvas.width = frames[0].naturalWidth);
                  const height = (canvas.height = frames[0].naturalHeight);
                  const context = canvas.getContext('2d', { willReadFrequently: true });
                  if (!context)
                      return reject('[Error]Can not get canvas context');
                  const data = [];
                  const delay = convertMeta.source.framesInfo.map((frameInfo) => {
                      return Number(frameInfo.delay);
                  });
                  frames.forEach((frame) => {
                      if (convertMeta.isAborted)
                          throw ('[Info]Convert stop manually, reject when drawImage. ' +
                              convertMeta.id);
                      context.clearRect(0, 0, width, height);
                      context.drawImage(frame, 0, 0, width, height);
                      data.push(context.getImageData(0, 0, width, height).data);
                  });
                  debugLog('[Info]Start convert:', convertMeta.id);
                  let worker;
                  if (freeApngWorkers.length) {
                      worker = freeApngWorkers.shift();
                  }
                  else {
                      worker = new Worker(createConverter._deps.apng);
                  }
                  convertMeta.abort = () => {
                      reject('[Info]Convert stop manually, reject when convert apng. ' +
                          convertMeta.id);
                      convertMeta.isAborted = true;
                      worker.terminate();
                  };
                  worker.onmessage = function (e) {
                      freeApngWorkers.push(worker);
                      if (!e.data) {
                          return reject('[Error]apng data is null. ' + convertMeta.id);
                      }
                      const pngBlob = new Blob([e.data], { type: 'image/png' });
                      resolve(pngBlob);
                  };
                  const cfg = { data, width, height, delay };
                  worker.postMessage(cfg);
              });
          },
          webm: (frames, convertMeta) => {
              return new Promise((resolve, reject) => {
                  const canvas = document.createElement('canvas');
                  const width = (canvas.width = frames[0].naturalWidth);
                  const height = (canvas.height = frames[0].naturalHeight);
                  const context = canvas.getContext('2d');
                  if (!context)
                      return reject('[Error]Can not get canvas context');
                  const stream = canvas.captureStream();
                  const recorder = new MediaRecorder(stream, {
                      mimeType: 'video/webm',
                      videoBitsPerSecond: 80000000
                  });
                  const delay = convertMeta.source.framesInfo.map((frame) => {
                      return Number(frame.delay);
                  });
                  const data = [];
                  let frame = 0;
                  const displayFrame = () => {
                      context.clearRect(0, 0, width, height);
                      context.drawImage(frames[frame], 0, 0);
                      if (convertMeta.isAborted) {
                          return recorder.stop();
                      }
                      setTimeout(() => {
                          if (typeof convertMeta.onProgress === 'function')
                              convertMeta.onProgress((frame + 1) / frames.length, 'webm');
                          if (frame === frames.length - 1) {
                              return recorder.stop();
                          }
                          else {
                              frame++;
                          }
                          displayFrame();
                      }, delay[frame]);
                  };
                  recorder.ondataavailable = (event) => {
                      if (event.data && event.data.size) {
                          data.push(event.data);
                      }
                  };
                  recorder.onstop = () => {
                      if (convertMeta.isAborted) {
                          return reject('[info]Convert stop manually, reject when convert webm.' +
                              convertMeta.id);
                      }
                      resolve(new Blob(data, { type: 'video/webm' }));
                  };
                  displayFrame();
                  recorder.start();
              });
          }
      };
      const convert = (convertMeta) => {
          const { id, source, resolve, reject } = convertMeta;
          let frames;
          active.push(convertMeta);
          if (typeof convertMeta.onProgress === 'function')
              convertMeta.onProgress(0, 'zip');
          const newFolder = zip.folder(id);
          if (!newFolder)
              throw '[Error]Can not get new root folder';
          newFolder
              .loadAsync(source.data)
              .then(_createImgElements)
              .then((imgEles) => {
              zip.remove(id);
              frames = imgEles;
              if (convertMeta.isAborted)
                  throw '[Info]Convert stop manually, reject when unzip. ' + id;
              return convertTo[source.format](frames, convertMeta);
          })
              .then(resolve)
              .catch(reject)
              .finally(() => {
              frames.forEach((frame) => URL.revokeObjectURL(frame.src));
              active.splice(active.indexOf(convertMeta), 1);
              if (queue.length)
                  convert(queue.shift());
          });
      };
      return {
          add: (convertSource, handler) => {
              debugLog('[Info]Converter add', convertSource.id);
              return new Promise((resolve, reject) => {
                  queue.push({
                      id: convertSource.id,
                      isAborted: false,
                      source: convertSource,
                      onProgress: handler?.onProgress,
                      resolve,
                      reject,
                      abort() {
                          this.isAborted = true;
                      }
                  });
                  while (active.length < MAX_CONVERT && queue.length && !isStop) {
                      convert(queue.shift());
                  }
              });
          },
          del: (taskIds) => {
              if (!taskIds.length)
                  return;
              isStop = true;
              active = active.filter((convertMeta) => {
                  if (taskIds.includes(convertMeta.id)) {
                      convertMeta.abort();
                  }
                  else {
                      return true;
                  }
              });
              queue = queue.filter((convertMeta) => !taskIds.includes(convertMeta.id));
              isStop = false;
              while (active.length < MAX_CONVERT && queue.length) {
                  convert(queue.shift());
              }
          }
      };
  }
  const createConverter = {
      _deps: {
          gif: '',
          apng: '',
          webp: ''
      },
      initialDeps,
      createInstance
  };

  const zip = new JSZip__default["default"]();
  function add(id, name, data) {
      zip.folder(id)?.file(name, data);
  }
  function bundle(id) {
      const folder = zip.folder(id);
      if (!folder)
          throw new Error('[Error]no folder:' + id);
      return folder.generateAsync({ type: 'blob' });
  }
  function remove(ids) {
      if (typeof ids === 'string') {
          zip.remove(ids);
      }
      else {
          const dirs = zip.filter((_, file) => file.dir).map((dir) => dir.name);
          const dirsToDel = ids.filter((id) => dirs.some((dir) => dir.includes(id)));
          dirsToDel.forEach((dir) => zip.remove(dir));
      }
  }
  function fileCount(id) {
      let count = 0;
      zip.folder(id)?.forEach(() => count++);
      return count;
  }
  const compressor = {
      add,
      bundle,
      remove,
      fileCount
  };

  const _saveWithoutSubpath = (blob, downloadMeta) => {
      const dlEle = document.createElement('a');
      dlEle.href = URL.createObjectURL(blob);
      dlEle.download = downloadMeta.source.path;
      dlEle.click();
      URL.revokeObjectURL(dlEle.href);
      downloadMeta.resolve(downloadMeta.taskId);
  };
  const _saveWithSubpath = (blob, downloadMeta) => {
      const imgUrl = URL.createObjectURL(blob);
      const request = {
          url: imgUrl,
          name: downloadMeta.source.path,
          onerror: (error) => {
              console.log('[pixiv downloader]Error when saving', downloadMeta.source.path);
              URL.revokeObjectURL(imgUrl);
              downloadMeta.reject(error);
          },
          onload: () => {
              if (typeof downloadMeta.onLoad === 'function')
                  downloadMeta.onLoad();
              URL.revokeObjectURL(imgUrl);
              downloadMeta.resolve(downloadMeta.taskId);
          }
      };
      downloadMeta.abort = GM_download(request).abort;
  };
  function createDownloader(converter) {
      const MAX_DOWNLOAD = 5;
      const MAX_RETRY = 3;
      const INTERVAL = 500;
      const TIMEOUT = 20000;
      let isStop = false;
      let queue = [];
      let active = [];
      let save;
      if (env.isBlobDlAvaliable && env.isSupportSubpath) {
          save = _saveWithSubpath;
      }
      else {
          debugLog('[Info]scriptHandler:', GM_info.scriptHandler, GM_info.version);
          save = _saveWithoutSubpath;
      }
      const download = (downloadMeta) => {
          const { taskId, source } = downloadMeta;
          debugLog('[Info]Start download:', source.path);
          active.push(downloadMeta);
          let abortObj;
          const errorHandler = errorHandlerFactory(downloadMeta);
          if ((!env.isBlobDlAvaliable || env.isViolentmonkey) &&
              !('kind' in source)) {
              abortObj = GM_download({
                  url: source.src,
                  name: source.path,
                  headers: {
                      referer: 'https://www.pixiv.net'
                  },
                  ontimeout: errorHandler,
                  onerror: errorHandler,
                  onload: async () => {
                      debugLog('[Info]Download complete', source.path);
                      if (typeof downloadMeta.onLoad === 'function')
                          downloadMeta.onLoad();
                      downloadMeta.resolve(taskId);
                      await sleep(INTERVAL);
                      active.splice(active.indexOf(downloadMeta), 1);
                      if (queue.length && !isStop)
                          download(queue.shift());
                  }
              });
          }
          else {
              abortObj = GM_xmlhttpRequest({
                  url: source.src,
                  timeout: TIMEOUT,
                  method: 'GET',
                  headers: {
                      referer: 'https://www.pixiv.net'
                  },
                  responseType: 'blob',
                  ontimeout: errorHandler,
                  onerror: errorHandler,
                  onprogress: (e) => {
                      if (e.lengthComputable &&
                          typeof downloadMeta.onProgress === 'function') {
                          downloadMeta.onProgress(e.loaded / e.total);
                      }
                  },
                  onload: async (e) => {
                      if (downloadMeta.state === 0 )
                          return debugLog();
                      if (!('kind' in source)) {
                          save(e.response, downloadMeta);
                      }
                      else if (source.kind === 'convert') {
                          const convertSource = {
                              id: taskId,
                              data: e.response,
                              format: settings.ugoriaFormat,
                              framesInfo: source.ugoiraMeta?.frames
                          };
                          converter
                              .add(convertSource, { onProgress: downloadMeta.onProgress })
                              .then((blob) => {
                              save(blob, downloadMeta);
                          }, downloadMeta.reject);
                      }
                      else if (source.kind === 'bundle') {
                          compressor.add(taskId, source.filename, e.response);
                          if (compressor.fileCount(taskId) === source.pageCount) {
                              compressor.bundle(taskId).then((blob) => {
                                  save(blob, downloadMeta);
                                  compressor.remove(taskId);
                              });
                          }
                          else {
                              downloadMeta.resolve(taskId);
                              if (typeof downloadMeta.onLoad === 'function')
                                  downloadMeta.onLoad();
                          }
                      }
                      await sleep(INTERVAL);
                      active.splice(active.indexOf(downloadMeta), 1);
                      if (queue.length && !isStop)
                          download(queue.shift());
                  }
              });
          }
          downloadMeta.abort = () => {
              downloadMeta.state = 0 ;
              abortObj.abort();
              downloadMeta.reject('[Warning]xhr abort manually. ' + taskId);
          };
      };
      function errorHandlerFactory(downloadMeta) {
          return function () {
              const { source } = downloadMeta;
              debugLog('[Error]xmlhttpRequest timeout:', source.src);
              downloadMeta.retry++;
              if (downloadMeta.retry > MAX_RETRY) {
                  downloadMeta.reject('[Error]xmlhttpRequest failed: ' + source.src);
                  console.log('[pixiv downloader]Network error:', source.path, source.src);
                  active.splice(active.indexOf(downloadMeta), 1);
                  if (queue.length && !isStop)
                      download(queue.shift());
              }
              else {
                  debugLog('[Warning]retry xhr:', downloadMeta.retry, source.src);
                  download(downloadMeta);
              }
          };
      }
      function add(metas, handler = {}) {
          if (metas.length < 1)
              return Promise.resolve('');
          const promises = [];
          metas.forEach((source) => {
              let downloadMeta;
              promises.push(new Promise((resolve, reject) => {
                  downloadMeta = {
                      taskId: source.taskId,
                      source,
                      state: 1 ,
                      retry: 0,
                      onProgress: handler.onProgress,
                      onLoad: handler.onLoad,
                      resolve,
                      reject
                  };
                  queue.push(downloadMeta);
              }));
          });
          while (active.length < MAX_DOWNLOAD && queue.length && !isStop) {
              download(queue.shift());
          }
          return Promise.all(promises).then(([taskId]) => taskId);
      }
      function del(taskIds) {
          if (!taskIds.length)
              return;
          isStop = true;
          active = active.filter((downloadMeta) => {
              if (taskIds.includes(downloadMeta.taskId)) {
                  downloadMeta.abort?.();
              }
              else {
                  return true;
              }
          });
          queue = queue.filter((downloadMeta) => !taskIds.includes(downloadMeta.taskId));
          isStop = false;
          while (active.length < MAX_DOWNLOAD && queue.length) {
              download(queue.shift());
          }
      }
      return {
          add,
          del
      };
  }

  const needBundle = (type) => {
      return ((type === IllustType.manga && settings.bundleManga) ||
          (type === IllustType.illusts && settings.bundleIllusts));
  };
  const replaceInvalidChar = (string) => {
      if (!string)
          return '';
      const temp = document.createElement('div');
      temp.innerHTML = string;
      if (!temp.textContent)
          return '';
      return temp.textContent
          .trim()
          .replace(/^\.|\.$/g, '')
          .replace(/[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?|]/g, '')
          .replace(/"/g, "'")
          .replace(/</g, '﹤')
          .replace(/>/g, '﹥');
  };
  const getFilePath = ({ user, userId, title, tagStr, illustId, page, ext }, option = { needBundle: false, needConvert: false }) => {
      let pathPattern;
      if (settings.folderPattern &&
          env.isSupportSubpath &&
          (!option.needConvert || env.isBlobDlAvaliable) &&
          !option.needBundle) {
          pathPattern = settings.folderPattern + '/' + settings.filenamePattern;
      }
      else {
          pathPattern = settings.filenamePattern;
      }
      if (option.needBundle && !settings.filenamePattern.includes('{page}')) {
          pathPattern += '_{page}';
      }
      return (pathPattern
          .replaceAll('{artist}', user)
          .replaceAll('{artistID}', userId)
          .replaceAll('{title}', title)
          .replaceAll('{tags}', tagStr)
          .replaceAll('{page}', String(page))
          .replaceAll('{id}', illustId) + ext);
  };
  const makeTagsStr = (prev, cur, index, tagsArr) => {
      const tag = settings.tagLang === 'ja' ? cur.tag : cur.translation?.['en'] || cur.tag;
      if (index < tagsArr.length - 1) {
          return prev + tag + '_';
      }
      else {
          return prev + tag;
      }
  };
  function filterWorks(works, option) {
      const obj = {
          unavaliable: [],
          avaliable: [],
          invalid: []
      };
      works.forEach((work) => {
          if (!work.isBookmarkable) {
              obj.unavaliable.push(work.id);
          }
          else if (option.exclude_downloaded && pixivHistory.has(work.id)) {
              obj.invalid.push(work.id);
          }
          else if (!option[work.illustType]) {
              obj.invalid.push(work.id);
          }
          else {
              obj.avaliable.push(work.id);
          }
      });
      return obj;
  }
  async function getFollowLatestGenerator(filterOption, mode, page) {
      const MAX_PAGE = 34;
      const MAX_ILLUSTS_PER_PAGE = 60;
      let lastId;
      let total;
      let data;
      let cache;
      function findLastId(ids) {
          return Math.min(...ids.map((id) => Number(id)));
      }
      if (page === undefined) {
          data = await api.getFollowLatestWorks(1, mode);
          const ids = data.page.ids;
          total = ids.length;
          lastId = findLastId(ids);
          if (total === MAX_ILLUSTS_PER_PAGE) {
              const secondPageData = await api.getFollowLatestWorks(2, mode);
              const secondIds = secondPageData.page.ids;
              const secondLastId = findLastId(secondIds);
              if (secondLastId < lastId) {
                  lastId = secondLastId;
                  cache = secondPageData;
                  total += secondIds.length;
              }
          }
      }
      else {
          data = await api.getFollowLatestWorks(page, mode);
          total = data.page.ids.length;
      }
      async function* generateIds() {
          yield filterWorks(data.thumbnails.illust, filterOption);
          if (page === undefined) {
              if (total === MAX_ILLUSTS_PER_PAGE)
                  return;
              if (total < MAX_ILLUSTS_PER_PAGE * 2) {
                  yield filterWorks(cache.thumbnails.illust, filterOption);
                  return;
              }
              let currentPage = 3;
              while (currentPage <= MAX_PAGE) {
                  const data = await api.getFollowLatestWorks(currentPage, mode);
                  const ids = data.page.ids;
                  const pageLastId = findLastId(ids);
                  if (pageLastId >= lastId) {
                      yield filterWorks(cache.thumbnails.illust, filterOption);
                      break;
                  }
                  lastId = pageLastId;
                  total += ids.length;
                  yield { ...filterWorks(cache.thumbnails.illust, filterOption), total };
                  cache = data;
                  currentPage++;
                  await sleep(3000);
              }
          }
      }
      return {
          total,
          generator: generateIds()
      };
  }
  async function getChunksGenerator(userId, category, tag, rest, filterOption) {
      const OFFSET = 48;
      let requestUrl;
      if (category === 'bookmarks') {
          requestUrl = `https://www.pixiv.net/ajax/user/${userId}/illusts/bookmarks?tag=${tag}&offset=0&limit=${OFFSET}&rest=${rest}&lang=ja`;
      }
      else {
          requestUrl = `https://www.pixiv.net/ajax/user/${userId}/${category}/tag?tag=${tag}&offset=0&limit=${OFFSET}&lang=ja`;
      }
      let head = 0;
      const firstPageData = await api.getJson(requestUrl);
      const total = firstPageData.total;
      async function* generateIds() {
          yield filterWorks(firstPageData.works, filterOption);
          head += OFFSET;
          while (head < total) {
              const data = await api.getJson(requestUrl.replace('offset=0', 'offset=' + head));
              head += OFFSET;
              await sleep(3000);
              yield filterWorks(data.works, filterOption);
          }
      }
      return {
          total,
          generator: generateIds()
      };
  }
  async function getAllWorksGenerator(userId, filterOption) {
      const requestUrl = 'https://www.pixiv.net/ajax/user/' + userId + '/profile/all';
      const profile = await api.getJson(requestUrl);
      let illustIds = [];
      let mangaIds = [];
      if ((filterOption[IllustType.illusts] || filterOption[IllustType.ugoira]) &&
          typeof profile.illusts === 'object') {
          illustIds.push(...Object.keys(profile.illusts).reverse());
      }
      if (filterOption[IllustType.manga] && typeof profile.manga === 'object') {
          mangaIds.push(...Object.keys(profile.manga).reverse());
      }
      if (filterOption.exclude_downloaded) {
          illustIds = illustIds.filter((id) => !pixivHistory.has(id));
          mangaIds = mangaIds.filter((id) => !pixivHistory.has(id));
      }
      async function* generateIds() {
          const OFFSET = 48;
          const baseUrl = 'https://www.pixiv.net/ajax/user/' + userId + '/profile/illusts';
          let workCategory = 'illust';
          while (illustIds.length > 0) {
              let searchStr = '?';
              const chunk = illustIds.splice(0, OFFSET);
              searchStr +=
                  chunk.map((id) => 'ids[]=' + id).join('&') +
                      `&work_category=${workCategory}&is_first_page=0&lang=ja`;
              const data = await api.getJson(baseUrl + searchStr);
              await sleep(3000);
              yield filterWorks(Object.values(data.works), filterOption);
          }
          workCategory = 'manga';
          while (mangaIds.length > 0) {
              let searchStr = '?';
              const chunk = mangaIds.splice(0, OFFSET);
              searchStr +=
                  chunk.map((id) => 'ids[]=' + id).join('&') +
                      `&work_category=${workCategory}&is_first_page=0&lang=ja`;
              const data = await api.getJson(baseUrl + searchStr);
              await sleep(3000);
              yield filterWorks(Object.values(data.works), filterOption);
          }
      }
      return {
          total: illustIds.length + mangaIds.length,
          generator: generateIds()
      };
  }
  async function getArtworkData(illustId) {
      const htmlText = await api.getArtworkHtml(illustId);
      const preloadDataText = htmlText.match(regexp.preloadData);
      if (!preloadDataText)
          throw new Error('[Error]Fail to parse preload data.');
      const preloadData = JSON.parse(preloadDataText[1]);
      const illustData = preloadData.illust[illustId];
      const globalDataText = htmlText.match(regexp.globalData);
      if (!globalDataText)
          throw new Error('[Error]Fail to parse global data.');
      const globalData = JSON.parse(globalDataText[1]);
      let ugoiraMeta;
      if (illustData.illustType === IllustType.ugoira) {
          ugoiraMeta = await api.getJson('/ajax/illust/' + illustId + '/ugoira_meta');
      }
      return {
          illustData,
          globalData,
          ugoiraMeta
      };
  }
  function getDownloadSource(artworkData, seletedPage) {
      const { illustData, ugoiraMeta } = artworkData;
      const { illustType, userName, userId, illustTitle, illustId, tags, pageCount } = illustData;
      const pathInfo = {
          user: replaceInvalidChar(userName) || 'userId-' + userId,
          title: replaceInvalidChar(illustTitle) || 'illustId-' + illustId,
          tagStr: replaceInvalidChar(tags.tags.reduce(makeTagsStr, '')),
          illustId,
          userId,
          ext: '',
          page: 0
      };
      const metas = [];
      const taskId = illustId + '_' + Math.random().toString(36).slice(2);
      if (illustType === IllustType.illusts || illustType === IllustType.manga) {
          const firstImgSrc = illustData.urls.original;
          const srcPrefix = firstImgSrc.slice(0, firstImgSrc.indexOf('_') + 2);
          const extendName = firstImgSrc.slice(-4);
          pathInfo.ext = extendName;
          if (pageCount > 1 && seletedPage === undefined) {
              if (needBundle(illustType)) {
                  const path = getFilePath({ ...pathInfo, ext: '.zip', page: pageCount });
                  for (let i = 0; i < pageCount; i++) {
                      pathInfo.page = i;
                      metas.push({
                          kind: 'bundle',
                          taskId,
                          path,
                          src: srcPrefix + i + extendName,
                          filename: getFilePath(pathInfo, { needBundle: true }),
                          pageCount
                      });
                  }
              }
              else {
                  for (let i = 0; i < pageCount; i++) {
                      pathInfo.page = i;
                      metas.push({
                          taskId,
                          path: getFilePath(pathInfo),
                          src: srcPrefix + i + extendName
                      });
                  }
              }
          }
          else {
              let src = firstImgSrc;
              if (seletedPage !== undefined) {
                  src = srcPrefix + seletedPage + extendName;
                  pathInfo.page = seletedPage;
              }
              metas.push({
                  taskId,
                  path: getFilePath(pathInfo),
                  src
              });
          }
      }
      else if (illustType === IllustType.ugoira && ugoiraMeta) {
          pathInfo.ext = '.' + settings.ugoriaFormat;
          if (settings.ugoriaFormat !== 'zip') {
              metas.push({
                  kind: 'convert',
                  ugoiraMeta,
                  taskId,
                  src: ugoiraMeta.originalSrc,
                  path: getFilePath(pathInfo, { needConvert: true })
              });
          }
          else {
              metas.push({
                  taskId,
                  src: ugoiraMeta.originalSrc,
                  path: getFilePath(pathInfo)
              });
          }
      }
      return metas;
  }
  const parser = {
      getChunksGenerator,
      getAllWorksGenerator,
      getFollowLatestGenerator,
      getArtworkData,
      getDownloadSource
  };

  let converter;
  let downloader;
  async function initial() {
      converter = await createConverter
          .initialDeps()
          .then((createConverter) => createConverter.createInstance());
      downloader = createDownloader(converter);
  }

  function addBookmark(pdlBtn, illustId, token, tags) {
      if (!settings.addBookmark)
          return;
      api
          .addBookmark(illustId, token, settings.addBookmarkWithTags ? tags : [], settings.privateR18 && tags.includes('R-18')
          ? BookmarkRestrict.private
          : BookmarkRestrict.public)
          .then(() => {
          const bookmarkBtnRef = findBookmarkBtn(pdlBtn);
          if (!bookmarkBtnRef)
              return;
          switch (bookmarkBtnRef.kind) {
              case "main" : {
                  const pathBorder = bookmarkBtnRef.button.querySelector('svg g path');
                  pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)');
                  break;
              }
              case "sub" : {
                  const pathBorder = bookmarkBtnRef.button.querySelector('path');
                  pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)');
                  break;
              }
              case "rank" : {
                  bookmarkBtnRef.button.style.backgroundColor = 'rgb(255, 64, 96)';
                  break;
              }
          }
      })
          .catch((reason) => {
          console.error(reason.message);
      });
  }
  function findBookmarkBtn(pdlBtn) {
      const bookmarkBtnRef = {};
      if (pdlBtn.classList.contains('pdl-btn-sub')) {
          const btn = pdlBtn.parentElement?.nextElementSibling?.querySelector('button[type="button"]');
          if (btn) {
              bookmarkBtnRef.kind = "sub" ;
              bookmarkBtnRef.button = btn;
          }
          else {
              const btn = pdlBtn.parentElement?.querySelector('div._one-click-bookmark');
              if (btn) {
                  bookmarkBtnRef.kind = "rank" ;
                  bookmarkBtnRef.button = btn;
              }
          }
      }
      else if (pdlBtn.classList.contains('pdl-btn-main')) {
          const btn = pdlBtn.parentElement?.parentElement?.querySelector('button.gtm-main-bookmark');
          if (btn) {
              bookmarkBtnRef.kind = "main" ;
              bookmarkBtnRef.button = btn;
          }
      }
      else {
          return debugLog();
      }
      return bookmarkBtnRef;
  }

  function handleDownload(pdlBtn, illustId) {
      let pageCount, pageComplete = 0, shouldDownloadPage;
      const pageAttr = pdlBtn.getAttribute('should-download');
      if (pageAttr) {
          shouldDownloadPage = Number(pageAttr);
      }
      const onProgress = (progress = 0, type = null) => {
          if (pageCount > 1)
              return;
          progress = Math.floor(progress * 100);
          switch (type) {
              case null:
                  pdlBtn.style.setProperty('--pdl-progress', progress + '%');
              case 'gif':
              case 'webm':
              case 'webp':
                  pdlBtn.textContent = String(progress);
                  break;
              case 'zip':
                  pdlBtn.textContent = '';
                  break;
          }
      };
      const onLoad = function () {
          if (pageCount < 2)
              return;
          const progress = Math.floor((++pageComplete / pageCount) * 100);
          pdlBtn.textContent = String(progress);
          pdlBtn.style.setProperty('--pdl-progress', progress + '%');
      };
      pdlBtn.classList.add('pdl-progress');
      parser
          .getArtworkData(illustId)
          .then((artworkData) => {
          const { illustData, globalData } = artworkData;
          const { illustId, tags, bookmarkData } = illustData;
          if (!bookmarkData) {
              const { token } = globalData;
              const tagsArr = tags.tags.map((item) => item.tag);
              addBookmark(pdlBtn, illustId, token, tagsArr);
          }
          return parser.getDownloadSource(artworkData, shouldDownloadPage);
      })
          .then((sources) => {
          pageCount = sources.length;
          return downloader.add(sources, { onLoad, onProgress });
      })
          .then(() => {
          pixivHistory.add(illustId);
          pdlBtn.classList.remove('pdl-error');
          pdlBtn.classList.add('pdl-complete');
      })
          .catch((err) => {
          if (err)
              console.log(err);
          pdlBtn.classList.remove('pdl-complete');
          pdlBtn.classList.add('pdl-error');
      })
          .finally(() => {
          pdlBtn.innerHTML = '';
          pdlBtn.style.removeProperty('--pdl-progress');
          pdlBtn.classList.remove('pdl-progress');
      });
  }

  function createPdlBtn(attributes, textContent = '', { addEvent } = { addEvent: true }) {
      const ele = document.createElement('button');
      ele.textContent = textContent;
      if (!attributes)
          return ele;
      const { attrs, classList } = attributes;
      if (classList && classList.length > 0) {
          for (const cla of classList) {
              ele.classList.add(cla);
          }
      }
      if (attrs) {
          for (const key in attrs) {
              ele.setAttribute(key, attrs[key]);
          }
      }
      if (addEvent) {
          ele.addEventListener('click', (evt) => {
              evt.preventDefault();
              evt.stopPropagation();
              const ele = evt.currentTarget;
              if (!ele.classList.contains('pdl-progress')) {
                  handleDownload(ele, ele.getAttribute('pdl-id'));
              }
          });
      }
      return ele;
  }
  function getIllustId(node) {
      const isLinkToArtworksPage = regexp.artworksPage.exec(node.href);
      if (isLinkToArtworksPage) {
          if (node.getAttribute('data-gtm-value') ||
              node.classList.contains('gtm-illust-recommend-node-node') ||
              node.classList.contains('gtm-discover-user-recommend-node') ||
              node.classList.contains('work')) {
              return isLinkToArtworksPage[1];
          }
      }
      else {
          const isActivityThumb = regexp.activityHref.exec(node.href);
          if (isActivityThumb && node.classList.contains('work')) {
              return isActivityThumb[1];
          }
      }
      return '';
  }
  function createThunbnailsBtn(nodes) {
      const isBookmarkPage = regexp.bookmarkPage.test(location.pathname);
      nodes.forEach((e) => {
          if (e.childElementCount !== 0 &&
              !e.querySelector('.pdl-btn-sub')) {
              const illustId = getIllustId(e);
              if (illustId) {
                  const attrs = {
                      attrs: { 'pdl-id': illustId },
                      classList: ['pdl-btn', 'pdl-btn-sub']
                  };
                  if (pixivHistory.has(illustId))
                      attrs.classList.push('pdl-complete');
                  if (isBookmarkPage)
                      attrs.classList.push('pdl-btn-sub-bookmark');
                  e.appendChild(createPdlBtn(attrs));
              }
          }
      });
  }
  function compatPixivPreviewer(nodes) {
      const isPpSearchPage = regexp.ppSearchPage.test(location.pathname);
      if (!isPpSearchPage)
          return;
      nodes.forEach((node) => {
          const pdlEle = node.querySelector('.pdl-btn');
          if (!pdlEle)
              return false;
          pdlEle.remove();
      });
  }

  function createMainBtn(id) {
      if (document.querySelector('.pdl-btn-main'))
          return;
      const handleBar = document.querySelector('main section section');
      if (handleBar) {
          const pdlBtnWrap = handleBar.lastElementChild.cloneNode();
          const attrs = {
              attrs: { 'pdl-id': id },
              classList: ['pdl-btn', 'pdl-btn-main']
          };
          if (pixivHistory.has(id))
              attrs.classList.push('pdl-complete');
          pdlBtnWrap.appendChild(createPdlBtn(attrs));
          handleBar.appendChild(pdlBtnWrap);
      }
  }
  function createMultyWorksBtn(id) {
      const works = document.querySelectorAll("[role='presentation'] > a");
      if (works.length < 2)
          return;
      const containers = Array.from(works).map((node) => node.parentElement.parentElement);
      if (containers[0].querySelector('.pdl-btn'))
          return;
      containers.forEach((node, idx) => {
          const wrapper = document.createElement('div');
          wrapper.classList.add('pdl-wrap-artworks');
          const attrs = {
              attrs: { 'pdl-id': id, 'should-download': String(idx) },
              classList: ['pdl-btn', 'pdl-btn-sub', 'artworks']
          };
          wrapper.appendChild(createPdlBtn(attrs));
          node.appendChild(wrapper);
      });
  }
  const createPresentationBtn = (() => {
      let observer, btn;
      function cb(mutationList) {
          const newImg = mutationList[1]['addedNodes'][0];
          const [pageNum] = regexp.originSrcPageNum.exec(newImg.src) ?? [];
          if (!pageNum)
              throw new Error('[Error]Invalid Image Element.');
          btn?.setAttribute('should-download', String(pageNum));
      }
      return (id) => {
          const containers = document.querySelector("body > [role='presentation'] > div");
          if (!containers) {
              if (observer) {
                  observer.disconnect();
                  observer = null;
                  btn = null;
              }
              return;
          }
          if (containers.querySelector('.pdl-btn'))
              return;
          const img = containers.querySelector('img');
          if (!img)
              return;
          const isOriginImg = regexp.originSrcPageNum.exec(img.src);
          if (!isOriginImg)
              return;
          const [pageNum] = isOriginImg;
          const attrs = {
              attrs: { 'pdl-id': id, 'should-download': pageNum },
              classList: ['pdl-btn', 'pdl-btn-sub', 'presentation']
          };
          btn = createPdlBtn(attrs);
          containers.appendChild(btn);
          if (!img.parentElement)
              return;
          observer = new MutationObserver(cb);
          observer.observe(img.parentElement, { childList: true, subtree: true });
      };
  })();
  function createPreviewModalBtn() {
      const illustModalBtn = document.querySelectorAll('.gtm-manga-viewer-preview-modal-open');
      const mangaModalBtn = document.querySelectorAll('.gtm-manga-viewer-open-preview');
      const mangaViewerModalBtn = document.querySelectorAll('.gtm-manga-viewer-close-icon')?.[1];
      if (!illustModalBtn.length && !mangaModalBtn.length)
          return;
      const btns = [...illustModalBtn, ...mangaModalBtn];
      if (mangaViewerModalBtn)
          btns.push(mangaViewerModalBtn);
      btns.forEach((node) => {
          node.addEventListener('click', handleModalClick);
      });
  }
  function handleModalClick() {
      const timer = setInterval(() => {
          const ulList = document.querySelectorAll('ul');
          const previewList = ulList[ulList.length - 1];
          if (getComputedStyle(previewList).display !== 'grid')
              return;
          clearInterval(timer);
          const [, id] = regexp.artworksPage.exec(location.pathname) ?? [];
          previewList.childNodes.forEach((node, idx) => {
              node.style.position = 'relative';
              const attrs = {
                  attrs: { 'pdl-id': id, 'should-download': String(idx) },
                  classList: ['pdl-btn', 'pdl-btn-sub']
              };
              node.appendChild(createPdlBtn(attrs));
          });
      }, 300);
  }

  function changeDlbarDisplay() {
      document.querySelectorAll('.pdl-dlbar .pdl-btn-all').forEach((ele) => {
          ele.classList.toggle('pdl-hide');
      });
      document.querySelector('.pdl-dlbar .pdl-stop')?.classList.toggle('pdl-hide');
      document.querySelectorAll('.pdl-tag').forEach((ele) => {
          ele.classList.toggle('pdl-tag-hide');
      });
      document.querySelector('.pdl-filter-wrap')?.classList.toggle('unavailable');
  }
  let isDownloading = false;
  const dlBarRef = {
      filter: {
          exclude_downloaded: undefined,
          [IllustType.illusts]: undefined,
          [IllustType.manga]: undefined,
          [IllustType.ugoira]: undefined
      },
      statusBar: undefined,
      abortBtn: undefined
  };
  function getFilterOption() {
      return {
          exclude_downloaded: dlBarRef.filter.exclude_downloaded?.checked ??
              defaultSettings.filter.exclude_downloaded,
          [IllustType.illusts]: dlBarRef.filter[IllustType.illusts]?.checked ??
              defaultSettings.filter[IllustType.illusts],
          [IllustType.manga]: dlBarRef.filter[IllustType.manga]?.checked ??
              defaultSettings.filter[IllustType.manga],
          [IllustType.ugoira]: dlBarRef.filter[IllustType.ugoira]?.checked ??
              defaultSettings.filter[IllustType.ugoira]
      };
  }
  function updateStatus(str) {
      dlBarRef.statusBar && (dlBarRef.statusBar.textContent = str);
  }
  function onProgressCB(progressData) {
      if (typeof progressData === 'string') {
          updateStatus(progressData);
      }
      else {
          debugLog('update progress by', progressData.illustId);
          updateStatus(`Downloading: ${progressData.completed} / ${progressData.avaliable}`);
      }
  }
  async function useDownload(chunksGenerators) {
      if (!dlBarRef.abortBtn)
          return;
      isDownloading = true;
      changeDlbarDisplay();
      let total = 0;
      let failedResult;
      const idsGenerators = [];
      try {
          if (chunksGenerators instanceof Array) {
              await Promise.all(chunksGenerators).then((chunksGenerator) => {
                  chunksGenerator.forEach((val) => {
                      total += val.total;
                      idsGenerators.push(val.generator);
                  });
              });
          }
          else {
              await chunksGenerators.then((val) => {
                  total = val.total;
                  idsGenerators.push(val.generator);
              });
          }
      }
      catch (error) {
          console.error(error);
          updateStatus('Network error, see console.');
          changeDlbarDisplay();
          isDownloading = false;
          return;
      }
      try {
          if (total === 0) {
              throw 'No works.';
          }
          debugLog('[Info]Total:', total);
          const { failed, unavaliable } = await downloadByIds(total, idsGenerators, dlBarRef.abortBtn, onProgressCB);
          if (failed.length || unavaliable.length) {
              updateStatus(`Failed: ${failed.length + unavaliable.length}. See console.`);
              console.log('[Pixiv Downloader]Failed: ', failed.join(', '));
              console.log('[Pixiv Downloader]Unavaliable: ', unavaliable.join(', '));
              if (failed.length)
                  failedResult = failed;
          }
          else {
              updateStatus('Complete');
          }
      }
      catch (error) {
          updateStatus(error);
      }
      changeDlbarDisplay();
      isDownloading = false;
      return failedResult;
  }
  async function downloadByIds(total, idsGenerators, abortBtn, onProgress) {
      let resolve;
      let reject;
      const done = new Promise((r, j) => {
          resolve = r;
          reject = j;
      });
      let wakeFn;
      let completed = 0;
      const failed = [], unavaliable = [], invalid = [];
      let isCanceled = false;
      const tasks = [];
      let tooManyRequests = false;
      abortBtn.onclick = () => {
          isCanceled = true;
          abortBtn.onclick = null;
          reject(`Stopped. ${completed} / ${total - failed.length - unavaliable.length - invalid.length}`);
          wakeFn && wakeFn();
          if (tasks.length) {
              downloader.del(tasks);
              converter.del(tasks);
              compressor.remove(tasks);
              tasks.length = 0;
          }
      };
      const afterEach = (illustId) => {
          const avaliable = total - failed.length - unavaliable.length - invalid.length;
          onProgress({
              illustId,
              avaliable,
              completed
          });
          if (completed === avaliable) {
              resolve({ failed, unavaliable });
          }
      };
      onProgress('Downloading...');
      try {
          for (const idsGenerator of idsGenerators) {
              if (isCanceled)
                  return done;
              for await (const ids of idsGenerator) {
                  debugLog('[Info]ids:', ids);
                  if (isCanceled)
                      return done;
                  if (ids.unavaliable.length) {
                      unavaliable.push(...ids.unavaliable);
                  }
                  if (ids.invalid.length) {
                      invalid.push(...ids.invalid);
                  }
                  if (typeof ids.total === 'number' && !Number.isNaN(ids.total)) {
                      total = ids.total;
                  }
                  if (ids.avaliable.length) {
                      for (const id of ids.avaliable) {
                          if (isCanceled)
                              return done;
                          if (tooManyRequests) {
                              onProgress('Too many requests, wait 30s');
                              const { wake, sleep } = wakeableSleep(30000);
                              wakeFn = wake;
                              await sleep;
                              tooManyRequests = false;
                              if (isCanceled)
                                  return done;
                              onProgress('Downloading...');
                          }
                          parser
                              .getArtworkData(id)
                              .then(parser.getDownloadSource)
                              .then((sources) => {
                              if (isCanceled) {
                                  throw '[Warning]Download stop manually: ' + id;
                              }
                              tasks.push(sources[0].taskId);
                              return downloader.add(sources);
                          })
                              .then((taskId) => {
                              pixivHistory.add(id);
                              if (!isCanceled) {
                                  tasks.splice(tasks.indexOf(taskId), 1);
                                  completed++;
                                  afterEach(id);
                              }
                          }, (reason) => {
                              if (!isCanceled) {
                                  if (reason.message && reason.message === '429')
                                      tooManyRequests = true;
                                  if (reason.message &&
                                      reason.message === '[Error]Fail to parse preload data.') {
                                      unavaliable.push(id);
                                  }
                                  else {
                                      failed.push(id);
                                  }
                                  afterEach(id);
                              }
                          });
                          await sleep(1000);
                      }
                  }
                  else {
                      afterEach();
                  }
              }
          }
      }
      catch (error) {
          console.error(error);
          reject('Network Error, see console');
          abortBtn.click();
      }
      return done;
  }
  function downloadWorks(evt) {
      evt.preventDefault();
      evt.stopPropagation();
      if (isDownloading)
          return;
      const btn = evt.target;
      const userId = btn.getAttribute('pdl-userid');
      const filterOption = getFilterOption();
      const ids = parser.getAllWorksGenerator(userId, filterOption);
      useDownload(ids).then((failed) => {
          if (failed instanceof Array && failed.length) {
              const gen = async function* () {
                  yield {
                      avaliable: failed,
                      unavaliable: [],
                      invalid: []
                  };
              };
              useDownload(Promise.resolve({ total: failed.length, generator: gen() }));
          }
      });
  }
  async function downloadBookmarksOrTags(evt) {
      evt.preventDefault();
      evt.stopPropagation();
      if (isDownloading)
          return;
      const btn = evt.target;
      const userId = btn.getAttribute('pdl-userid');
      const category = btn.getAttribute('category');
      const tag = btn.getAttribute('tag') || '';
      const rest = (btn.getAttribute('rest') || 'show');
      const filterOption = getFilterOption();
      let idsGenerators;
      if (rest === 'all') {
          const idsShowPromise = parser.getChunksGenerator(userId, 'bookmarks', '', 'show', filterOption);
          const idsHidePromise = parser.getChunksGenerator(userId, 'bookmarks', '', 'hide', filterOption);
          idsGenerators = [idsShowPromise, idsHidePromise];
      }
      else {
          idsGenerators = parser.getChunksGenerator(userId, category, tag, rest, filterOption);
      }
      useDownload(idsGenerators).then((failed) => {
          if (failed instanceof Array && failed.length) {
              const gen = async function* () {
                  yield {
                      avaliable: failed,
                      unavaliable: [],
                      invalid: []
                  };
              };
              useDownload(Promise.resolve({ total: failed.length, generator: gen() }));
          }
      });
  }
  function downloadFollowLatest(evt) {
      evt.preventDefault();
      evt.stopPropagation();
      if (isDownloading)
          return;
      const btn = evt.target;
      let idsGenerators;
      const mode = location.pathname.includes('r18') ? 'r18' : 'all';
      const filterOption = getFilterOption();
      if (btn.classList.contains('pdl-dl-all')) {
          idsGenerators = parser.getFollowLatestGenerator(filterOption, mode);
      }
      else {
          const params = new URLSearchParams(location.search);
          const page = Number(params.get('p')) || 1;
          idsGenerators = parser.getFollowLatestGenerator(filterOption, mode, page);
      }
      useDownload(idsGenerators).then((failed) => {
          if (failed instanceof Array && failed.length) {
              const gen = async function* () {
                  yield {
                      avaliable: failed,
                      unavaliable: [],
                      invalid: []
                  };
              };
              useDownload(Promise.resolve({ total: failed.length, generator: gen() }));
          }
      });
  }

  function createFilterEl(id, filterType, text) {
      const checkbox = document.createElement('input');
      const label = document.createElement('label');
      checkbox.id = id;
      checkbox.type = 'checkbox';
      checkbox.classList.add('pdl-checkbox');
      checkbox.setAttribute('category', String(filterType));
      checkbox.checked = settings.filter[filterType];
      label.setAttribute('for', id);
      label.setAttribute('category', String(filterType));
      label.textContent = text;
      checkbox.addEventListener('change', (evt) => {
          const checkbox = evt.currentTarget;
          const category = checkbox.getAttribute('category');
          upgradeSettings('filter', {
              ...settings.filter,
              [category]: checkbox.checked
          });
      });
      dlBarRef.filter[filterType] = checkbox;
      const wrap = document.createElement('div');
      wrap.classList.add('pdl-filter');
      wrap.appendChild(checkbox);
      wrap.appendChild(label);
      return wrap;
  }
  function createFilter() {
      const wrapper = document.createElement('div');
      wrapper.classList.add('pdl-filter-wrap');
      wrapper.appendChild(createFilterEl('pdl-filter-exclude_downloaded', 'exclude_downloaded', i18n('dlbar_filter_exclude_downloaded')));
      wrapper.appendChild(createFilterEl('pdl-filter-illusts', IllustType.illusts, i18n('dlbar_filter_illusts')));
      wrapper.appendChild(createFilterEl('pdl-filter-manga', IllustType.manga, i18n('dlbar_filter_manga')));
      wrapper.appendChild(createFilterEl('pdl-filter-ugoria', IllustType.ugoira, i18n('dlbar_filter_ugoria')));
      return wrapper;
  }
  function createDownloadBar(userId) {
      const nav = document.querySelector('nav[class~="sc-192ftwf-0"]');
      if (!nav)
          return;
      const dlBtn = nav.querySelector('.pdl-btn-all');
      if (dlBtn) {
          if (dlBtn.getAttribute('pdl-userid') === userId)
              return;
          removeDownloadBar();
      }
      const dlBar = document.createElement('div');
      dlBar.classList.add('pdl-dlbar');
      const statusBar = document.createElement('div');
      statusBar.classList.add('pdl-dlbar-status_bar');
      dlBarRef.statusBar = dlBar.appendChild(statusBar);
      const baseClasses = nav.querySelector('a:not([aria-current])').classList;
      dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
          attrs: { 'pdl-userId': userId },
          classList: [...baseClasses, 'pdl-stop', 'pdl-hide']
      }, i18n('stop'), { addEvent: false }));
      if (userId !== getSelfId()) {
          const hasWorks = ["a[href$='illustrations']", "a[href$='manga']"].some((selector) => !!nav.querySelector(selector));
          if (hasWorks) {
              const el = createPdlBtn({
                  attrs: { 'pdl-userid': userId },
                  classList: [...baseClasses, 'pdl-btn-all']
              }, i18n('dlbar_category_works'), { addEvent: false });
              el.addEventListener('click', downloadWorks);
              dlBar.appendChild(el);
          }
          if (nav.querySelector("a[href*='bookmarks']")) {
              const el = createPdlBtn({
                  attrs: { 'pdl-userid': userId, category: 'bookmarks' },
                  classList: [...baseClasses, 'pdl-btn-all']
              }, i18n('bookmarks'), { addEvent: false });
              el.addEventListener('click', downloadBookmarksOrTags);
              dlBar.appendChild(el);
          }
      }
      else {
          if (nav.querySelector("a[href*='bookmarks']")) {
              dlBar.appendChild(createPdlBtn({
                  attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'all' },
                  classList: [...baseClasses, 'pdl-btn-all']
              }, i18n('bookmarks'), { addEvent: false }));
              dlBar.appendChild(createPdlBtn({
                  attrs: {
                      'pdl-userid': userId,
                      category: 'bookmarks',
                      rest: 'show'
                  },
                  classList: [...baseClasses, 'pdl-btn-all']
              }, i18n('bookmarks_public'), { addEvent: false }));
              dlBar.appendChild(createPdlBtn({
                  attrs: {
                      'pdl-userid': userId,
                      category: 'bookmarks',
                      rest: 'hide'
                  },
                  classList: [...baseClasses, 'pdl-btn-all']
              }, i18n('bookmarks_private'), { addEvent: false }));
              dlBar
                  .querySelectorAll('.pdl-btn-all')
                  .forEach((node) => {
                  node.addEventListener('click', downloadBookmarksOrTags);
              });
          }
      }
      const filter = createFilter();
      nav.parentElement.insertBefore(filter, nav);
      nav.appendChild(dlBar);
  }
  function removeDownloadBar() {
      const dlBarWrap = document.querySelector('.pdl-dlbar');
      if (dlBarWrap) {
          dlBarWrap.remove();
          document.querySelector('.pdl-filter-wrap')?.remove();
      }
  }
  function createTagsBtn(userId, category) {
      const tagsEles = document.querySelectorAll('section> div:nth-child(2) > div > div');
      if (!tagsEles.length)
          return;
      let cate;
      if (category === 'illustrations' || category === 'artworks') {
          cate = 'illusts';
      }
      else {
          cate = category;
      }
      let rest = 'show';
      if (userId === getSelfId() &&
          category === 'bookmarks' &&
          location.search.includes('rest=hide'))
          rest = 'hide';
      tagsEles.forEach((ele) => {
          const tagBtn = ele.querySelector('.pdl-btn');
          if (tagBtn) {
              const btnRest = tagBtn.getAttribute('rest');
              if (rest !== btnRest)
                  tagBtn.setAttribute('rest', rest);
              return;
          }
          let tag;
          const tagLink = ele.querySelector('a');
          if (!tagLink)
              return;
          if (tagLink.getAttribute('status') !== 'active') {
              if (rest === 'hide') {
                  tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1, tagLink.href.lastIndexOf('?'));
              }
              else {
                  tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1);
              }
          }
          else {
              const tagTextEles = ele.querySelectorAll('div[title]');
              if (!tagTextEles.length)
                  return console.log('[Info]No Tags Element found.');
              tag = tagTextEles[tagTextEles.length - 1].getAttribute('title').slice(1);
          }
          const attrs = {
              attrs: { 'pdl-userId': userId, category: cate, tag, rest },
              classList: ['pdl-btn', 'pdl-tag']
          };
          if (isDownloading)
              attrs.classList.push('pdl-tag-hide');
          const dlBtn = createPdlBtn(attrs, '', { addEvent: false });
          if (!(tagLink.href.includes('bookmarks') &&
              tagLink.getAttribute('status') !== 'active')) {
              dlBtn.style.backgroundColor = tagLink.getAttribute('color') + '80';
          }
          dlBtn.addEventListener('click', downloadBookmarksOrTags);
          ele.appendChild(dlBtn);
      });
      let modalTagsEles;
      let modal;
      if (category === 'bookmarks') {
          modal = document.querySelector('div[role="presentation"]');
          if (!modal)
              return;
          modalTagsEles = modal.querySelectorAll('a');
      }
      else {
          const charcoalTokens = document.querySelectorAll('.charcoal-token');
          modal = charcoalTokens[charcoalTokens.length - 1];
          if (!modal)
              return;
          modalTagsEles = modal.querySelectorAll('a');
      }
      if (!regexp.userPageTags.exec(modalTagsEles[0]?.href))
          return;
      modalTagsEles.forEach((ele) => {
          if (ele.querySelector('.pdl-btn'))
              return;
          let tag;
          if (rest === 'hide') {
              tag = ele.href.slice(ele.href.lastIndexOf('/') + 1, ele.href.lastIndexOf('?'));
          }
          else {
              tag = ele.href.slice(ele.href.lastIndexOf('/') + 1);
          }
          const attrs = {
              attrs: { 'pdl-userId': userId, category: cate, tag, rest },
              classList: ['pdl-btn', 'pdl-modal-tag']
          };
          if (isDownloading)
              attrs.classList.push('pdl-tag-hide');
          const dlBtn = createPdlBtn(attrs, '', { addEvent: false });
          dlBtn.addEventListener('click', (evt) => {
              modal.querySelector('svg').parentElement.click();
              downloadBookmarksOrTags(evt);
          });
          ele.appendChild(dlBtn);
      });
  }

  function createFollowLatestDownloadBar() {
      const prevDlBtn = document.querySelector('.pdl-btn-all');
      if (prevDlBtn) {
          const prevDlAllBtn = document.querySelector('.pdl-dl-all');
          if (location.pathname.includes('r18')) {
              prevDlBtn.textContent !==
                  i18n('dlbar.follow_latest.category_r18.single') &&
                  (prevDlBtn.textContent = i18n('dlbar.follow_latest.category_r18.single'));
              prevDlAllBtn.textContent !==
                  i18n('dlbar.follow_latest.category_r18.all') &&
                  (prevDlAllBtn.textContent = i18n('dlbar.follow_latest.category_r18.all'));
          }
          else {
              prevDlBtn.textContent !==
                  i18n('dlbar.follow_latest.category_all.single') &&
                  (prevDlBtn.textContent = i18n('dlbar.follow_latest.category_all.single'));
              prevDlAllBtn.textContent !==
                  i18n('dlbar.follow_latest.category_all.all') &&
                  (prevDlAllBtn.textContent = i18n('dlbar.follow_latest.category_all.all'));
          }
          return;
      }
      const nav = document.querySelector('nav');
      if (!nav || nav.parentElement.childElementCount === 1)
          return;
      const navBar = nav.parentElement;
      const modeSwitch = nav.nextElementSibling;
      const filter = createFilter();
      navBar.parentElement.insertBefore(filter, navBar);
      const dlBar = document.createElement('div');
      dlBar.classList.add('pdl-dlbar');
      dlBar.classList.add('pdl-dlbar-follow_latest');
      const statusBar = document.createElement('div');
      statusBar.classList.add('pdl-dlbar-status_bar');
      dlBarRef.statusBar = dlBar.appendChild(statusBar);
      const baseClasses = nav.querySelector('a:not([aria-current])').classList;
      dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
          attrs: { 'pdl-userid': '' },
          classList: [...baseClasses, 'pdl-stop', 'pdl-hide']
      }, i18n('stop'), { addEvent: false }));
      const dlBtn = createPdlBtn({
          attrs: { 'pdl-userid': '' },
          classList: [...baseClasses, 'pdl-btn-all']
      },
      i18n('dlbar_category_works'), { addEvent: false });
      dlBtn.addEventListener('click', downloadFollowLatest);
      dlBar.appendChild(dlBtn);
      const dlAllBtn = createPdlBtn({
          attrs: { 'pdl-userid': '' },
          classList: [...baseClasses, 'pdl-btn-all', 'pdl-dl-all']
      },
      i18n('dlbar_category_works'), { addEvent: false });
      dlAllBtn.addEventListener('click', downloadFollowLatest);
      dlBar.appendChild(dlAllBtn);
      navBar.insertBefore(dlBar, modeSwitch);
  }

  let firstRun = true;
  function observerCallback(records) {
      const addedNodes = [];
      records.forEach((record) => {
          if (!record.addedNodes.length)
              return;
          record.addedNodes.forEach((node) => {
              if (node.nodeType === Node.ELEMENT_NODE &&
                  node.tagName !== 'BUTTON' &&
                  node.tagName !== 'IMG') {
                  addedNodes.push(node);
              }
          });
      });
      if (!addedNodes.length) {
          return;
      }
      if (firstRun) {
          createThunbnailsBtn(document.querySelectorAll('a'));
          firstRun = false;
      }
      else {
          compatPixivPreviewer(addedNodes);
          const thunmnails = addedNodes.reduce((prev, current) => {
              return prev.concat(Array.from(current.querySelectorAll('a')));
          }, []);
          createThunbnailsBtn(thunmnails);
      }
      const isArtworksPage = regexp.artworksPage.exec(location.pathname);
      const isUserPage = regexp.userPage.exec(location.pathname);
      const isTagsPage = regexp.userPageTags.exec(location.pathname);
      if (isArtworksPage) {
          const id = isArtworksPage[1];
          createMainBtn(id);
          createMultyWorksBtn(id);
          createPresentationBtn(id);
          createPreviewModalBtn();
      }
      else if (isUserPage) {
          createDownloadBar(isUserPage[1]);
          if (isTagsPage) {
              createTagsBtn(isUserPage[1], isTagsPage[1]);
          }
      }
      else if (regexp.followLatest.test(location.pathname)) {
          createFollowLatestDownloadBar();
      }
      else {
          removeDownloadBar();
      }
  }

  function createModal({ header = '', content, footer = '' }, option) {
      const modal = document.createElement('div');
      const dialog = document.createElement('div');
      modal.classList.add('pdl-modal');
      dialog.classList.add('pdl-dialog');
      if (option.closeOnClickModal) {
          dialog.onclick = (e) => {
              e.stopPropagation();
          };
          modal.onclick = () => {
              modal.remove();
          };
      }
      dialog.innerHTML = `  <header class="pdl-dialog-header">${header}</header>
  <div class="pdl-dialog-content">${content}</div>
  <footer class="pdl-dialog-footer">${footer}</footer>`;
      const closeBtn = document.createElement('button');
      closeBtn.classList.add('pdl-dialog-close');
      closeBtn.onclick = () => {
          modal.remove();
      };
      dialog.insertBefore(closeBtn, dialog.firstChild);
      modal.appendChild(dialog);
      return modal;
  }

  function showUpgradeMsg() {
      document.body.appendChild(createModal({
          header: modalHtml.upgradeMsgTitle,
          content: modalHtml.upgradeMsgContent,
          footer: modalHtml.modalCreditFooter + modalHtml.modalFeedback
      }, { closeOnClickModal: false }));
  }

  function createTabsPart({ tabName, paneHtml }) {
      const tab = document.createElement('div');
      tab.classList.add('pdl-tab-item');
      tab.textContent = tabName;
      const pane = document.createElement('div');
      pane.classList.add('pdl-tab-pane');
      pane.innerHTML = paneHtml;
      return {
          tab,
          pane
      };
  }
  function createTabFilename({ tabName, paneHtml }) {
      const { tab, pane } = createTabsPart({ tabName, paneHtml });
      const folder = pane.querySelector('#pdlfolder');
      const folderReset = pane.querySelector('#pdl-filename-folder-reset');
      const folderUpdate = pane.querySelector('#pdl-filename-folder-confirm');
      const filename = pane.querySelector('#pdlfilename');
      const filenameReset = pane.querySelector('#pdl-filename-filename-reset');
      const filenameUpdate = pane.querySelector('#pdl-filename-filename-confirm');
      if (!folder || !filename)
          throw new Error('[Error]Can not create modal.');
      filename.value = settings.filenamePattern;
      if (!env.isSupportSubpath) {
          folder.setAttribute('disabled', '');
          folder.value = '';
      }
      else {
          folder.value = settings.folderPattern;
      }
      folder.placeholder = env.isViolentmonkey
          ? i18n('folder_vm_tips')
          : !env.isSupportSubpath
              ? i18n('folder_api_tips')
              : i18n('folder_tips');
      folder.addEventListener('input', () => {
          folderReset?.removeAttribute('disabled');
          folderUpdate?.removeAttribute('disabled');
      });
      folderReset?.addEventListener('click', () => {
          folder.value = settings.folderPattern;
          folderReset?.setAttribute('disabled', '');
          folderUpdate?.setAttribute('disabled', '');
      });
      folderUpdate?.addEventListener('click', () => {
          const folderPattern = folder.value
              .split('/')
              .map((path) => path
              .trim()
              .replace(/^\.+|\.+$|[\u200b-\u200f\uFEFF\u202a-\u202e\\:*?"|<>]/g, ''))
              .filter((path) => !!path)
              .join('/') ?? '';
          upgradeSettings('folderPattern', folderPattern);
          folder.value = folderPattern;
          folderReset?.setAttribute('disabled', '');
          folderUpdate?.setAttribute('disabled', '');
      });
      filename.addEventListener('input', () => {
          filenameReset?.removeAttribute('disabled');
          filenameUpdate?.removeAttribute('disabled');
      });
      filenameReset?.addEventListener('click', () => {
          filename.value = settings.filenamePattern;
          filenameReset?.setAttribute('disabled', '');
          filenameUpdate?.setAttribute('disabled', '');
      });
      filenameUpdate?.addEventListener('click', () => {
          const filenamePattern = filename.value
              .trim()
              .replace(/^\.+|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, '') ??
              '';
          if (filenamePattern === '')
              return filenameReset?.click();
          upgradeSettings('filenamePattern', filenamePattern);
          filename.value = filenamePattern;
          filenameReset?.setAttribute('disabled', '');
          filenameUpdate?.setAttribute('disabled', '');
      });
      pane
          .querySelectorAll('.tags-content .tags-item input')
          .forEach((input) => {
          if (settings.tagLang === input.value)
              input.checked = true;
          input.addEventListener('change', (ev) => {
              upgradeSettings('tagLang', ev.currentTarget.value);
          });
      });
      return {
          tab,
          pane
      };
  }
  function createTabUgoria({ tabName, paneHtml }) {
      const { tab, pane } = createTabsPart({ tabName, paneHtml });
      pane
          .querySelectorAll('.pdl-ugoria-format-item input[type="radio"]')
          .forEach((el) => {
          if (settings.ugoriaFormat === el.value)
              el.checked = true;
          el.addEventListener('change', (ev) => {
              upgradeSettings('ugoriaFormat', ev.currentTarget.value);
          });
      });
      return {
          tab,
          pane
      };
  }
  function createTabHistory({ tabName, paneHtml }) {
      const { tab, pane } = createTabsPart({ tabName, paneHtml });
      const file = pane.querySelector('#pdl-import');
      file?.addEventListener('change', (evt) => {
          const file = evt.currentTarget.files?.[0];
          if (!file)
              return;
          if (file.type === 'text/plain') {
              const reader = new FileReader();
              reader.readAsText(file);
              reader.onload = (readEvt) => {
                  const text = readEvt.target?.result;
                  try {
                      if (typeof text !== 'string')
                          throw new Error('Invalid file');
                      const history = JSON.parse(text);
                      if (!(history instanceof Array))
                          throw new Error('Invalid file');
                      if (history.length &&
                          !history.every((id) => typeof id === 'string')) {
                          throw new Error('Invalid id type');
                      }
                      pixivHistory.saveHistory(history);
                      location.reload();
                  }
                  catch (error) {
                      alert(error.message);
                  }
              };
          }
          else {
              alert('Invalid file');
          }
      });
      const importBtn = pane.querySelector('#pdl-import-btn');
      importBtn?.addEventListener('click', () => file?.click());
      const exportBtn = pane.querySelector('#pdl-export');
      exportBtn?.addEventListener('click', () => {
          const dlEle = document.createElement('a');
          const history = JSON.stringify(pixivHistory.getAll());
          dlEle.href = URL.createObjectURL(new Blob([history], { type: 'text/plain' }));
          dlEle.download = 'Pixiv Downloader ' + new Date().toLocaleString() + '.txt';
          dlEle.click();
          URL.revokeObjectURL(dlEle.href);
      });
      const clearBtn = pane.querySelector('#pdl-clear-history');
      clearBtn?.addEventListener('click', () => pixivHistory.clearHistory());
      return {
          tab,
          pane
      };
  }
  function createTabOthers({ tabName, paneHtml }) {
      const { tab, pane } = createTabsPart({ tabName, paneHtml });
      [
          { selector: '#pdl-options-bundle-illusts', settingKey: 'bundleIllusts' },
          { selector: '#pdl-options-bundle-manga', settingKey: 'bundleManga' },
          { selector: '#pdl-options-add-bookmark', settingKey: 'addBookmark' },
          {
              selector: '#pdl-options-add-bookmark-tags',
              settingKey: 'addBookmarkWithTags'
          },
          {
              selector: '#pdl-options-add-bookmark-private-r18',
              settingKey: 'privateR18'
          }
      ].forEach(({ selector, settingKey }) => {
          const optionEl = pane.querySelector(selector);
          optionEl && (optionEl.checked = settings[settingKey]);
          optionEl?.addEventListener('change', (ev) => {
              upgradeSettings(settingKey, ev.currentTarget.checked);
          });
      });
      return {
          tab,
          pane
      };
  }
  function showSettings() {
      if (document.querySelector('.pdl-modal'))
          return;
      const modal = createModal({
          content: modalHtml.modalSettingsContent
      }, { closeOnClickModal: false });
      const tabsNav = modal.querySelector('.pdl-tabs-nav');
      const tabContent = modal.querySelector('.pdl-tabs-content');
      [
          createTabFilename({
              tabName: i18n('tabs_nav_filename'),
              paneHtml: modalHtml.modalSettingFilename
          }),
          createTabUgoria({
              tabName: i18n('tabs_nav_ugoria'),
              paneHtml: modalHtml.modalSettingUgoria
          }),
          createTabHistory({
              tabName: i18n('tabs_nav_history'),
              paneHtml: modalHtml.modalSettingHistory
          }),
          createTabOthers({
              tabName: i18n('tabs_nav_others'),
              paneHtml: modalHtml.modalSettingOthers
          }),
          createTabsPart({
              tabName: i18n('tabs_nav_donate'),
              paneHtml: modalHtml.modalSettingDonate
          })
      ].forEach(({ tab, pane }) => {
          tabsNav.appendChild(tab);
          tabContent.appendChild(pane);
      });
      const panes = Array.from(tabContent.querySelectorAll('.pdl-tab-pane'));
      panes.forEach((el) => {
          el.style.setProperty('display', 'none');
      });
      const activeBar = tabsNav.querySelector('.pdl-tabs__active-bar');
      const tabs = Array.from(modal.querySelectorAll('.pdl-tabs-nav .pdl-tab-item'));
      tabs.forEach((el) => {
          el.addEventListener('click', (ev) => {
              const tab = ev.currentTarget;
              if (!tab)
                  return;
              tabs.forEach((tab) => tab.classList.remove('active'));
              tab.classList.add('active');
              activeBar.style.width = getComputedStyle(tab).width;
              activeBar.style.transform = `translateX(${tab.offsetLeft + parseFloat(getComputedStyle(tab).paddingLeft)}px)`;
              panes.forEach((pane) => pane.style.setProperty('display', 'none'));
              panes[tabs.indexOf(tab)].style.removeProperty('display');
          });
      });
      tabs[0].classList.add('active');
      panes[0].style.removeProperty('display');
      document.body.appendChild(modal);
      activeBar.style.width = getComputedStyle(tabs[0]).width;
      activeBar.style.transform = `translateX(${tabs[0].offsetLeft + parseFloat(getComputedStyle(tabs[0]).paddingLeft)}px)`;
  }

  pixivHistory.updateHistory();
  GM_registerMenuCommand(i18n('gm_menu_setting'), showSettings, 's');
  initial()
      .then(() => {
      if (settings.showMsg) {
          showUpgradeMsg();
          upgradeSettings('showMsg', false);
      }
      new MutationObserver(observerCallback).observe(document.body, {
          childList: true,
          subtree: true
      });
      document.addEventListener('keydown', (e) => {
          if (e.ctrlKey && e.key === 'q') {
              const pdlMainBtn = document.querySelector('.pdl-btn-main');
              if (pdlMainBtn) {
                  e.preventDefault();
                  if (!e.repeat) {
                      pdlMainBtn.dispatchEvent(new MouseEvent('click'));
                  }
              }
          }
      });
  })
      .catch(console.error);

})(workerChunk, JSZip, GIF);