- // ==UserScript==
- // @name Pixiv Downloader
- // @namespace https://greasyfork.org/zh-CN/scripts/432150
- // @version 0.7.3
- // @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;
- min-width: 500px;
- 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,
- .pdl-btn.pdl-modal-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:not(.pdl-tag-hide):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.3",
- ugoriaFormat: "zip",
- folderPattern: "pixiv/{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 creditCode = `<img style="display: block; margin: 1em auto; width: 200px"
- src=""
- />`;
-
- 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 setFormatFactory(format) {
- return () => {
- if (settings.ugoriaFormat !== format) {
- upgradeSettings('ugoriaFormat', format);
- }
- };
- }
-
- function sleep(delay) {
- return new Promise((resolve) => {
- setTimeout(resolve, delay);
- });
- }
- function getSelfId() {
- return document.querySelector('#qualtrics_user-id')?.textContent;
- }
- const env = {
- isViolentmonkey: GM_info.scriptHandler === 'Violentmonkey',
- isBlobDlAvaliable: !(
- navigator.userAgent.includes('Firefox') &&
- GM_info.scriptHandler === 'Tampermonkey' &&
- parseFloat(GM_info.version) > 4.17
- ),
- isSupportSubpath: GM_info.downloadMode && GM_info.downloadMode === 'browser',
- };
-
- const _isNeedConvert = (meta) => {
- return meta.illustType === artworkType.UGOIRA && settings.ugoriaFormat !== 'zip';
- };
- const _saveWithoutSubpath = (blob, meta) => {
- const dlEle = document.createElement('a');
- dlEle.href = URL.createObjectURL(blob);
- dlEle.download = meta.path;
- dlEle.click();
- URL.revokeObjectURL(dlEle.href);
- meta.resolve(meta);
- };
- const _saveWithSubpath = (blob, meta) => {
- const imgUrl = URL.createObjectURL(blob);
- const request = {
- url: imgUrl,
- name: meta.path,
- onerror: (error) => {
- console.log('[pixiv downloader]Error when saving', meta.path);
- URL.revokeObjectURL(imgUrl);
- meta.reject && meta.reject(error);
- },
- onload: () => {
- if (typeof meta.onLoad === 'function') meta.onLoad();
- URL.revokeObjectURL(imgUrl);
- meta.resolve(meta);
- },
- };
- 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 (env.isBlobDlAvaliable && env.isSupportSubpath) {
- save = _saveWithSubpath;
- } else {
- debugLog('[Info]scriptHandler:', GM_info.scriptHandler, GM_info.version);
- save = _saveWithoutSubpath;
- }
- const download = (meta) => {
- debugLog('[Info]Start download:', meta.path);
- active.push(meta);
- let abortObj;
- if ((!env.isBlobDlAvaliable || env.isViolentmonkey) && !_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 && env.isSupportSubpath
- ? 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);
- }
-
- const lang =
- document.documentElement.getAttribute("lang").toLowerCase() || "en";
- const i18nLib = {
- en: {
- illusts: "Illusts",
- manga: "Manga",
- illusts_manga: "Illusts & Manga",
- bookmarks: "Bookmarks",
- bookmarks_public: "Public",
- bookmarks_private: "Private",
- exclude_downloaded: "Exclude downloaded",
- stop: "Stop",
- edit_filename: "Edit filename",
- clear_history: "Clear history",
- clear_history_tips: "Do you really want to clear history?",
- feedback: "Feedback",
- modal_cancel: "Cancel",
- modal_confirm: "OK",
- tags_lang: "Tags language: ",
- tags_tips: "{artist}, {artistID}, {title}, {id}, {page}, {tags}",
- tags_tips2:
- 'Note: Tags language may not be the language you selected, <a href="https://crowdin.com/project/pixiv-tags" target="_blank">some tags without translations</a> may still be in other languages.',
- folder: "Folder:",
- folder_tips: "I don't need subfolder",
- folder_tips2:
- "If you don't need a subfolder, just leave the folder name blank",
- folder_vm_tips: "VM doesn't support",
- folder_api_tips: "Need Browser Api",
- filename: "FileName:",
- filename_tips: "Your Name?",
- },
- "zh-cn": {
- illusts: "插画",
- manga: "漫画",
- illusts_manga: "插画 & 漫画",
- bookmarks: "收藏",
- bookmarks_public: "公开",
- bookmarks_private: "不公开",
- exclude_downloaded: "排除已下载图片",
- stop: "停止",
- edit_filename: "编辑文件名",
- clear_history: "清除下载历史",
- clear_history_tips: "真的要清除历史吗?",
- feedback: "有问题or想建议?这里反馈",
- modal_cancel: "取消",
- modal_confirm: "确认",
- tags_lang: "标签语言:",
- tags_tips:
- "{artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}:作品pixiv ID, {page}:页码, {tags}:作品标签。",
- tags_tips2:
- '请注意:标签翻译不一定是你选择的语言,部分<a href="https://crowdin.com/project/pixiv-tags" target="_blank">无对应语言翻译的标签</a>仍可能是其他语言。',
- folder: "文件夹名:",
- folder_tips: "我不想保存到子文件夹",
- folder_tips2: "如果不想保存到画师目录,文件夹名留空即可。",
- folder_vm_tips: "Violentmonkey不支持",
- folder_api_tips: "需要Browser Api",
- filename: "文件名:",
- filename_tips: "你的名字?",
- },
- };
- i18nLib.en = Object.create(
- i18nLib["zh-cn"],
- Object.getOwnPropertyDescriptors(i18nLib.en)
- );
- i18nLib.ja = Object.create(i18nLib.en);
- i18nLib.ko = Object.create(i18nLib.en);
- i18nLib["zh-tw"] = Object.create(i18nLib["zh-cn"]);
- i18nLib.zh = i18nLib["zh-cn"];
- const i18n = (key) =>
- i18nLib[lang]?.[key] || `i18n[${lang}][${key}] not found`;
- const modalHtml = {
- upgradeMsgTitle: `<h3>Pixiv Downloader ${defaultSettings.version}</h3>`,
- upgradeMsgContent: `<p>增加导出 / 导入下载记录的功能。</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>
- ${creditCode}
- <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">${i18n(
- "feedback"
- )}</a>`,
- filePathSettingTitle: `<h3>${i18n("edit_filename")}</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">${i18n(
- "folder"
- )}</label>
- <input type="text" id="pdlfolder" style="width: 200px;" maxlength='100'>
- </div>
- <div>
- <label style="display: block; cursor: default;" for="pdlfilename">${i18n(
- "filename"
- )}</label>
- <input type="text" id="pdlfilename" style="width: 300px;" placeholder="${i18n(
- "filename_tips"
- )}" required maxlength='100'>
- </div>
- </div>
- <div class="tags-option" style="margin: 0.7em 0;">
- <span>${i18n("tags_lang")}</span>
- <input type="radio" name="lang" id="lang_ja" value="ja"/>
- <label for="lang_ja">日本語(default)</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">
- ${i18n("tags_tips")}
- </p>
- <p style="font-size: 14px; margin: 0.5em 0">${i18n("folder_tips2")}</p>
- <p style="font-size: 14px; margin: 0.5em 0">${i18n("tags_tips2")}</p>
- </div>`,
- 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">${i18n(
- "modal_cancel"
- )}</button><button id="pdlconfirm" style="border-color: #01b468; background-color: #01b468; color: #fff;">${i18n(
- "modal_confirm"
- )}</button></div>`,
- };
-
- function add(pixivId) {
- this._records.add(pixivId);
- localStorage.setItem(`pdlTemp-${pixivId}`, "");
- }
- function has(pixivId) {
- return this._records.has(pixivId);
- }
- function getHistory() {
- const storage = localStorage.pixivDownloader || "[]";
- return new Set(JSON.parse(storage));
- }
- function updateHistory() {
- Object.keys(localStorage).forEach((key) => {
- const matchResult = /pdlTemp-(\d+)/.exec(key);
- if (matchResult) {
- this._records.add(matchResult[1]);
- localStorage.removeItem(matchResult[0]);
- }
- });
- this.saveHistory();
- }
- function clearHistory() {
- const isConfirm = confirm(i18n("clear_history_tips"));
- if (!isConfirm) return;
- this.updateHistory();
- this._records = new Set();
- localStorage.pixivDownloader = "[]";
- }
- function saveHistory(historyArr) {
- if (historyArr instanceof Array) {
- localStorage.pixivDownloader = JSON.stringify(historyArr);
- } else {
- 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);
- 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;
- const done = new Promise((r) => {
- resolve = r;
- });
- const abort = (msg) => {
- resolve();
- return Promise.reject(msg);
- };
- 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 = [];
- }
- };
- 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 abort(`Stopped. ${completed} / ${total}`);
- 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 abort(`Stopped. ${completed} / ${total}`);
- 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);
- 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.error(error);
- return abort('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: modalHtml.upgradeMsgTitle,
- content: modalHtml.upgradeMsgContent,
- footer: modalHtml.modalCreditFooter + modalHtml.modalFeedback,
- })
- );
- }
- function showFilePathSetting() {
- if (document.querySelector("#pdlfolder")) return;
- const modal = createModal(
- {
- header: modalHtml.filePathSettingTitle,
- content: modalHtml.filePathSettingContent,
- footer: modalHtml.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
- .split("/")
- .map((path) =>
- path
- .trim()
- .replace(/^\.+|\.+$|[\u200b-\u200f\uFEFF\u202a-\u202e\\:*?"|<>]/g, "")
- )
- .filter((path) => !!path)
- .join("/");
- 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;
- filename.value = settings.filenamePattern;
- folder.value = !env.isSupportSubpath
- ? folder.setAttribute("disabled", "") || ""
- : settings.folderPattern;
- folder.placeholder = env.isViolentmonkey
- ? i18n("folder_vm_tips")
- : !env.isSupportSubpath
- ? i18n("folder_api_tips")
- : i18n("folder_tips");
- 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"],
- },
- i18n("stop"),
- { addEvent: false }
- )
- );
- if (userId !== getSelfId()) {
- 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"],
- },
- i18n("illusts_manga"),
- { addEvent: false }
- )
- );
- fragment.appendChild(
- createPdlBtn(
- {
- attrs: { "pdl-userid": userId, category: "illusts" },
- classList: [...baseClasses, "pdl-btn-all"],
- },
- i18n("illusts"),
- { addEvent: false }
- )
- );
- fragment.appendChild(
- createPdlBtn(
- {
- attrs: { "pdl-userid": userId, category: "manga" },
- classList: [...baseClasses, "pdl-btn-all"],
- },
- i18n("manga"),
- { addEvent: false }
- )
- );
- } else if (nav.querySelector("a[href$='illustrations']")) {
- fragment.appendChild(
- createPdlBtn(
- {
- attrs: { "pdl-userid": userId, category: "illusts" },
- classList: [...baseClasses, "pdl-btn-all"],
- },
- i18n("illusts"),
- { addEvent: false }
- )
- );
- } else if (nav.querySelector("a[href$='manga']")) {
- fragment.appendChild(
- createPdlBtn(
- {
- attrs: { "pdl-userid": userId, category: "manga" },
- classList: [...baseClasses, "pdl-btn-all"],
- },
- i18n("manga"),
- { addEvent: false }
- )
- );
- }
- if (nav.querySelector("a[href*='bookmarks']")) {
- fragment.appendChild(
- createPdlBtn(
- {
- attrs: { "pdl-userid": userId, category: "bookmarks" },
- classList: [...baseClasses, "pdl-btn-all"],
- },
- i18n("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"],
- },
- i18n("bookmarks"),
- { addEvent: false }
- )
- );
- fragment.appendChild(
- createPdlBtn(
- {
- attrs: {
- "pdl-userid": userId,
- category: "bookmarks",
- rest: "show",
- },
- classList: [...baseClasses, "pdl-btn-all"],
- },
- i18n("bookmarks_public"),
- { addEvent: false }
- )
- );
- fragment.appendChild(
- createPdlBtn(
- {
- attrs: {
- "pdl-userid": userId,
- category: "bookmarks",
- rest: "hide",
- },
- classList: [...baseClasses, "pdl-btn-all"],
- },
- i18n("bookmarks_private"),
- { addEvent: false }
- )
- );
- }
- }
- fragment.querySelectorAll(".pdl-btn-all").forEach((node) => {
- node.addEventListener("click", 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 = i18n("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 === getSelfId() &&
- 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"],
- };
- if (isDownloading) attrs.classList.push("pdl-tag-hide");
- 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]);
- }
- }
- }
-
- function getHistoryStr() {
- return JSON.stringify([...pixivHistory._records]);
- }
- function createImportModal() {
- if (document.querySelector("#pdlfolder")) return;
- const html = {
- header: "<h3>导入 / 导出</h3>",
- content: `<p>
- <p><label for="pdl-import">选择要导入的文件:</label></p>
- <input type="file" id="pdl-import" accept=".txt">
- </p>
- <p>
- <hr style="border-top:solid 1px grey">
- <p>
- <button id="pdl-output">导出记录</button>
- </p>`,
- };
- const modal = createModal(html);
- const file = modal.querySelector("#pdl-import");
- file.addEventListener("change", () => {
- const file = modal.querySelector("#pdl-import").files[0];
- if (file && file.type === "text/plain") {
- const reader = new FileReader();
- reader.readAsText(file);
- reader.onload = (evt) => {
- const text = evt.target.result;
- try {
- const history = JSON.parse(text);
- if (!(history instanceof Array)) throw new Error("Invalid file");
- if (history.length) {
- if (!history.every((id) => typeof id === "string"))
- throw new Error("Invalid file");
- }
- pixivHistory.saveHistory(history);
- location.reload();
- } catch (error) {
- alert(error.message);
- }
- };
- } else {
- alert("Invalid file");
- }
- });
- const btn = modal.querySelector("#pdl-output");
- btn.addEventListener("click", () => {
- const dlEle = document.createElement("a");
- const history = getHistoryStr();
- dlEle.href = URL.createObjectURL(
- new Blob([history], { type: "text/plain" })
- );
- dlEle.download = "Pixiv Downloader " + new Date().toLocaleString();
- dlEle.click();
- URL.revokeObjectURL(dlEle.href);
- });
- document.body.appendChild(modal);
- }
-
- addStyle();
- pixivHistory.updateHistory();
- GM_registerMenuCommand("Apng", setFormatFactory("png"), "a");
- GM_registerMenuCommand("Gif", setFormatFactory("gif"), "g");
- GM_registerMenuCommand("Zip", setFormatFactory("zip"), "z");
- GM_registerMenuCommand("Webm", setFormatFactory("webm"), "w");
- GM_registerMenuCommand("Webp", setFormatFactory("webp"), "p");
- GM_registerMenuCommand(
- i18n("clear_history"),
- pixivHistory.clearHistory.bind(pixivHistory),
- "c"
- );
- GM_registerMenuCommand(i18n("edit_filename"), showFilePathSetting, "e");
- GM_registerMenuCommand("导入/导出", createImportModal, "i");
- initial()
- .then(() => {
- if (settings.showMsg) {
- showUpgradeMsg();
- upgradeSettings("showMsg", false);
- }
- new MutationObserver(observerCallback).observe(document.body, {
- childList: true,
- subtree: true,
- });
- document.addEventListener("keydown", (e) => {
- if (e.ctrlKey && e.key === "q") {
- const pdlMainBtn = document.querySelector(".pdl-btn-main");
- if (pdlMainBtn) {
- e.preventDefault();
- if (!e.repeat) {
- pdlMainBtn.dispatchEvent(new MouseEvent("click"));
- }
- }
- }
- });
- })
- .catch(console.error);
-
- })();