Pixiv Downloader

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

As of 2022-10-24. 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 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.5.5
// @description:en  Download the original images of Pixiv pages with one click. Supports:multiple illustrations, ugoira(animation), and batch downloads of artists' work. Ugoira support format conversion: Gif | Apng | 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 | Webm。下载的图片将保存到以画师名命名的单独文件夹(需要调整tampermonkey“下载”设置为“浏览器API”)。保留已下载图片的记录。
// @description:zh-TW  一鍵下載Pixiv各頁面原圖。支持多圖下載,動圖下載,畫師作品批次下載。動圖支持格式轉換:Gif | Apng | 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
// ==/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);
  }
}
.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 {
  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 {
  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 {
  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-stop {
  background-color: transparent;
  border: none;
}
.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;
}`;
  function addStyle() {
    const sty = document.createElement("style");
    sty.innerHTML = style;
    document.head.appendChild(sty);
  }

  function debugLog(...msgs) {
  }

  const defaultSettings = {
    version: "0.5.5",
    ugoriaFormat: "zip",
    folderPattern: "{artist}",
    filenamePattern: "{artist}_{title}_{id}_p{page}",
    tagLang: "ja",
    showMsg: true,
    log: false,
  };
  function migraSettings(settings) {
    if (localStorage.pdlFormat) {
      settings.ugoriaFormat = localStorage.pdlFormat;
      localStorage.removeItem("pdlFormat");
    }
    if (localStorage.pdlFilename) {
      settings.filenamePattern = localStorage.pdlFilename.replace("{author}", "{artist}");
      localStorage.removeItem("pdlFilename");
    }
  }
  function getSettings() {
    let settings;
    if (!localStorage.pdlSetting) {
      settings = defaultSettings;
      migraSettings(settings);
      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);
      }
    };
  }

  const regexp = {
    artworksPage: /artworks\/(\d+)$/,
    userPage: /users\/(\d+)/,
    ppSearchPage: /\/tags\/.*\/(artworks|illustrations|manga)/,
    bookmarkPage: /users\/\d+\/bookmarks\/artworks/,
    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 ${settings.version}</h3>`,
    upgradeMsgContent: `<p>1. 现在支持在文件名中加入标签了,请使用 {tags} 来分辨AI作品吧。搜索想看的图也会更方便。</p><p>2. 标签支持选择语言翻译。但具体取决于Pixiv的数据,没有提供对应语言翻译的标签可能是其他语言。</p><p>请注意:FireFox浏览器 Tampermonkey 升级4.18.0后,需转换格式的动图无法保存到画师目录,请理解。</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">祝你每天都能找到对的色图,就像我每天都能喝到香草味可乐</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>`,
  };

  function initialDeps(urls) {
    return Promise.all([_getGifWS(urls.gifWorker), _getApngWS(urls.pako, urls.upng)]).then(
      ([gif, apng]) => {
        this._deps.gif = URL.createObjectURL(new Blob([gif], { type: "text/javascript" }));
        this._deps.apng = URL.createObjectURL(new Blob([apng], { 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 _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 apngWorkers = [];
    const MAX_CONVERT = 2;
    let queue = [];
    let active = [];
    let isStop = false;
    const convertTo = {
      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 (apngWorkers.length === MAX_CONVERT) {
            worker = freeApngWorkers.shift();
          } else {
            worker = new Worker(this._deps.apng);
            apngWorkers.push(worker);
          }
          convertMeta.abort = convertMeta._baseAbort.bind(null, () => {
            reject("[Info]Convert stop manually, reject when convert apng. " + convertMeta.id);
            worker.terminate();
            apngWorkers.splice(apngWorkers.indexOf(worker), 1);
          });
          worker.onmessage = function (e) {
            if (queue.length) {
              freeApngWorkers.push(worker);
            } else {
              worker.terminate();
              apngWorkers.splice(apngWorkers.indexOf(worker), 1);
            }
            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: "",
    },
    initialDeps,
    createInstance,
  };

  const _isBlobDlAvaliable = !(
    navigator.userAgent.includes("Firefox") &&
    GM_info.scriptHandler === "Tampermonkey" &&
    parseFloat(GM_info.version) > 4.17
  );
  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) {
      save = _normalSave;
    } else {
      debugLog("[Info]Run at firefox && TM version:", GM_info.version);
      save = _ffSave;
    }
    const download = (meta) => {
      debugLog("[Info]Start download:", meta.path);
      active.push(meta);
      let abortObj;
      if (!_isBlobDlAvaliable && !_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();
            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
        ? settings.folderPattern + "/" + settings.filenamePattern
        : settings.filenamePattern;
      return (
        path
          .replaceAll("{author}", user)
          .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 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("[Error]fetch artworksURL failed: " + res.status);
      const htmlText = await res.text();
      const preloadData = JSON.parse(htmlText.match(/"meta-preload-data" content='(.*)'>/)[1]);
      if (!preloadData.illust) throw new Error("[Error]Fail to parse meta preload data.");
      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 ugoiraRes = await fetch(
          "https://www.pixiv.net/ajax/illust/" + illustId + "/ugoira_meta"
        );
        if (!ugoiraRes.ok) throw new Error("[Error]fetch ugoira meta failed: " + res.status);
        const ugoira = await ugoiraRes.json();
        pathInfo.ext = "." + settings.ugoriaFormat;
        metas.push({
          id: illustId,
          illustType: illustType,
          path: getFilePath(pathInfo),
          src: ugoira.body.originalSrc,
          ugoiraMeta: ugoira.body,
        });
      }
      return metas;
    };
    const parseByUser = async (userId, type) => {
      const res = await fetch("https://www.pixiv.net/ajax/user/" + userId + "/profile/all");
      if (!res.ok) throw new Error("fetch user profile failed: " + res.status);
      const profile = await res.json();
      let illustIds;
      if (type) {
        illustIds = Object.keys(profile.body[type]);
      } else {
        illustIds = Object.keys(profile.body.illusts).concat(Object.keys(profile.body.manga));
      }
      return illustIds;
    };
    return {
      id: parseByIllust,
      user: parseByUser,
    };
  }

  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":
          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 handleDownloadAll(userId, type = "") {
    let worksCount = 0,
      worksComplete = 0,
      failed = [];
    let isCanceled = false;
    let metasRecord = [];
    const timers = [];
    const placeholder = document.querySelector(".pdl-nav-placeholder");
    const control = document.querySelector(".pdl-stop");
    const isExcludeDled = document.querySelector("#pdl-filter").checked;
    return new Promise((resolve, reject) => {
      control.onclick = () => {
        isCanceled = true;
        for (const timer of timers) {
          if (timer) clearTimeout(timer);
        }
        if (metasRecord.length) {
          downloader.del(metasRecord);
          converter.del(metasRecord);
          metasRecord = [];
        }
        control.onclick = null;
        reject("Download stopped");
      };
      const onProgressCB = (illustId) => {
        placeholder.textContent = `Downloading: ${++worksComplete} / ${worksCount}`;
        if (worksComplete === worksCount - failed.length) {
          if (failed.length) {
            placeholder.textContent = `Complete. Failed: ${failed.length} / ${worksCount}. See console.`;
            console.log("Failed: ", failed.join(", "));
          } else {
            placeholder.textContent = "Complete";
          }
          resolve();
        }
      };
      placeholder.textContent = "Download...";
      parser
        .user(userId, type)
        .then((illustIds) => {
          if (isCanceled) throw "Download stopped";
          if (isExcludeDled) {
            pixivHistory.updateHistory();
            debugLog("Before filter", illustIds.length);
            illustIds = illustIds.filter((illustId) => !pixivHistory.has(illustId));
            debugLog("After filter", illustIds.length);
          }
          if (!illustIds.length) throw "All Exclude";
          worksCount = illustIds.length;
          illustIds.forEach((illustId, idx) => {
            if (isCanceled) throw "[Warning]Download stopped";
            let timer = setTimeout(() => {
              timer = null;
              parser
                .id(illustId)
                .then((metas) => {
                  if (isCanceled) {
                    throw "[Warning]Download stop manually: " + metas[0].id;
                  }
                  metasRecord = metasRecord.concat(metas);
                  return downloader.add(metas);
                })
                .then((metas) => {
                  pixivHistory.add(illustId);
                  localStorage.setItem(`pdlTemp-${illustId}`, "");
                  if (isCanceled) {
                    return;
                  }
                  metasRecord = metasRecord.filter((meta) => !metas.includes(meta));
                  onProgressCB();
                })
                .catch((err) => {
                  failed.push(illustId);
                });
            }, idx * 600);
            timers.push(timer);
          });
        })
        .catch((err) => {
          reject(err);
        });
    });
  }
  function toggleDlAll(evt) {
    const target = evt.target;
    if (target.classList.contains("pdl-btn-all")) {
      evt.preventDefault();
      evt.stopPropagation();
      const dlBarsBtn = target.parentElement.querySelectorAll("[pdl-userid]");
      const placeholder = document.querySelector(".pdl-nav-placeholder");
      const userId = target.getAttribute("pdl-userid");
      dlBarsBtn.forEach((ele) => {
        ele.classList.toggle("pdl-hide");
      });
      handleDownloadAll(userId, target.getAttribute("pdl-type"))
        .catch((err) => {
          placeholder.textContent = err;
        })
        .finally(() => {
          dlBarsBtn.forEach((ele) => {
            ele.classList.toggle("pdl-hide");
          });
        });
    }
  }

  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;
    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");
    fragment.appendChild(placeholder);
    const baseClasses = nav.querySelector("a:not([aria-current])").classList;
    fragment.appendChild(
      createPdlBtn(
        {
          attrs: { "pdl-userId": userId },
          classList: [...baseClasses, "pdl-stop", "pdl-hide"],
        },
        "Stop",
        { addEvent: false }
      )
    );
    fragment.appendChild(
      createPdlBtn(
        {
          attrs: { "pdl-userId": userId },
          classList: [...baseClasses, "pdl-btn-all"],
        },
        "All",
        { addEvent: false }
      )
    );
    if (nav.querySelector("a[href$=illustrations]") && nav.querySelector("a[href$=manga]")) {
      fragment.appendChild(
        createPdlBtn(
          {
            attrs: { "pdl-userid": userId, "pdl-type": "illusts" },
            classList: [...baseClasses, "pdl-btn-all"],
          },
          "Illusts",
          { addEvent: false }
        )
      );
      fragment.appendChild(
        createPdlBtn(
          {
            attrs: { "pdl-userid": userId, "pdl-type": "manga" },
            classList: [...baseClasses, "pdl-btn-all"],
          },
          "Manga",
          { addEvent: false }
        )
      );
    }
    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);
    nav.parentElement.insertBefore(wrapper, nav);
    nav.appendChild(fragment);
    nav.addEventListener("click", toggleDlAll);
  }
  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 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);
    if (isArtworksPage) {
      const id = isArtworksPage[1];
      createMainBtn(id);
      createMultyWorksBtn(id);
      createPresentationBtn(id);
      createPreviewModalBtn();
    } else if (isUserPage) {
      createDownloadBar(isUserPage[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("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", { bubbles: true }));
          }
        }
      }
    });
  });

})();