Pixiv Downloader

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

As of 2022-11-23. See the latest version.

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 or Violentmonkey 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.

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

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.7.0
// @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 () {
  'use strict';

  const style = `
@property --pdl-progress {
  syntax: '<percentage>';
  inherits: true;
  initial-value: 0%;
}
@keyframes pdl_loading {
  100% {
    transform: translate(-50%, -50%) rotate(360deg);
  }
}
[data-theme="dark"] .pdl-btn-all::before {
  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");
}
[data-theme="dark"] .pdl-btn-main,
[data-theme="dark"] .pdl-btn-all:hover::before {
  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");
}
[data-theme="dark"] .pdl-stop::before {
  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");
}
[data-theme="dark"] .pdl-stop:hover::before {
  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");
}
[data-theme="dark"] .pdl-wrap input:not(:checked):hover {
  background-color: rgba(155, 155, 155);
}
[data-theme="dark"] .pdl-btn.pdl-tag{
  background-color: rgba(255, 255, 255, 0.4);
}
[data-theme="dark"] .pdl-btn.pdl-modal-tag {
  background-color: rgba(255, 255, 255, 0.4);
}
[data-theme="dark"] .pdl-btn.pdl-modal-tag:hover {
  background-color: rgba(255, 255, 255, 0.6);
}
[data-theme="dark"] .pdl-wrap:hover,
[data-theme="dark"] .pdl-stop.pdl-stop:hover,
[data-theme="dark"] .pdl-btn-all.pdl-btn-all:hover {
  color: rgb(214, 214, 214);
}
[data-theme="dark"] .pdl-dialog {
  background-color: rgb(31, 31, 31);
}
[data-theme="dark"] .pdl-dialog-footer button {
  background-color: rgb(245, 245, 245);
}
.pdl-btn {
  position: relative;
  border-top-right-radius: 8px;
  background: no-repeat center/85%;
  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");
  color: #01b468;
  display: inline-block;
  font-size: 13px;
  font-weight: bold;
  height: 32px;
  line-height: 32px;
  margin: 0;
  overflow: hidden;
  padding: 0;
  border: none;
  text-decoration: none!important;
  text-align: center;
  text-overflow: ellipsis;
  user-select: none;
  white-space: nowrap;
  width: 32px;
  z-index: 1;
  cursor: pointer;
}
.pdl-btn-main {
  margin: 0 0 0 10px;
}
.pdl-btn-sub {
  bottom: 0;
  background-color: rgba(255, 255, 255, .5);
  left: 0;
  position: absolute;
}
.pdl-btn-sub.artworks{
  position: sticky;
  top: 40px;
  border-radius: 4px;
}
.pdl-btn-sub.presentation{
  position: fixed;
  top: 50px;
  right: 16px;
  border-radius: 8px;
  left: auto;
}
.pdl-btn-sub-bookmark.pdl-btn-sub-bookmark {
  left: auto;
  right: 0;
  bottom: 34px;
  border-radius: 8px;
  border-top-right-radius: 0px;
  border-bottom-right-radius: 0px;
}
.pdl-error.pdl-error {
  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");
}
.pdl-complete.pdl-complete {
  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");
}
.pdl-progress.pdl-progress {
  background-image: none;
  cursor: default;
}
.pdl-progress:after{
  content: '';
  display: inline-block;
  position: absolute;
  top: 50%;
  left: 50%;
  width: 27px;
  height: 27px;
  transform: translate(-50%, -50%);
  -webkit-mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);
  mask: radial-gradient(transparent, transparent 54%, #000 57%, #000);
  border-radius: 50%;
}
.pdl-progress:not(:empty):after {
  background: conic-gradient(#01B468 0, #01B468 var(--pdl-progress), transparent var(--pdl-progress), transparent);
  transition: --pdl-progress .2s ease;
}
.pdl-progress:empty:after {
  background: conic-gradient(#01B468 0, #01B468 25%, #01B46833 25%, #01B46833);
  animation: 1.5s infinite linear pdl_loading;
}
.pdl-nav-placeholder {
  flex-grow: 1;
  height: 42px;
  line-height: 42px;
  text-align: right;
  font-weight: bold;
  font-size: 16px;
  color: rgb(133, 133, 133);
  border-top: 4px solid transparent;
  cursor: default;
  white-space: nowrap;
}
.pdl-btn-all.pdl-btn-all,
.pdl-stop.pdl-stop {
  background-color: transparent;
  border: none;
  padding: 0 10px;
}
.pdl-btn-all.pdl-btn-all:hover,
.pdl-stop.pdl-stop:hover {
  color: rgb(31, 31, 31);
}
.pdl-btn-all::before,
.pdl-stop::before {
  content: '';
  height: 24px;
  width: 24px;
  transition: background-image 0.2s ease 0s;
  background: no-repeat center/85%;
}
.pdl-btn-all::before {
  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");
}
.pdl-stop::before {
  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");
}
.pdl-btn-all:hover::before{
  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");
}
.pdl-stop:hover::before {
  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");
} 
.pdl-hide {
  display: none!important;
}
.pdl-wrap {
  text-align: right;
  padding-right: 24px;
  font-weight: bold;
  font-size: 14px;
  line-height: 14px;
  color: rgb(133, 133, 133);
  transition: color 0.2s ease 0s;
}
.pdl-wrap:hover {
  color: rgb(31, 31, 31);
}
.pdl-wrap label {
  padding-left: 8px;
  cursor: pointer;
}
.pdl-wrap input {
  vertical-align: top;
  appearance: none;
  position: relative;
  box-sizing: border-box;
  width: 28px;
  border: 2px solid transparent;
  cursor: pointer;
  border-radius: 14px;
  height: 14px;
  background-color: rgba(133, 133, 133);
  transition: background-color 0.2s ease 0s, box-shadow 0.2s ease 0s;
}
.pdl-wrap input:hover {
  background-color: rgba(31, 31, 31);
}
.pdl-wrap input::after {
  content: "";
  position: absolute;
  display: block;
  top: 0px;
  left: 0px;
  width: 10px;
  height: 10px;
  transform: translateX(0px);
  background-color: rgb(255, 255, 255);
  border-radius: 10px;
  transition: transform 0.2s ease 0s;
}
.pdl-wrap input:checked {
  background-color: rgb(0, 150, 250);
}
.pdl-wrap input:checked::after {
  transform: translateX(14px);
}
.pdl-wrap-artworks {
  position: absolute;
  right: 8px;
  top: 0px;
  bottom: 0px;
  margin-top: 40px;
}
.pdl-modal * {
  font-family: 'win-bug-omega, system-ui, -apple-system, "Segoe UI", Roboto, Ubuntu, Cantarell, "Noto Sans", "Hiragino Kaku Gothic ProN", Meiryo, sans-serif';
  line-height: 1.15;
}
.pdl-modal {
  position: fixed;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  display: flex;
  z-index: 99;
  background-color: rgba(0, 0, 0, 0.32);
  user-select: none;
}
.pdl-dialog {
  position: relative;
  background-color: #fff;
  border-radius: 24px;
  margin: auto;
  padding: 20px 40px 30px 40px;
  max-width: 720px;
  font-size: 16px;
}
.pdl-dialog-header > h3 {
  font-weight: bold;
  font-size: 1.17em;
  margin: 1em 0;
}
.pdl-dialog p {
  margin: 1em 0px;
  overflow-wrap: break-word;
}
.pdl-dialog-close {
  position: absolute;
  top: 10px;
  right: 10px;
  margin: 0;
  padding: 0;
  width: 25px;
  height: 25px;
  border: none;
  cursor: pointer;
  border-radius: 50%;
  background-color: transparent;
  transform: rotate(45deg);
  transition: 0.25s background-color;
  background: linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/18px 2px no-repeat,
    linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/2px 18px no-repeat;
}
.pdl-dialog-close:hover {
  background-color: rgba(0, 0, 0, 0.05);
}
.pdl-dialog-content {
  user-select: text;
}
.pdl-btn.pdl-tag {
  height: auto;
  border-top-right-radius: 4px;
  border-bottom-right-radius: 4px;
  left: -1px;
  background-color: rgba(0, 0, 0, 0.12);
  transition: background-image 0.5s;
}
.pdl-btn.pdl-tag.pdl-tag-hide{
  background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E  %3C/svg%3E");
  pointer-events: none;
}
.pdl-btn.pdl-modal-tag {
  position: absolute;
  right: 65px;
  top: 6px;
  background-origin: content-box;
  border-radius: 4px;
  padding: 5px;
  width: 42px;
  height: 50px;
  background-color: rgba(0,0,0,0.04);
  transition: .25s background-color;
}
.pdl-btn.pdl-modal-tag:hover {
  background-color: rgba(0,0,0,0.12);
}
`;
  function addStyle() {
    const sty = document.createElement('style');
    sty.innerHTML = style;
    document.head.appendChild(sty);
  }

  function debugLog(...msgs) {
  }

  const defaultSettings = {
    version: '0.7.0',
    ugoriaFormat: 'zip',
    folderPattern: '{artist}',
    filenamePattern: '{artist}_{title}_{id}_p{page}',
    tagLang: 'ja',
    showMsg: true,
    log: false,
  };
  const regexp = {
    artworksPage: /artworks\/(\d+)$/,
    userPage: /users\/(\d+)/,
    bookmarkPage: /users\/\d+\/bookmarks\/artworks/,
    userPageTags: /users\/\d+\/(artworks|illustrations|manga|bookmarks(?!artworks))/,
    ppSearchPage: /\/tags\/.*\/(artworks|illustrations|manga)/,
    suscribePage: /bookmark_new_illust/,
    activityHref: /illust_id=(\d+)/,
    originSrcPageNum: /(?<=_p)\d+/,
  };
  const artworkType = {
    ILLUSTS: 0,
    MANGA: 1,
    UGOIRA: 2,
  };
  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 text = {
    upgradeMsgTitle: `<h3>Pixiv Downloader ${defaultSettings.version}</h3>`,
    upgradeMsgContent: `<p>1. 现在可以将动图转换为<strong>Webp格式</strong>了。Webp相比Gif质量更高,体积更小,如<a style="color: rgb(0, 0, 238); text-decoration: underline" href="https://www.pixiv.net/artworks/102966692" target="_blank">102966692</a>。</p><p>请将webp格式加入到Tampermonkey的下载<strong>文件扩展名白名单</strong>中,否则下载会失败。</p><p>如  /\\.(ico|gif|png|jpe?g)/  修改为  /\\.(ico|gif|png|jpe?g|webp)/</p><p>2. Webp格式目前默认无损转换,部分动图转换时间可能比较长。</p><p>3. 优化了夜间模式的样式,现在能看清了。</p>`,
    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: rgb(0, 0, 238); text-decoration: underline">脚本还行?请我喝杯可乐吧!</summary>
  <img style="display: block; margin: 1em auto; width: 200px"
    src=""
  />
  <p style="text-align: center">什么时候才有无糖香草味可乐喝啊w(゚Д゚)w</p>
</details>`,
    modalFeedback: `<a target="_blank" style="position: absolute; right: 0px; top: 0px; color: rgb(0, 0, 238); text-decoration: underline" href="https://greasyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback">有问题or想建议?这里反馈</a>`,
    filePathSettingTitle: `<h3>设置文件名</h3>`,
    filePathSettingContent: `<style>.pdl-dialog-content input[type="text"] {height: auto; padding: 0.5em; line-height: 1.5; margin: 0.6em 0 0.3em 0; font-size: 16px;}.pdl-dialog-content a{color: rgb(0, 0, 238); text-decoration: underline;} .tags-option label,.tags-option input {cursor: pointer;}</style><div style="display: flex; gap: 20px; justify-content: space-between;">
  <div>
    <label style="display: block; cursor: default;" for="pdlfolder">文件夹名:</label>
    <input type="text" id="pdlfolder" style="width: 200px;" placeholder="我不想保存到画师文件夹">
  </div>
  <div>
    <label style="display: block; cursor: default;" for="pdlfilename">文件名:</label>
    <input type="text" id="pdlfilename" style="width: 300px;" placeholder="你的名字?">
  </div>
</div>
<div class="tags-option" style="margin: 0.7em 0;">
  <span>标签翻译:</span>
  <input type="radio" name="lang" id="lang_ja" value="ja"/>
  <label for="lang_ja">日本語(不翻译)</label>
  <input type="radio" name="lang" id="lang_zh" value="zh" />
  <label for="lang_zh">简中</label>
  <input type="radio" name="lang" id="lang_zh_tw" value="zh_tw" />
  <label for="lang_zh_tw">繁中</label>
  <input type="radio" name="lang" id="lang_en" value="en" />
  <label for="lang_en">English</label>
</div>
<p style="font-size: 14px; margin: 0.5em 0">
  {artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}:作品pixiv ID, {page}:页码, {tags}:作品标签。
</p>
<p style="font-size: 14px; margin: 0.5em 0">如果不想保存到画师目录,文件夹名留空即可。</p>
<p style="font-size: 14px; margin: 0.5em 0">请注意:标签翻译不一定是你选择的语言,部分<a href="https://crowdin.com/project/pixiv-tags" target="_blank">无对应语言翻译的标签</a>仍可能是其他语言。</p>
</div>`,
    modalOperationBar: `<style>
  .pdl-dialog-footer button {
    font-size: 16px;
    background-color: transparent;
    border: 1px solid;
    color: rgb(125,125,125);
    border-radius: 5px;
    padding: 0.5em 1.5em;
    cursor: pointer;
    transition: .2s opacity;
    line-height: 1.15;
  }
  .pdl-dialog-footer button:hover{
    opacity: 0.7;
  }
</style>
<div style="display: flex; justify-content: flex-end; margin-top: 1.5em; gap: 1.5em;">
  <button id="pdlcancel">取消</button><button id="pdlconfirm" style="border-color: #01b468; background-color: #01b468; color: #fff;">确认</button></div>`,
  };

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

  postMessage('ok');
};

onmessage = (evt) => {
  const { dataURLs, delays, lossless = 1, quality = 75, method = 4} = evt.data;
  
  webpApi.init(lossless, quality, method);
  dataURLs.forEach((dataURL, idx) => {
    const base64 = dataURL.split(',')[1];
    const binStr = atob(base64);
    const u8a = new Uint8Array(binStr.length);
    let p = binStr.length;
    while (p) {
      p--;
      u8a[p] = binStr.codePointAt(p);
    }

    const pointer = webpApi.createBuffer(u8a.length);
    Module.HEAPU8.set(u8a, pointer);
    webpApi.addFrame(pointer, u8a.length, delays[idx]);
    postMessage(idx);
  });

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

  function initialDeps(urls) {
    return Promise.all([
      _getGifWS(urls.gifWorker),
      _getApngWS(urls.pako, urls.upng),
      _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 _fetchDeps(url) {
    return fetch(url)
      .then((res) => {
        if (res.ok) return res.text();
        throw new Error(res.status + res.statusText);
      })
      .catch((err) => {
        console.log('[Pixiv Downloader]Fetch dependency failed.', url, err);
        return '';
      });
  }
  async function _getGifWS(url) {
    let gifWS;
    if (!(gifWS = await GM_getValue('gifWS'))) {
      gifWS = await _fetchDeps(url);
      if (!gifWS) throw new Error('[Pixiv Downloader]Can not fetch gif worker script.');
      GM_setValue('gifWS', gifWS);
    }
    return gifWS;
  }
  async function _getApngWS(pakoUrl, upngUrl) {
    let apngWS;
    if (!(apngWS = await GM_getValue('apngWS'))) {
      let pako = _fetchDeps(pakoUrl);
      let upng = _fetchDeps(upngUrl);
      pako = await pako;
      upng = await upng;
      if (!pako || !upng) throw new Error('[Pixiv Downloader]Can not fetch apng script.');
      upng = 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 + upng;
      GM_setValue('apngWS', apngWS);
    }
    return apngWS;
  }
  function _getWebpWS() {
    return workerChunk + handleWorker;
  }
  function _createImgElements(zip) {
    const eles = [];
    zip.forEach((relativePath, file) => {
      eles.push(
        new Promise((resolve) => {
          const image = new Image();
          image.onload = () => {
            resolve(image);
          };
          file.async('blob').then((blob) => void (image.src = URL.createObjectURL(blob)));
        })
      );
    });
    return Promise.all(eles);
  }
  function createInstance() {
    const zip = new JSZip();
    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(this._deps.webp);
          }
          convertMeta.abort = convertMeta._baseAbort.bind(null, () => {
            reject('[Info]Convert stop manually, reject when convert webp. ' + convertMeta.id);
            worker.terminate();
          });
          const workerLoad = new Promise((resolve) => {
            if (reuse) return resolve();
            worker.onmessage = (evt) => {
              if (evt.data === 'ok') {
                resolve();
              }
            };
          });
          let dataURLs = [];
          let 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 });
          const delays = convertMeta.framesInfo.map((frameInfo) => {
            return Number(frameInfo.delay);
          });
          dataURLs = frames.map((frame, idx) => {
            if (convertMeta.isAborted)
              throw '[Info]Convert stop manually when converting image to webp. ' + convertMeta.id;
            context.clearRect(0, 0, width, height);
            context.drawImage(frame, 0, 0, width, height);
            const dataURL = canvas.toDataURL('image/webp', 1);
            if (typeof convertMeta.onProgress === 'function') {
              debugLog('[Info]Webp convert phrase 1:', convertMeta.id);
              convertMeta.onProgress((idx / frames.length) * 0.5, 'webp');
            }
            return dataURL;
          });
          workerLoad.then(() => {
            worker.onmessage = (evt) => {
              if (typeof evt.data !== 'object') {
                if (typeof convertMeta.onProgress === 'function') {
                  debugLog('[Info]Webp convert phrase 2:', convertMeta.id, evt.data);
                  convertMeta.onProgress(0.5 + (evt.data / frames.length) * 0.5, 'webp');
                }
              } else {
                freeWebpWorkers.push(worker);
                resolve(new Blob([evt.data], { type: 'image/webp' }));
              }
            };
            worker.postMessage({ dataURLs, delays });
          });
        });
      },
      gif: (frames, convertMeta) => {
        return new Promise((resolve, reject) => {
          let gif = new GIF({
            workers: 2,
            quality: 10,
            workerScript: this._deps.gif,
          });
          convertMeta.abort = convertMeta._baseAbort.bind(null, gif.abort.bind(gif));
          debugLog('[Info]Start convert:', convertMeta.id);
          frames.forEach((frame, i) => {
            gif.addFrame(frame, { delay: convertMeta.framesInfo[i].delay });
          });
          gif.on(
            'progress',
            (() => {
              const type = 'gif';
              return (progress) => {
                debugLog('[Info]Convert progress:', convertMeta.id);
                if (typeof convertMeta.onProgress === 'function')
                  convertMeta.onProgress(progress, type);
              };
            })()
          );
          gif.on('finished', (gifBlob) => {
            gif = null;
            resolve(gifBlob);
          });
          gif.on('abort', () => {
            gif = null;
            reject('[Info]Convert stop: abort. ' + convertMeta.id);
          });
          gif.render();
        });
      },
      png: (frames, convertMeta) => {
        return new Promise((resolve, reject) => {
          let 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 });
          const data = [];
          const delay = convertMeta.framesInfo.map((frameInfo) => {
            return Number(frameInfo.delay);
          });
          frames.forEach((frame) => {
            if (convertMeta.isAborted)
              throw '[Info]Convert stop manually, reject when drawImage. ' + convertMeta.id;
            context.clearRect(0, 0, width, height);
            context.drawImage(frame, 0, 0, width, height);
            data.push(context.getImageData(0, 0, width, height).data);
          });
          canvas = null;
          debugLog('[Info]Start convert:', convertMeta.id);
          let worker;
          if (freeApngWorkers.length) {
            worker = freeApngWorkers.shift();
          } else {
            worker = new Worker(this._deps.apng);
          }
          convertMeta.abort = convertMeta._baseAbort.bind(null, () => {
            reject('[Info]Convert stop manually, reject when convert apng. ' + convertMeta.id);
            worker.terminate();
          });
          worker.onmessage = function (e) {
            freeApngWorkers.push(worker);
            if (!e.data) {
              return reject('[Error]apng data is null. ' + convertMeta.id);
            }
            const pngBlob = new Blob([e.data], { type: 'image/png' });
            resolve(pngBlob);
          };
          const cfg = { data, width, height, delay };
          worker.postMessage(cfg);
        });
      },
      webm: (frames, convertMeta) => {
        return new Promise((resolve, reject) => {
          let canvas = document.createElement('canvas');
          const width = (canvas.width = frames[0].naturalWidth);
          const height = (canvas.height = frames[0].naturalHeight);
          const context = canvas.getContext('2d');
          const stream = canvas.captureStream();
          const recorder = new MediaRecorder(stream, {
            mimeType: 'video/webm',
            videoBitsPerSecond: 80000000,
          });
          const delay = convertMeta.framesInfo.map((frame) => {
            return Number(frame.delay);
          });
          let 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 = () => {
            canvas = null;
            if (convertMeta.isAborted) {
              return reject(
                '[info]Convert stop manually, reject when convert webm.' + convertMeta.id
              );
            }
            resolve(new Blob(data, { type: 'video/webm' }));
          };
          displayFrame();
          recorder.start();
        });
      },
    };
    const convert = (convertMeta) => {
      const { id, data, convertResolve, convertReject } = convertMeta;
      let frames;
      active.push(convertMeta);
      if (typeof convertMeta.onProgress === 'function') convertMeta.onProgress(0, 'zip');
      zip
        .folder(id)
        .loadAsync(data)
        .then(_createImgElements)
        .then((imgEles) => {
          zip.remove(id);
          frames = imgEles;
          if (convertMeta.isAborted) throw '[Info]Convert stop manually, reject when unzip. ' + id;
          return convertTo[convertMeta.format](frames, convertMeta);
        })
        .then(convertResolve)
        .catch(convertReject)
        .finally(() => {
          frames.forEach((frame) => URL.revokeObjectURL(frame.src));
          frames = null;
          active.splice(active.indexOf(convertMeta), 1);
          if (queue.length) convert(queue.shift());
        });
    };
    return {
      add: (convertMeta) => {
        debugLog('[Info]Converter add', convertMeta.id);
        return new Promise((convertResolve, convertReject) => {
          convertMeta.isAborted = false;
          convertMeta.convertResolve = convertResolve;
          convertMeta.convertReject = convertReject;
          convertMeta._baseAbort = (callBack) => {
            if (typeof callBack === 'function') callBack();
            convertMeta.isAborted = true;
          };
          convertMeta.abort = convertMeta._baseAbort;
          queue.push(convertMeta);
          while (active.length < MAX_CONVERT && queue.length && !isStop) {
            convert(queue.shift());
          }
        });
      },
      del: (metas) => {
        if (!metas.length) return;
        isStop = true;
        active = active.filter((convertMeta) => {
          if (metas.find((meta) => meta.id === convertMeta.id)) {
            convertMeta.abort();
          } else {
            return true;
          }
        });
        queue = queue.filter((convertMeta) => !metas.find((meta) => meta.id === convertMeta.id));
        isStop = false;
        while (active.length < MAX_CONVERT && queue.length) {
          convert(queue.shift());
        }
      },
    };
  }
  const createConverter = {
    _deps: {
      gif: '',
      apng: '',
      webp: '',
    },
    initialDeps,
    createInstance,
  };

  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;
        for (const key in defaultSettings) {
          if (!(key in settings)) {
            settings[key] = defaultSettings[key];
          }
        }
        saveSettings(settings);
      }
    }
    return settings;
  }
  function saveSettings(settingObj) {
    settingObj = settingObj || settings;
    localStorage.pdlSetting = JSON.stringify(settingObj);
  }
  function upgradeSettings(key, value) {
    if (key in settings) {
      if (settings[key] === value) return;
      settings[key] = value;
      saveSettings();
    }
  }
  const settings = getSettings();
  function createSetFormatFn(format) {
    return () => {
      if (settings.ugoriaFormat !== format) {
        upgradeSettings('ugoriaFormat', format);
      }
    };
  }

  function sleep(delay) {
    return new Promise((resolve) => {
      setTimeout(resolve, delay);
    });
  }
  function getOwnerId() {
    return document.querySelector('#qualtrics_user-id')?.textContent;
  }
  const isBlobDlAvaliable = !(
    navigator.userAgent.includes('Firefox') &&
    GM_info.scriptHandler === 'Tampermonkey' &&
    parseFloat(GM_info.version) > 4.17
  );
  const isViolentmonkey = GM_info.scriptHandler === 'Violentmonkey';

  const _isNeedConvert = (meta) => {
    return meta.illustType === artworkType.UGOIRA && settings.ugoriaFormat !== 'zip';
  };
  const _ffSave = (blob, meta) => {
    const dlEle = document.createElement('a');
    dlEle.href = URL.createObjectURL(blob);
    dlEle.download = meta.path.slice(meta.path.indexOf('/') + 1);
    dlEle.click();
    URL.revokeObjectURL(dlEle.href);
    meta.resolve(meta);
  };
  const _normalSave = (blob, meta) => {
    const imgUrl = URL.createObjectURL(blob);
    const request = {
      url: imgUrl,
      name: meta.path,
      onerror: (error) => {
        console.log('[pixiv downloader]Error when saving', meta.path);
        URL.revokeObjectURL(imgUrl);
        meta.reject && meta.reject(error);
      },
      onload: () => {
        if (typeof meta.onLoad === 'function') meta.onLoad();
        URL.revokeObjectURL(imgUrl);
        meta.resolve(meta);
      },
    };
    meta.abort = GM_download(request).abort;
  };
  function createDownloader(converter) {
    const MAX_DOWNLOAD = 5;
    const MAX_RETRY = 3;
    let isStop = false;
    let queue = [];
    let active = [];
    let save;
    if (isBlobDlAvaliable && !isViolentmonkey) {
      save = _normalSave;
    } else {
      debugLog('[Info]scriptHandler:', GM_info.scriptHandler, GM_info.version);
      save = _ffSave;
    }
    const download = (meta) => {
      debugLog('[Info]Start download:', meta.path);
      active.push(meta);
      let abortObj;
      if ((!isBlobDlAvaliable || isViolentmonkey) && !_isNeedConvert(meta)) {
        abortObj = GM_download({
          url: meta.src,
          name: meta.path,
          headers: {
            referer: 'https://www.pixiv.net',
          },
          ontimeout: errHandler.bind(null, meta),
          onerror: errHandler.bind(null, meta),
          onload: () => {
            debugLog('[Info]Download complete', meta.path);
            if (typeof meta.onLoad === 'function') meta.onLoad();
            active.splice(active.indexOf(meta), 1);
            if (queue.length && !isStop) download(queue.shift());
            meta.resolve(meta);
          },
        });
      } else {
        const request = {
          url: meta.src,
          timeout: 20000,
          method: 'GET',
          headers: {
            referer: 'https://www.pixiv.net',
          },
          responseType: 'blob',
          ontimeout: errHandler.bind(null, meta),
          onprogress: (e) => {
            if (e.lengthComputable && typeof meta.onProgress === 'function') {
              meta.onProgress(e.loaded / e.total);
            }
          },
          onload: (e) => {
            debugLog('[Info]Download complete', meta.id);
            if (!meta.state) return debugLog('[Warning]But download was canceled.', meta.id);
            if (_isNeedConvert(meta)) {
              const convertMeta = {
                id: meta.id,
                data: e.response,
                format: settings.ugoriaFormat,
                framesInfo: meta.ugoiraMeta.frames,
                onProgress: meta.onProgress,
              };
              converter.add(convertMeta).then((blob) => {
                save(blob, meta);
              }, meta.reject);
            } else {
              save(e.response, meta);
            }
            active.splice(active.indexOf(meta), 1);
            if (queue.length && !isStop) download(queue.shift());
          },
          onerror: errHandler.bind(null, meta),
        };
        abortObj = GM_xmlhttpRequest(request);
      }
      meta.abort = () => {
        meta.state = 0;
        abortObj.abort();
        meta.reject('[Warning]xhr abort manually. ' + meta.id);
      };
    };
    const add = (metas) => {
      if (metas.length < 1) return;
      const promises = [];
      metas.forEach((meta) => {
        promises.push(
          new Promise((resolve, reject) => {
            meta.state = 1;
            meta.resolve = resolve;
            meta.reject = reject;
          })
        );
      });
      queue = queue.concat(metas);
      while (active.length < MAX_DOWNLOAD && queue.length && !isStop) {
        download(queue.shift());
      }
      return Promise.all(promises);
    };
    const del = (metas) => {
      if (!metas.length) return;
      isStop = true;
      active = active.filter((meta) => {
        if (metas.includes(meta)) {
          meta.abort();
        } else {
          return true;
        }
      });
      queue = queue.filter((meta) => !metas.includes(meta));
      isStop = false;
      while (active.length < MAX_DOWNLOAD && queue.length) {
        download(queue.shift());
      }
    };
    const errHandler = (meta) => {
      debugLog('[Error]xmlhttpRequest timeout:', meta.src);
      if (!meta.retries) {
        meta.retries = 1;
      } else {
        meta.retries++;
      }
      if (meta.retries > MAX_RETRY) {
        meta.reject('[Error]xmlhttpRequest failed: ' + meta.src);
        console.log('[pixiv downloader]Network error:', meta.path, meta.src);
        active.splice(active.indexOf(meta), 1);
        if (queue.length && !isStop) download(queue.shift());
      } else {
        debugLog('[Warning]retry xhr:', meta.retries, meta.src);
        download(meta);
      }
    };
    return {
      add: add,
      del: del,
    };
  }

  function createParser() {
    const replaceInvalidChar = (string) => {
      if (!string) return;
      const temp = document.createElement('div');
      temp.innerHTML = string;
      return temp.textContent
        .trim()
        .replace(/^\.|\.$/g, '')
        .replace(/[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?|]/g, '')
        .replace(/"/g, "'")
        .replace(/</g, '﹤')
        .replace(/>/g, '﹥');
    };
    const getFilePath = ({ user, userId, title, tags, illustId, page, ext }) => {
      const path =
        settings.folderPattern && !isViolentmonkey
          ? settings.folderPattern + '/' + settings.filenamePattern
          : settings.filenamePattern;
      return (
        path
          .replaceAll('{artist}', user)
          .replaceAll('{artistID}', userId)
          .replaceAll('{title}', title)
          .replaceAll('{tags}', tags)
          .replaceAll('{page}', page)
          .replaceAll('{id}', illustId) + ext
      );
    };
    const makeTagsStr = (prev, cur, index, tagsArr) => {
      const tag = settings.tagLang === 'jp' ? cur.tag : cur.translation?.['en'] || cur.tag;
      if (index < tagsArr.length - 1) {
        return prev + tag + '_';
      } else {
        return prev + tag;
      }
    };
    const getData = async (url) => {
      const res = await fetch(url);
      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;
    };
    const fetchJson = async (url) => {
      let json;
      let retry = 0;
      do {
        try {
          debugLog('[Info]fetch url:', url);
          json = await getData(url);
        } catch (error) {
          retry++;
          if (retry === 3) throw error;
          sleep(3000);
        }
      } while (!json);
      return json;
    };
    const parseByIllust = async (illustId) => {
      let params = '';
      if (settings.tagLang !== 'jp') params = '?lang=' + settings.tagLang;
      const res = await fetch('https://www.pixiv.net/artworks/' + illustId + params);
      if (!res.ok) throw new Error(res.status);
      const htmlText = await res.text();
      const matchText = htmlText.match(/"meta-preload-data" content='(.*)'>/);
      if (!matchText) throw new Error('[Error]Fail to parse preload data.');
      const preloadData = JSON.parse(htmlText.match(/"meta-preload-data" content='(.*)'>/)[1]);
      const illustInfo = preloadData.illust[illustId];
      const user = replaceInvalidChar(illustInfo.userName) || 'userId-' + illustInfo.userId;
      const title = replaceInvalidChar(illustInfo.illustTitle) || 'illustId-' + illustInfo.illustId;
      const tags = replaceInvalidChar(illustInfo.tags.tags.reduce(makeTagsStr, ''));
      const illustType = illustInfo.illustType;
      let metas = [];
      const pathInfo = {
        user,
        title,
        tags,
        illustId,
        userId: illustInfo.userId,
        ext: '',
        page: 0,
      };
      if (illustType === artworkType.ILLUSTS || illustType === artworkType.MANGA) {
        const firstImgSrc = illustInfo.urls.original;
        const srcPrefix = firstImgSrc.slice(0, firstImgSrc.indexOf('_') + 2);
        const srcSuffix = firstImgSrc.slice(-4);
        pathInfo.ext = srcSuffix;
        for (let i = 0; i < illustInfo.pageCount; i++) {
          pathInfo.page = i;
          metas.push({
            id: illustId,
            illustType: illustType,
            path: getFilePath(pathInfo),
            src: srcPrefix + i + srcSuffix,
          });
        }
      }
      if (illustType === artworkType.UGOIRA) {
        const ugoira = await fetchJson(
          'https://www.pixiv.net/ajax/illust/' + illustId + '/ugoira_meta'
        );
        pathInfo.ext = '.' + settings.ugoriaFormat;
        metas.push({
          id: illustId,
          illustType: illustType,
          path: getFilePath(pathInfo),
          src: ugoira.body.originalSrc,
          ugoiraMeta: ugoira.body,
        });
      }
      return metas;
    };
    function _filterBookmarks(works) {
      const unavaliable = [];
      function filterFn(work) {
        if (work.isBookmarkable) {
          return true;
        } else {
          unavaliable.push(work.id);
        }
      }
      const avaliable = works.filter(filterFn).map((work) => work.id);
      return { avaliable, unavaliable };
    }
    async function* generateIds(userId, category, tag = '', rest = 'show') {
      let requestUrl;
      if (tag || category === 'bookmarks') {
        const OFFSET = 48;
        if (category !== 'bookmarks') {
          requestUrl = `https://www.pixiv.net/ajax/user/${userId}/${category}/tag?tag=${tag}&offset=0&limit=${OFFSET}&lang=ja`;
        } else {
          requestUrl = `https://www.pixiv.net/ajax/user/${userId}/illusts/bookmarks?tag=${tag}&offset=0&limit=${OFFSET}&rest=${rest}&lang=ja`;
        }
        let head = 0;
        const firstPageData = await fetchJson(requestUrl);
        const total = firstPageData.body.total;
        yield total;
        yield _filterBookmarks(firstPageData.body.works);
        head += OFFSET;
        while (head < total) {
          const data = await fetchJson(requestUrl.replace('offset=0', 'offset=' + head));
          head += OFFSET;
          await sleep(3000);
          yield _filterBookmarks(data.body.works);
        }
      } else {
        requestUrl = 'https://www.pixiv.net/ajax/user/' + userId + '/profile/all';
        const profile = await fetchJson(requestUrl);
        let illustIds;
        if (category !== 'both') {
          illustIds = Reflect.ownKeys(profile.body[category]);
        } else {
          illustIds = Reflect.ownKeys(profile.body.illusts).concat(
            Reflect.ownKeys(profile.body.manga)
          );
        }
        yield illustIds.length;
        yield { avaliable: illustIds, unavaliable: [] };
      }
    }
    return {
      id: parseByIllust,
      generateIds,
    };
  }

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

  function add(ele) {
    this._records.add(ele);
  }
  function has(ele) {
    return this._records.has(ele);
  }
  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("Do you really want to clear history?");
    if (!isConfirm) return;
    this.updateHistory();
    this._records = new Set();
    localStorage.pixivDownloader = "[]";
  }
  function saveHistory() {
    localStorage.pixivDownloader = JSON.stringify([...this._records]);
  }
  const pixivHistory = {
    _records: getHistory(),
    add,
    has,
    updateHistory,
    saveHistory,
    clearHistory,
  };

  function handleDownload(pdlBtn, illustId) {
    let pageCount,
      pageComplete = 0;
    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 = progress;
          break;
        case 'zip':
          pdlBtn.textContent = '';
          break;
      }
    };
    const onLoad = function () {
      if (pageCount < 2) return;
      const progress = Math.floor((++pageComplete / pageCount) * 100);
      pdlBtn.textContent = progress;
      pdlBtn.style.setProperty('--pdl-progress', progress + '%');
    };
    pdlBtn.classList.add('pdl-progress');
    parser
      .id(illustId)
      .then((metas) => {
        let shouldDownloadPage;
        if ((shouldDownloadPage = pdlBtn.getAttribute('should-download'))) {
          metas = [metas[shouldDownloadPage]];
        }
        pageCount = metas.length;
        metas.forEach((meta) => {
          meta.onProgress = onProgress;
          meta.onLoad = onLoad;
        });
        return downloader.add(metas);
      })
      .then(() => {
        pixivHistory.add(illustId);
        localStorage.setItem(`pdlTemp-${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 changeDlbarDisplay() {
    document.querySelectorAll('nav [pdl-userid]').forEach((ele) => {
      ele.classList.toggle('pdl-hide');
    });
    document.querySelectorAll('section [pdl-userid]').forEach((ele) => {
      ele.classList.toggle('pdl-tag-hide');
    });
  }
  let isDownloading = false;
  let dlBarRef = {};
  async function downloadByIds(idsGenerators, abortBtn, isExcludeDled, updataStatus, onProgressCB) {
    if (!(idsGenerators instanceof Array)) idsGenerators = [idsGenerators];
    let resolve, reject;
    const done = new Promise((r, j) => {
      resolve = r;
      reject = j;
    });
    let total = 0,
      completed = 0,
      failed = [],
      unavaliable = [];
    let isCanceled = false;
    let metasRecord = [];
    let tooManyRequests = false;
    if (isExcludeDled) pixivHistory.updateHistory();
    abortBtn.onclick = () => {
      isCanceled = true;
      abortBtn.onclick = null;
      if (metasRecord.length) {
        downloader.del(metasRecord);
        converter.del(metasRecord);
        metasRecord = [];
      }
      reject(`Stopped. ${completed} / ${total}`);
    };
    const afterEach = (illustId) => {
      onProgressCB({
        illustId,
        total,
        completed,
      });
      if (completed === total - failed.length - unavaliable.length) {
        resolve({ failed, unavaliable });
      }
    };
    total = await idsGenerators.reduce(async (prev, cur, index) => {
      const count = (await cur.next()).value;
      return (await prev) + count;
    }, 0);
    if (total === 0) {
      resolve();
      throw 'No Works.';
    }
    updataStatus('Downloading...');
    try {
      for (const idsGenerator of idsGenerators) {
        if (isCanceled) return done;
        for await (const ids of idsGenerator) {
          debugLog('[Info]ids:', ids);
          if (isCanceled) return done;
          if (ids.unavaliable.length) {
            unavaliable.push(...ids.unavaliable);
            debugLog('[Info]unavaliable ids:', unavaliable.length);
          }
          for (const id of ids.avaliable) {
            if (isCanceled) return done;
            if (isExcludeDled && pixivHistory.has(id)) {
              total--;
              afterEach(id);
              continue;
            }
            if (tooManyRequests) {
              updataStatus('Too many requests, wait 30s');
              console.log('[Pixiv Downloader]Too many requests, wait 30s');
              await sleep(30000);
              tooManyRequests = false;
              updataStatus('Downloading...');
            }
            parser
              .id(id)
              .then((metas) => {
                if (isCanceled) {
                  throw '[Warning]Download stop manually: ' + metas[0].id;
                }
                metasRecord = metasRecord.concat(metas);
                return downloader.add(metas);
              })
              .then(
                (metas) => {
                  pixivHistory.add(id);
                  localStorage.setItem(`pdlTemp-${id}`, '');
                  if (!isCanceled) {
                    metasRecord = metasRecord.filter((meta) => !metas.includes(meta));
                    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(600);
          }
        }
      }
    } catch (error) {
      console.log(error);
      reject('Error, see console.');
    }
    return done;
  }
  function downloadWorks(evt) {
    evt.preventDefault();
    evt.stopPropagation();
    if (isDownloading) return;
    const btn = evt.target;
    const userId = btn.getAttribute('pdl-userid');
    const category = btn.getAttribute('category');
    const tag = btn.getAttribute('tag') || undefined;
    const rest = btn.getAttribute('rest') || undefined;
    const isExcludeDled = dlBarRef.filter.checked;
    function updataStatus(str) {
      dlBarRef.statusBar.textContent = str;
    }
    function onProgressCB({ illustId, total, completed }) {
      updataStatus(`Downloading: ${completed} / ${total}`);
    }
    let idsGenerators;
    if (category === 'bookmarks' && rest === 'all') {
      const idsShow = parser.generateIds(userId, category, tag, 'show');
      const idsHide = parser.generateIds(userId, category, tag, 'hide');
      idsGenerators = [idsShow, idsHide];
    } else {
      idsGenerators = parser.generateIds(userId, category, tag, rest);
    }
    function download(idsGenerators) {
      isDownloading = true;
      changeDlbarDisplay();
      return downloadByIds(
        idsGenerators,
        dlBarRef.abortBtn,
        isExcludeDled,
        updataStatus,
        onProgressCB
      )
        .then(
          ({ failed, unavaliable }) => {
            if (failed.length || unavaliable.length) {
              updataStatus(`Failed: ${failed.length + unavaliable.length}. See console.`);
              console.log('[Pixiv Downloader]Failed: ', failed.join(', '));
              console.log('[Pixiv Downloader]Unavaliable: ', unavaliable.join(', '));
              if (failed.length) return failed;
            } else {
              updataStatus('Complete');
            }
          },
          (reason) => {
            if (reason) updataStatus(reason);
          }
        )
        .finally((failed) => {
          changeDlbarDisplay();
          isDownloading = false;
          return failed;
        });
    }
    download(idsGenerators).then((failed) => {
      if (failed instanceof Array) {
        const idsGenerator = [failed.length, { avaliable: failed, unavaliable: [] }][
          Symbol.iterator
        ]();
        download(idsGenerator);
      }
    });
  }

  function createModal({ header, content, footer = '' }, option = { closeOnClickModal: true }) {
    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: text.upgradeMsgTitle,
        content: text.upgradeMsgContent,
        footer: text.modalCreditFooter + text.modalFeedback,
      })
    );
  }
  function showFilePathSetting() {
    if (document.querySelector('#pdlfolder')) return;
    const modal = createModal(
      {
        header: text.filePathSettingTitle,
        content: text.filePathSettingContent,
        footer: text.modalOperationBar,
      },
      { closeOnClickModal: false }
    );
    const folder = modal.querySelector('#pdlfolder');
    const filename = modal.querySelector('#pdlfilename');
    modal.querySelector('#pdlcancel').onclick = () => {
      modal.remove();
    };
    modal.querySelector('#pdlconfirm').onclick = () => {
      if (filename.value === '') return;
      const folderPattern = folder.value
        .trim()
        .replace(/^\.|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, '');
      const filenamePattern = filename.value
        .trim()
        .replace(/^\.|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, '');
      if (filenamePattern === '') return;
      upgradeSettings('tagLang', modal.querySelector(".tags-option [name='lang']:checked").value);
      upgradeSettings('folderPattern', folderPattern);
      upgradeSettings('filenamePattern', filenamePattern);
      modal.remove();
    };
    modal.querySelector(`.tags-option [value="${settings.tagLang}"]`).checked = true;
    folder.value = settings.folderPattern;
    filename.value = settings.filenamePattern;
    if (isViolentmonkey) {
      folder.value = '';
      folder.disabled = true;
      folder.placeholder = 'Violentmonkey不支持';
    }
    document.body.appendChild(modal);
  }
  function getIllustId(node) {
    const isLinkToArtworksPage = regexp.artworksPage.exec(node.href);
    if (isLinkToArtworksPage) {
      if (
        node.getAttribute('data-gtm-value') ||
        node.classList.contains('gtm-illust-recommend-node-node') ||
        node.classList.contains('gtm-discover-user-recommend-node') ||
        node.classList.contains('work')
      ) {
        return isLinkToArtworksPage[1];
      }
    } else {
      const isActivityThumb = regexp.activityHref.exec(node.href);
      if (isActivityThumb && node.classList.contains('work')) {
        return isActivityThumb[1];
      }
    }
    return '';
  }
  function createPdlBtn(attributes, textContent = '', { addEvent } = { addEvent: true }) {
    const ele = document.createElement('button');
    ele.textContent = textContent;
    if (!attributes) return ele;
    const { attrs, classList } = attributes;
    if (classList && classList.length > 0) {
      for (const cla of classList) {
        ele.classList.add(cla);
      }
    }
    if (attrs) {
      for (const key in attrs) {
        ele.setAttribute(key, attrs[key]);
      }
    }
    if (addEvent) {
      ele.addEventListener('click', (evt) => {
        evt.preventDefault();
        evt.stopPropagation();
        const ele = evt.currentTarget;
        if (!evt.currentTarget.classList.contains('pdl-progress')) {
          handleDownload(ele, ele.getAttribute('pdl-id'));
        }
      });
    }
    return ele;
  }
  function createMainBtn(id) {
    if (document.querySelector('.pdl-btn-main')) return;
    const handleBar = document.querySelector('main section section');
    if (handleBar) {
      const pdlBtnWrap = handleBar.lastElementChild.cloneNode();
      const attrs = {
        attrs: { 'pdl-id': id },
        classList: ['pdl-btn', 'pdl-btn-main'],
      };
      if (pixivHistory.has(id)) attrs.classList.push('pdl-complete');
      pdlBtnWrap.appendChild(createPdlBtn(attrs));
      handleBar.appendChild(pdlBtnWrap);
    }
  }
  function createDownloadBar(userId) {
    const nav = document.querySelector('nav');
    if (!nav || document.querySelector('.pdl-nav-placeholder')) return;
    const fragment = document.createDocumentFragment();
    const placeholder = document.createElement('div');
    placeholder.classList.add('pdl-nav-placeholder');
    dlBarRef.statusBar = fragment.appendChild(placeholder);
    const baseClasses = nav.querySelector('a:not([aria-current])').classList;
    dlBarRef.abortBtn = fragment.appendChild(
      createPdlBtn(
        {
          attrs: { 'pdl-userId': userId },
          classList: [...baseClasses, 'pdl-stop', 'pdl-hide'],
        },
        'Stop',
        { addEvent: false }
      )
    );
    if (userId !== getOwnerId()) {
      if (nav.querySelector("a[href$='illustrations']") && nav.querySelector("a[href$='manga']")) {
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userId': userId, category: 'both' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Illusts & Manga',
            { addEvent: false }
          )
        );
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'illusts' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Illusts',
            { addEvent: false }
          )
        );
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'manga' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Manga',
            { addEvent: false }
          )
        );
      } else if (nav.querySelector("a[href$='illustrations']")) {
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'illusts' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Illusts',
            { addEvent: false }
          )
        );
      } else if (nav.querySelector("a[href$='manga']")) {
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'manga' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Manga',
            { addEvent: false }
          )
        );
      }
      if (nav.querySelector("a[href*='bookmarks']")) {
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'bookmarks' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Bookmarks',
            { addEvent: false }
          )
        );
      }
    } else {
      if (nav.querySelector("a[href*='bookmarks']")) {
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'all' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Bookmarks',
            { addEvent: false }
          )
        );
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'show' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Public',
            { addEvent: false }
          )
        );
        fragment.appendChild(
          createPdlBtn(
            {
              attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'hide' },
              classList: [...baseClasses, 'pdl-btn-all'],
            },
            'Private',
            { addEvent: false }
          )
        );
      }
    }
    fragment.querySelectorAll('.pdl-btn-all').forEach((node) => {
      node.addEventListener('click', downloadWorks);
    });
    const wrapper = document.createElement('div');
    const checkbox = document.createElement('input');
    const label = document.createElement('label');
    wrapper.classList.add('pdl-wrap');
    checkbox.id = 'pdl-filter';
    checkbox.type = 'checkbox';
    label.setAttribute('for', 'pdl-filter');
    label.textContent = 'Exclude downloaded';
    wrapper.appendChild(checkbox);
    wrapper.appendChild(label);
    dlBarRef.filter = checkbox;
    nav.parentElement.insertBefore(wrapper, nav);
    nav.appendChild(fragment);
  }
  function createSubBtn(nodes) {
    const isBookmarkPage = regexp.bookmarkPage.test(location.pathname);
    nodes.forEach((e) => {
      if (e.childElementCount !== 0) {
        const illustId = getIllustId(e);
        if (illustId) {
          const attrs = {
            attrs: { 'pdl-id': illustId },
            classList: ['pdl-btn', 'pdl-btn-sub'],
          };
          if (pixivHistory.has(illustId)) attrs.classList.push('pdl-complete');
          if (isBookmarkPage) attrs.classList.push('pdl-btn-sub-bookmark');
          e.appendChild(createPdlBtn(attrs));
        }
      }
    });
  }
  function createMultyWorksBtn(id) {
    const works = document.querySelectorAll("[role='presentation'] > a");
    if (works.length < 2) return;
    const containers = Array.from(works).map((node) => node.parentElement.parentElement);
    if (containers[0].querySelector('.pdl-btn')) return;
    containers.forEach((node, idx) => {
      const wrapper = document.createElement('div');
      wrapper.classList.add('pdl-wrap-artworks');
      const attrs = {
        attrs: { 'pdl-id': id, 'should-download': 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);
      const containers = btn.parentElement;
      const attrs = {
        attrs: {
          'pdl-id': btn.getAttribute('pdl-id'),
          'should-download': pageNum,
        },
        classList: ['pdl-btn', 'pdl-btn-sub', 'presentation'],
      };
      btn.remove();
      btn = createPdlBtn(attrs);
      containers.appendChild(btn);
    }
    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');
      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);
      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');
    let 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': idx },
          classList: ['pdl-btn', 'pdl-btn-sub'],
        };
        node.appendChild(createPdlBtn(attrs));
      });
    }, 300);
  }
  function createTagsBtn(userId, category) {
    const tagsEles = document.querySelectorAll('section> div:nth-child(2) > div > div');
    if (!tagsEles.length) return;
    if (category === 'illustrations' || category === 'artworks') category = 'illusts';
    let rest = 'show';
    if (userId === getOwnerId() && category === 'bookmarks' && location.search.includes('rest=hide'))
      rest = 'hide';
    tagsEles.forEach((ele) => {
      if (ele.querySelector('.pdl-btn')) 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]');
        tag = tagTextEles[tagTextEles.length - 1].getAttribute('title').slice(1);
      }
      const attrs = {
        attrs: { 'pdl-userId': userId, category, 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', downloadWorks);
      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];
      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, tag, rest },
        classList: ['pdl-btn', 'pdl-modal-tag'],
      };
      const dlBtn = createPdlBtn(attrs, '', { addEvent: false });
      dlBtn.addEventListener('click', (evt) => {
        modal.querySelector('svg').parentElement.click();
        downloadWorks(evt);
      });
      ele.appendChild(dlBtn);
    });
  }
  function compatPixivPreviewer(nodes) {
    const isPpSearchPage = regexp.ppSearchPage.test(location.pathname);
    if (!isPpSearchPage) return;
    nodes.forEach((node) => {
      const pdlEle = node.querySelector('.pdl-btn');
      if (!pdlEle) return false;
      pdlEle.remove();
    });
  }
  let firstRun = true;
  function observerCallback(records) {
    const addedNodes = [];
    records.forEach((record) => {
      if (!record.addedNodes.length) return;
      record.addedNodes.forEach((node) => {
        if (
          node.nodeType === Node.ELEMENT_NODE &&
          node.tagName !== 'BUTTON' &&
          node.tagName !== 'IMG'
        ) {
          addedNodes.push(node);
        }
      });
    });
    if (!addedNodes.length) {
      return;
    }
    if (firstRun) {
      createSubBtn(document.querySelectorAll('a'));
      firstRun = false;
    } else {
      compatPixivPreviewer(addedNodes);
      const thunmnails = addedNodes.reduce((prev, current) => {
        return prev.concat(Array.from(current.querySelectorAll('a')));
      }, []);
      createSubBtn(thunmnails);
    }
    const isArtworksPage = regexp.artworksPage.exec(location.pathname);
    const isUserPage = regexp.userPage.exec(location.pathname);
    const isTagsPage = regexp.userPageTags.exec(location.pathname);
    if (isArtworksPage) {
      const id = isArtworksPage[1];
      createMainBtn(id);
      createMultyWorksBtn(id);
      createPresentationBtn(id);
      createPreviewModalBtn();
    } else if (isUserPage) {
      createDownloadBar(isUserPage[1]);
      if (isTagsPage) {
        createTagsBtn(isUserPage[1], isTagsPage[1]);
      }
    }
  }

  addStyle();
  pixivHistory.updateHistory();
  GM_registerMenuCommand('Apng', createSetFormatFn('png'), 'a');
  GM_registerMenuCommand('Gif', createSetFormatFn('gif'), 'g');
  GM_registerMenuCommand('Zip', createSetFormatFn('zip'), 'z');
  GM_registerMenuCommand('Webm', createSetFormatFn('webm'), 'w');
  GM_registerMenuCommand('Webp', createSetFormatFn('webp'), 'p');
  GM_registerMenuCommand('Clear history', pixivHistory.clearHistory.bind(pixivHistory), 'c');
  GM_registerMenuCommand('Edit filename', showFilePathSetting, 'e');
  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'));
          }
        }
      }
    });
  });

})();