Pixiv Downloader

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

Від 29.05.2023. Дивіться остання версія.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(У мене вже є менеджер скриптів, дайте мені встановити його!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

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