// ==UserScript==
// @name Pixiv Downloader
// @namespace https://greasyfork.org/zh-CN/scripts/432150
// @version 0.6.1
// @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-btn-all,
.pdl-stop.pdl-stop {
background-color: transparent;
border: none;
padding: 0 10px;
}
.pdl-btn-all::before,
.pdl-stop::before {
content: '';
height: 24px;
width: 24px;
transition: background-image 0.2s ease 0s;
background: no-repeat center/85%;
}
.pdl-btn-all::before {
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E");
}
.pdl-stop::before {
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%23858585' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E");
}
.pdl-btn-all:hover::before{
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm-32-316v116h-67c-10.7 0-16 12.9-8.5 20.5l99 99c4.7 4.7 12.3 4.7 17 0l99-99c7.6-7.6 2.2-20.5-8.5-20.5h-67V140c0-6.6-5.4-12-12-12h-40c-6.6 0-12 5.4-12 12z'%3E%3C/path%3E %3C/svg%3E");
}
.pdl-stop:hover::before {
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3Cpath fill='%231F1F1F' d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm0 448c-110.5 0-200-89.5-200-200S145.5 56 256 56s200 89.5 200 200-89.5 200-200 200zm101.8-262.2L295.6 256l62.2 62.2c4.7 4.7 4.7 12.3 0 17l-22.6 22.6c-4.7 4.7-12.3 4.7-17 0L256 295.6l-62.2 62.2c-4.7 4.7-12.3 4.7-17 0l-22.6-22.6c-4.7-4.7-4.7-12.3 0-17l62.2-62.2-62.2-62.2c-4.7-4.7-4.7-12.3 0-17l22.6-22.6c4.7-4.7 12.3-4.7 17 0l62.2 62.2 62.2-62.2c4.7-4.7 12.3-4.7 17 0l22.6 22.6c4.7 4.7 4.7 12.3 0 17z'%3E%3C/path%3E %3C/svg%3E");
}
.pdl-hide {
display: none!important;
}
.pdl-wrap {
text-align: right;
padding-right: 24px;
font-weight: bold;
font-size: 14px;
line-height: 14px;
color: rgb(133, 133, 133);
transition: color 0.2s ease 0s;
}
.pdl-wrap:hover {
color: rgb(31, 31, 31);
}
.pdl-wrap label {
padding-left: 8px;
cursor: pointer;
}
.pdl-wrap input {
vertical-align: top;
appearance: none;
position: relative;
box-sizing: border-box;
width: 28px;
border: 2px solid transparent;
cursor: pointer;
border-radius: 14px;
height: 14px;
background-color: rgba(133, 133, 133);
transition: background-color 0.2s ease 0s, box-shadow 0.2s ease 0s;
}
.pdl-wrap input:hover {
background-color: rgba(31, 31, 31);
}
.pdl-wrap input::after {
content: "";
position: absolute;
display: block;
top: 0px;
left: 0px;
width: 10px;
height: 10px;
transform: translateX(0px);
background-color: rgb(255, 255, 255);
border-radius: 10px;
transition: transform 0.2s ease 0s;
}
.pdl-wrap input:checked {
background-color: rgb(0, 150, 250);
}
.pdl-wrap input:checked::after {
transform: translateX(14px);
}
.pdl-wrap-artworks {
position: absolute;
right: 8px;
top: 0px;
bottom: 0px;
margin-top: 40px;
}
.pdl-modal * {
font-family: 'win-bug-omega, system-ui, -apple-system, "Segoe UI", Roboto, Ubuntu, Cantarell, "Noto Sans", "Hiragino Kaku Gothic ProN", Meiryo, sans-serif';
line-height: 1.15;
}
.pdl-modal {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
display: flex;
z-index: 99;
background-color: rgba(0, 0, 0, 0.32);
user-select: none;
}
.pdl-dialog {
position: relative;
background-color: #fff;
border-radius: 24px;
margin: auto;
padding: 20px 40px 30px 40px;
max-width: 720px;
font-size: 16px;
}
.pdl-dialog-header > h3 {
font-weight: bold;
font-size: 1.17em;
margin: 1em 0;
}
.pdl-dialog p {
margin: 1em 0px;
overflow-wrap: break-word;
}
.pdl-dialog-close {
position: absolute;
top: 10px;
right: 10px;
margin: 0;
padding: 0;
width: 25px;
height: 25px;
border: none;
cursor: pointer;
border-radius: 50%;
background-color: transparent;
transform: rotate(45deg);
transition: 0.25s background-color;
background: linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/18px 2px no-repeat,
linear-gradient(rgb(125, 125, 125) 0%, rgb(125, 125, 125) 100%) center/2px 18px no-repeat;
}
.pdl-dialog-close:hover {
background-color: rgba(0, 0, 0, 0.05);
}
.pdl-dialog-content {
user-select: text;
}
.pdl-btn.pdl-tag {
height: auto;
border-top-right-radius: 4px;
border-bottom-right-radius: 4px;
left: -1px;
background-color: rgba(0, 0, 0, 0.12);
transition: background-image 0.5s;
}
.pdl-btn.pdl-tag.pdl-tag-hide{
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E %3C/svg%3E");
pointer-events: none;
}
[data-theme="dark"] .pdl-btn.pdl-tag{
background-color: rgba(255, 255, 255, 0.4);
}
.pdl-btn.pdl-modal-tag {
position: absolute;
right: 60px;
top: 6px;
background-origin: content-box;
border-radius: 4px;
padding: 5px;
width: 42px;
height: 50px;
background-color: rgba(0,0,0,0.04);
transition: .25s background-color;
}
.pdl-btn.pdl-modal-tag:hover {
background-color: rgba(0,0,0,0.12);
}
[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);
}
`;
function addStyle() {
const sty = document.createElement("style");
sty.innerHTML = style;
document.head.appendChild(sty);
}
function debugLog(...msgs) {
}
const defaultSettings = {
version: "0.6.1",
ugoriaFormat: "zip",
folderPattern: "{artist}",
filenamePattern: "{artist}_{title}_{id}_p{page}",
tagLang: "ja",
showMsg: true,
log: false,
};
const regexp = {
artworksPage: /artworks\/(\d+)$/,
userPage: /users\/(\d+)/,
bookmarkPage: /users\/\d+\/bookmarks\/artworks/,
userPageTags: /users\/\d+\/(artworks|illustrations|manga|bookmarks(?!artworks))/,
ppSearchPage: /\/tags\/.*\/(artworks|illustrations|manga)/,
suscribePage: /bookmark_new_illust/,
activityHref: /illust_id=(\d+)/,
originSrcPageNum: /(?<=_p)\d+/,
};
const artworkType = {
ILLUSTS: 0,
MANGA: 1,
UGOIRA: 2,
};
const depsUrls = {
gifWorker: "https://cdnjs.cloudflare.com/ajax/libs/gif.js/0.2.0/gif.worker.js",
pako: "https://cdnjs.cloudflare.com/ajax/libs/pako/2.0.4/pako.min.js",
upng: "https://cdnjs.cloudflare.com/ajax/libs/upng-js/2.1.0/UPNG.min.js",
};
const text = {
upgradeMsgTitle: `<h3>Pixiv Downloader ${defaultSettings.version}</h3>`,
upgradeMsgContent: `<p>1. 现在可以<strong>下载收藏的作品</strong>了! 在画师主页点击<strong>Bookmarks</strong>即可下载。</p><p> 在自己的主页,<strong>Public</strong>可下载公开的收藏,<strong>Private</strong>下载不公开的收藏,<strong>Bookmarks</strong>则表示全都要。</p><p>2. 现在可以<strong>按标签下载作品</strong>了!找到画师或自己主页的标签点击右边下载按钮即可。</p><p>主页的标签太少了吗?点击用户主页插画/漫画的<strong>“高级搜索”</strong>或收藏页标签最后的<strong>“···”</strong>解锁更多标签吧。</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,
};
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 _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();
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 sleep(delay) {
return new Promise((resolve) => {
setTimeout(resolve, delay);
});
}
function getOwnerId() {
return document.querySelector("#qualtrics_user-id")?.textContent;
}
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 getData = async (url) => {
const res = await fetch(url);
if (!res.ok) throw new Error("[Error]fail to fetch:" + url + ", code:" + res.status);
const data = await res.json();
if (data.error) throw new Error("[Error]json return error." + data.message);
return data;
};
const fetchJson = async (url) => {
let json;
let retry = 0;
do {
try {
debugLog("[Info]fetch url:", url);
json = await getData(url);
} catch (error) {
retry++;
if (retry === 3) throw error;
sleep(3000);
}
} while (!json);
return json;
};
const parseByIllust = async (illustId) => {
let params = "";
if (settings.tagLang !== "jp") params = "?lang=" + settings.tagLang;
const res = await fetch("https://www.pixiv.net/artworks/" + illustId + params);
if (!res.ok) throw new Error(res.status);
const htmlText = await res.text();
const matchText = htmlText.match(/"meta-preload-data" content='(.*)'>/);
if (!matchText) throw new Error("[Error]Fail to parse preload data.");
const preloadData = JSON.parse(htmlText.match(/"meta-preload-data" content='(.*)'>/)[1]);
const illustInfo = preloadData.illust[illustId];
const user = replaceInvalidChar(illustInfo.userName) || "userId-" + illustInfo.userId;
const title = replaceInvalidChar(illustInfo.illustTitle) || "illustId-" + illustInfo.illustId;
const tags = replaceInvalidChar(illustInfo.tags.tags.reduce(makeTagsStr, ""));
const illustType = illustInfo.illustType;
let metas = [];
const pathInfo = {
user,
title,
tags,
illustId,
userId: illustInfo.userId,
ext: "",
page: 0,
};
if (illustType === artworkType.ILLUSTS || illustType === artworkType.MANGA) {
const firstImgSrc = illustInfo.urls.original;
const srcPrefix = firstImgSrc.slice(0, firstImgSrc.indexOf("_") + 2);
const srcSuffix = firstImgSrc.slice(-4);
pathInfo.ext = srcSuffix;
for (let i = 0; i < illustInfo.pageCount; i++) {
pathInfo.page = i;
metas.push({
id: illustId,
illustType: illustType,
path: getFilePath(pathInfo),
src: srcPrefix + i + srcSuffix,
});
}
}
if (illustType === artworkType.UGOIRA) {
const ugoira = await fetchJson(
"https://www.pixiv.net/ajax/illust/" + illustId + "/ugoira_meta"
);
pathInfo.ext = "." + settings.ugoriaFormat;
metas.push({
id: illustId,
illustType: illustType,
path: getFilePath(pathInfo),
src: ugoira.body.originalSrc,
ugoiraMeta: ugoira.body,
});
}
return metas;
};
function _filterBookmarks(works) {
const unavaliable = [];
function filterFn(work) {
if (work.isBookmarkable) {
return true;
} else {
unavaliable.push(work.id);
}
}
const avaliable = works.filter(filterFn).map((work) => work.id);
return { avaliable, unavaliable };
}
async function* generateIds(userId, category, tag = "", rest = "show") {
let requestUrl;
if (tag || category === "bookmarks") {
const OFFSET = 48;
if (category !== "bookmarks") {
requestUrl = `https://www.pixiv.net/ajax/user/${userId}/${category}/tag?tag=${tag}&offset=0&limit=${OFFSET}&lang=ja`;
} else {
requestUrl = `https://www.pixiv.net/ajax/user/${userId}/illusts/bookmarks?tag=${tag}&offset=0&limit=${OFFSET}&rest=${rest}&lang=ja`;
}
let head = 0;
const firstPageData = await fetchJson(requestUrl);
const total = firstPageData.body.total;
yield total;
yield _filterBookmarks(firstPageData.body.works);
head += OFFSET;
while (head < total) {
const data = await fetchJson(requestUrl.replace("offset=0", "offset=" + head));
head += OFFSET;
await sleep(3000);
yield _filterBookmarks(data.body.works);
}
} else {
requestUrl = "https://www.pixiv.net/ajax/user/" + userId + "/profile/all";
const profile = await fetchJson(requestUrl);
let illustIds;
if (category !== "both") {
illustIds = Reflect.ownKeys(profile.body[category]);
} else {
illustIds = Reflect.ownKeys(profile.body.illusts).concat(
Reflect.ownKeys(profile.body.manga)
);
}
yield illustIds.length;
yield { avaliable: illustIds, unavaliable: [] };
}
}
return {
id: parseByIllust,
generateIds,
};
}
let converter;
let downloader;
let parser;
async function initial() {
converter = await createConverter
.initialDeps(depsUrls)
.then((createConverter) => createConverter.createInstance());
parser = createParser();
downloader = createDownloader(converter);
}
function add(ele) {
this._records.add(ele);
}
function has(ele) {
return this._records.has(ele);
}
function getHistory() {
const storage = localStorage.pixivDownloader || "[]";
return new Set(JSON.parse(storage));
}
function updateHistory() {
Object.keys(localStorage).forEach((key) => {
const matchResult = /pdlTemp-(\d+)/.exec(key);
if (matchResult) {
this._records.add(matchResult[1]);
localStorage.removeItem(matchResult[0]);
}
});
this.saveHistory();
}
function clearHistory() {
const isConfirm = confirm("Do you really want to clear history?");
if (!isConfirm) return;
this.updateHistory();
this._records = new Set();
localStorage.pixivDownloader = "[]";
}
function saveHistory() {
localStorage.pixivDownloader = JSON.stringify([...this._records]);
}
const pixivHistory = {
_records: getHistory(),
add,
has,
updateHistory,
saveHistory,
clearHistory,
};
function handleDownload(pdlBtn, illustId) {
let pageCount,
pageComplete = 0;
const onProgress = (progress = 0, type = null) => {
if (pageCount > 1) return;
progress = Math.floor(progress * 100);
switch (type) {
case null:
pdlBtn.style.setProperty("--pdl-progress", progress + "%");
case "gif":
case "webm":
pdlBtn.textContent = progress;
break;
case "zip":
pdlBtn.textContent = "";
break;
}
};
const onLoad = function () {
if (pageCount < 2) return;
const progress = Math.floor((++pageComplete / pageCount) * 100);
pdlBtn.textContent = progress;
pdlBtn.style.setProperty("--pdl-progress", progress + "%");
};
pdlBtn.classList.add("pdl-progress");
parser
.id(illustId)
.then((metas) => {
let shouldDownloadPage;
if ((shouldDownloadPage = pdlBtn.getAttribute("should-download"))) {
metas = [metas[shouldDownloadPage]];
}
pageCount = metas.length;
metas.forEach((meta) => {
meta.onProgress = onProgress;
meta.onLoad = onLoad;
});
return downloader.add(metas);
})
.then(() => {
pixivHistory.add(illustId);
localStorage.setItem(`pdlTemp-${illustId}`, "");
pdlBtn.classList.remove("pdl-error");
pdlBtn.classList.add("pdl-complete");
})
.catch((err) => {
if (err) console.log(err);
pdlBtn.classList.remove("pdl-complete");
pdlBtn.classList.add("pdl-error");
})
.finally(() => {
pdlBtn.innerHTML = "";
pdlBtn.style.removeProperty("--pdl-progress");
pdlBtn.classList.remove("pdl-progress");
});
}
function changeDlbarDisplay() {
document.querySelectorAll("nav [pdl-userid]").forEach((ele) => {
ele.classList.toggle("pdl-hide");
});
document.querySelectorAll("section [pdl-userid]").forEach((ele) => {
ele.classList.toggle("pdl-tag-hide");
});
}
function* createRetryGen(failed, unavaliable) {
yield failed.length + unavaliable.length;
yield { avaliable: failed, unavaliable };
}
let isDownloading = false;
let dlBarRef = {};
async function downloadByIds(idsGenerators, retry = true) {
isDownloading = true;
if (!(idsGenerators instanceof Array)) idsGenerators = [idsGenerators];
const { statusBar, abortBtn, filter } = dlBarRef;
const isExcludeDled = filter.checked;
let total = 0,
completed = 0,
failed = [],
unavaliable = [];
let isCanceled = false;
let metasRecord = [];
let tooManyRequests = false;
changeDlbarDisplay();
if (isExcludeDled) pixivHistory.updateHistory();
abortBtn.onclick = () => {
isCanceled = true;
isDownloading = false;
changeDlbarDisplay();
statusBar.textContent = `Stopped. ${completed} / ${total}`;
abortBtn.onclick = null;
if (metasRecord.length) {
downloader.del(metasRecord);
converter.del(metasRecord);
metasRecord = [];
}
};
const onProgressCB = (illustId) => {
statusBar.textContent = `Downloading: ${completed} / ${total}`;
if (completed === total - failed.length - unavaliable.length) {
changeDlbarDisplay();
isDownloading = false;
if (failed.length && retry) {
downloadByIds(createRetryGen(failed, unavaliable), false);
} else {
if (failed.length || unavaliable.length) {
statusBar.textContent = `Failed: ${failed.length + unavaliable.length}. See console.`;
console.log("[Pixiv Downloader]Failed: ", failed.join(", "));
console.log("[Pixiv Downloader]Unavaliable: ", unavaliable.join(", "));
} else {
statusBar.textContent = "Complete";
}
}
}
};
total = await idsGenerators.reduce(async (prev, cur, index) => {
const count = (await cur.next()).value;
return (await prev) + count;
}, 0);
if (total === 0) {
changeDlbarDisplay();
statusBar.textContent = "No Works.";
isDownloading = false;
return;
}
statusBar.textContent = "Downloading...";
try {
for (const idsGenerator of idsGenerators) {
if (isCanceled) break;
for await (const ids of idsGenerator) {
debugLog("[Info]ids:", ids);
if (isCanceled) break;
if (ids.unavaliable.length) {
unavaliable.push(...ids.unavaliable);
debugLog("[Info]unavaliable ids:", unavaliable.length);
}
for (const id of ids.avaliable) {
if (isCanceled) break;
if (isExcludeDled && pixivHistory.has(id)) {
total--;
onProgressCB(id);
continue;
}
if (tooManyRequests) {
debugLog("[Warning]too many requests, wait 30s");
statusBar.textContent = "Too many requests, wait 30s";
await sleep(30000);
tooManyRequests = false;
statusBar.textContent = "Downloading...";
}
parser
.id(id)
.then((metas) => {
if (isCanceled) {
throw "[Warning]Download stop manually: " + metas[0].id;
}
metasRecord = metasRecord.concat(metas);
return downloader.add(metas);
})
.then(
(metas) => {
pixivHistory.add(id);
localStorage.setItem(`pdlTemp-${id}`, "");
if (!isCanceled) {
metasRecord = metasRecord.filter((meta) => !metas.includes(meta));
completed++;
onProgressCB(id);
}
},
(reason) => {
console.log(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);
}
onProgressCB(id);
}
}
);
await sleep(600);
}
}
}
} catch (error) {
console.log(error);
statusBar.textContent = "Error, see console.";
changeDlbarDisplay();
isDownloading = false;
}
}
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;
if (category === "bookmarks" && rest === "all") {
const idsShow = parser.generateIds(userId, category, tag, "show");
const idsHide = parser.generateIds(userId, category, tag, "hide");
downloadByIds([idsShow, idsHide]);
} else {
const idsGenerator = parser.generateIds(userId, category, tag, rest);
downloadByIds(idsGenerator);
}
}
function createModal({ header, content, footer = "" }, option = { closeOnClickModal: true }) {
const modal = document.createElement("div");
const dialog = document.createElement("div");
modal.classList.add("pdl-modal");
dialog.classList.add("pdl-dialog");
if (option.closeOnClickModal) {
dialog.onclick = (e) => {
e.stopPropagation();
};
modal.onclick = () => {
modal.remove();
};
}
dialog.innerHTML = ` <header class="pdl-dialog-header">${header}</header>
<div class="pdl-dialog-content">${content}</div>
<footer class="pdl-dialog-footer">${footer}</footer>`;
const closeBtn = document.createElement("button");
closeBtn.classList.add("pdl-dialog-close");
closeBtn.onclick = () => {
modal.remove();
};
dialog.insertBefore(closeBtn, dialog.firstChild);
modal.appendChild(dialog);
return modal;
}
function showUpgradeMsg() {
document.body.appendChild(
createModal({
header: text.upgradeMsgTitle,
content: text.upgradeMsgContent,
footer: text.modalCreditFooter + text.modalFeedback,
})
);
}
function showFilePathSetting() {
if (document.querySelector("#pdlfolder")) return;
const modal = createModal(
{
header: text.filePathSettingTitle,
content: text.filePathSettingContent,
footer: text.modalOperationBar,
},
{ closeOnClickModal: false }
);
const folder = modal.querySelector("#pdlfolder");
const filename = modal.querySelector("#pdlfilename");
modal.querySelector("#pdlcancel").onclick = () => {
modal.remove();
};
modal.querySelector("#pdlconfirm").onclick = () => {
if (filename.value === "") return;
const folderPattern = folder.value
.trim()
.replace(/^\.|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, "");
const filenamePattern = filename.value
.trim()
.replace(/^\.|[\u200b-\u200f\uFEFF\u202a-\u202e\\/:*?"|<>]/g, "");
if (filenamePattern === "") return;
upgradeSettings("tagLang", modal.querySelector(".tags-option [name='lang']:checked").value);
upgradeSettings("folderPattern", folderPattern);
upgradeSettings("filenamePattern", filenamePattern);
modal.remove();
};
modal.querySelector(`.tags-option [value="${settings.tagLang}"]`).checked = true;
folder.value = settings.folderPattern;
filename.value = settings.filenamePattern;
document.body.appendChild(modal);
}
function getIllustId(node) {
const isLinkToArtworksPage = regexp.artworksPage.exec(node.href);
if (isLinkToArtworksPage) {
if (
node.getAttribute("data-gtm-value") ||
node.classList.contains("gtm-illust-recommend-node-node") ||
node.classList.contains("gtm-discover-user-recommend-node") ||
node.classList.contains("work")
) {
return isLinkToArtworksPage[1];
}
} else {
const isActivityThumb = regexp.activityHref.exec(node.href);
if (isActivityThumb && node.classList.contains("work")) {
return isActivityThumb[1];
}
}
return "";
}
function createPdlBtn(attributes, textContent = "", { addEvent } = { addEvent: true }) {
const ele = document.createElement("button");
ele.textContent = textContent;
if (!attributes) return ele;
const { attrs, classList } = attributes;
if (classList && classList.length > 0) {
for (const cla of classList) {
ele.classList.add(cla);
}
}
if (attrs) {
for (const key in attrs) {
ele.setAttribute(key, attrs[key]);
}
}
if (addEvent) {
ele.addEventListener("click", (evt) => {
evt.preventDefault();
evt.stopPropagation();
const ele = evt.currentTarget;
if (!evt.currentTarget.classList.contains("pdl-progress")) {
handleDownload(ele, ele.getAttribute("pdl-id"));
}
});
}
return ele;
}
function createMainBtn(id) {
if (document.querySelector(".pdl-btn-main")) return;
const handleBar = document.querySelector("main section section");
if (handleBar) {
const pdlBtnWrap = handleBar.lastElementChild.cloneNode();
const attrs = {
attrs: { "pdl-id": id },
classList: ["pdl-btn", "pdl-btn-main"],
};
if (pixivHistory.has(id)) attrs.classList.push("pdl-complete");
pdlBtnWrap.appendChild(createPdlBtn(attrs));
handleBar.appendChild(pdlBtnWrap);
}
}
function createDownloadBar(userId) {
const nav = document.querySelector("nav");
if (!nav || document.querySelector(".pdl-nav-placeholder")) return;
const fragment = document.createDocumentFragment();
const placeholder = document.createElement("div");
placeholder.classList.add("pdl-nav-placeholder");
dlBarRef.statusBar = fragment.appendChild(placeholder);
const baseClasses = nav.querySelector("a:not([aria-current])").classList;
dlBarRef.abortBtn = fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userId": userId },
classList: [...baseClasses, "pdl-stop", "pdl-hide"],
},
"Stop",
{ addEvent: false }
)
);
if (userId !== getOwnerId()) {
if (nav.querySelector("a[href$='illustrations']") && nav.querySelector("a[href$='manga']")) {
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userId": userId, category: "both" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Illusts & Manga",
{ addEvent: false }
)
);
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "illusts" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Illusts",
{ addEvent: false }
)
);
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "manga" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Manga",
{ addEvent: false }
)
);
} else if (nav.querySelector("a[href$='illustrations']")) {
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "illusts" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Illusts",
{ addEvent: false }
)
);
} else if (nav.querySelector("a[href$='manga']")) {
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "manga" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Manga",
{ addEvent: false }
)
);
}
if (nav.querySelector("a[href*='bookmarks']")) {
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "bookmarks" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Bookmarks",
{ addEvent: false }
)
);
}
} else {
if (nav.querySelector("a[href*='bookmarks']")) {
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "bookmarks", rest: "all" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Bookmarks",
{ addEvent: false }
)
);
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "bookmarks", rest: "show" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Public",
{ addEvent: false }
)
);
fragment.appendChild(
createPdlBtn(
{
attrs: { "pdl-userid": userId, category: "bookmarks", rest: "hide" },
classList: [...baseClasses, "pdl-btn-all"],
},
"Private",
{ addEvent: false }
)
);
}
}
fragment.querySelectorAll(".pdl-btn-all").forEach((node) => {
node.addEventListener("click", downloadWorks);
});
const wrapper = document.createElement("div");
const checkbox = document.createElement("input");
const label = document.createElement("label");
wrapper.classList.add("pdl-wrap");
checkbox.id = "pdl-filter";
checkbox.type = "checkbox";
label.setAttribute("for", "pdl-filter");
label.textContent = "Exclude downloaded";
wrapper.appendChild(checkbox);
wrapper.appendChild(label);
dlBarRef.filter = checkbox;
nav.parentElement.insertBefore(wrapper, nav);
nav.appendChild(fragment);
}
function createSubBtn(nodes) {
const isBookmarkPage = regexp.bookmarkPage.test(location.pathname);
nodes.forEach((e) => {
if (e.childElementCount !== 0) {
const illustId = getIllustId(e);
if (illustId) {
const attrs = {
attrs: { "pdl-id": illustId },
classList: ["pdl-btn", "pdl-btn-sub"],
};
if (pixivHistory.has(illustId)) attrs.classList.push("pdl-complete");
if (isBookmarkPage) attrs.classList.push("pdl-btn-sub-bookmark");
e.appendChild(createPdlBtn(attrs));
}
}
});
}
function createMultyWorksBtn(id) {
const works = document.querySelectorAll("[role='presentation'] > a");
if (works.length < 2) return;
const containers = Array.from(works).map((node) => node.parentElement.parentElement);
if (containers[0].querySelector(".pdl-btn")) return;
containers.forEach((node, idx) => {
const wrapper = document.createElement("div");
wrapper.classList.add("pdl-wrap-artworks");
const attrs = {
attrs: { "pdl-id": id, "should-download": idx },
classList: ["pdl-btn", "pdl-btn-sub", "artworks"],
};
wrapper.appendChild(createPdlBtn(attrs));
node.appendChild(wrapper);
});
}
const createPresentationBtn = (() => {
let observer, btn;
function cb(mutationList) {
const newImg = mutationList[1]["addedNodes"][0];
const [pageNum] = regexp.originSrcPageNum.exec(newImg.src);
const containers = btn.parentElement;
const attrs = {
attrs: {
"pdl-id": btn.getAttribute("pdl-id"),
"should-download": pageNum,
},
classList: ["pdl-btn", "pdl-btn-sub", "presentation"],
};
btn.remove();
btn = createPdlBtn(attrs);
containers.appendChild(btn);
}
return (id) => {
const containers = document.querySelector("body > [role='presentation'] > div");
if (!containers) {
if (observer) {
observer.disconnect();
observer = null;
btn = null;
}
return;
}
if (containers.querySelector(".pdl-btn")) return;
const img = containers.querySelector("img");
const isOriginImg = regexp.originSrcPageNum.exec(img.src);
if (!isOriginImg) return;
const [pageNum] = isOriginImg;
const attrs = {
attrs: { "pdl-id": id, "should-download": pageNum },
classList: ["pdl-btn", "pdl-btn-sub", "presentation"],
};
btn = createPdlBtn(attrs);
containers.appendChild(btn);
observer = new MutationObserver(cb);
observer.observe(img.parentElement, { childList: true, subtree: true });
};
})();
function createPreviewModalBtn() {
const illustModalBtn = document.querySelectorAll(".gtm-manga-viewer-preview-modal-open");
const mangaModalBtn = document.querySelectorAll(".gtm-manga-viewer-open-preview");
let mangaViewerModalBtn = document.querySelectorAll(".gtm-manga-viewer-close-icon")?.[1];
if (!illustModalBtn.length && !mangaModalBtn.length) return;
const btns = [...illustModalBtn, ...mangaModalBtn];
if (mangaViewerModalBtn) btns.push(mangaViewerModalBtn);
btns.forEach((node) => {
node.addEventListener("click", handleModalClick);
});
}
function handleModalClick() {
const timer = setInterval(() => {
const ulList = document.querySelectorAll("ul");
const previewList = ulList[ulList.length - 1];
if (getComputedStyle(previewList).display !== "grid") return;
clearInterval(timer);
const [, id] = regexp.artworksPage.exec(location.pathname);
previewList.childNodes.forEach((node, idx) => {
node.style.position = "relative";
const attrs = {
attrs: { "pdl-id": id, "should-download": idx },
classList: ["pdl-btn", "pdl-btn-sub"],
};
node.appendChild(createPdlBtn(attrs));
});
}, 300);
}
function createTagsBtn(userId, category) {
const tagsEles = document.querySelectorAll("section> div:nth-child(2) > div > div");
if (!tagsEles.length) return;
if (category === "illustrations" || category === "artworks") category = "illusts";
let rest = "show";
if (userId === getOwnerId() && category === "bookmarks" && location.search.includes("rest=hide"))
rest = "hide";
tagsEles.forEach((ele) => {
if (ele.querySelector(".pdl-btn")) return;
let tag;
const tagLink = ele.querySelector("a");
if (!tagLink) return;
if (tagLink.getAttribute("status") !== "active") {
if (rest === "hide") {
tag = tagLink.href.slice(tagLink.href.lastIndexOf("/") + 1, tagLink.href.lastIndexOf("?"));
} else {
tag = tagLink.href.slice(tagLink.href.lastIndexOf("/") + 1);
}
} else {
const tagTextEles = ele.querySelectorAll("div[title]");
tag = tagTextEles[tagTextEles.length - 1].getAttribute("title").slice(1);
}
const attrs = {
attrs: { "pdl-userId": userId, category, tag, rest },
classList: ["pdl-btn", "pdl-tag"],
};
if (isDownloading) attrs.classList.push("pdl-tag-hide");
const dlBtn = createPdlBtn(attrs, "", { addEvent: false });
if (!(tagLink.href.includes("bookmarks") && tagLink.getAttribute("status") !== "active")) {
dlBtn.style.backgroundColor = tagLink.getAttribute("color") + "80";
}
dlBtn.addEventListener("click", downloadWorks);
ele.appendChild(dlBtn);
});
let modalTagsEles;
let modal;
if (category === "bookmarks") {
modal = document.querySelector('div[role="presentation"]');
if (!modal) return;
modalTagsEles = modal.querySelectorAll("a");
} else {
const charcoalTokens = document.querySelectorAll(".charcoal-token");
modal = charcoalTokens[charcoalTokens.length - 1];
modalTagsEles = modal.querySelectorAll("a");
}
if (!regexp.userPageTags.exec(modalTagsEles[0]?.href)) return;
modalTagsEles.forEach((ele) => {
if (ele.querySelector(".pdl-btn")) return;
let tag;
if (rest === "hide") {
tag = ele.href.slice(ele.href.lastIndexOf("/") + 1, ele.href.lastIndexOf("?"));
} else {
tag = ele.href.slice(ele.href.lastIndexOf("/") + 1);
}
const attrs = {
attrs: { "pdl-userId": userId, category, tag, rest },
classList: ["pdl-btn", "pdl-modal-tag"],
};
const dlBtn = createPdlBtn(attrs, "", { addEvent: false });
dlBtn.addEventListener("click", (evt) => {
modal.querySelector("svg").parentElement.click();
downloadWorks(evt);
});
ele.appendChild(dlBtn);
});
}
function compatPixivPreviewer(nodes) {
const isPpSearchPage = regexp.ppSearchPage.test(location.pathname);
if (!isPpSearchPage) return;
nodes.forEach((node) => {
const pdlEle = node.querySelector(".pdl-btn");
if (!pdlEle) return false;
pdlEle.remove();
});
}
let firstRun = true;
function observerCallback(records) {
const addedNodes = [];
records.forEach((record) => {
if (!record.addedNodes.length) return;
record.addedNodes.forEach((node) => {
if (
node.nodeType === Node.ELEMENT_NODE &&
node.tagName !== "BUTTON" &&
node.tagName !== "IMG"
) {
addedNodes.push(node);
}
});
});
if (!addedNodes.length) {
return;
}
if (firstRun) {
createSubBtn(document.querySelectorAll("a"));
firstRun = false;
} else {
compatPixivPreviewer(addedNodes);
const thunmnails = addedNodes.reduce((prev, current) => {
return prev.concat(Array.from(current.querySelectorAll("a")));
}, []);
createSubBtn(thunmnails);
}
const isArtworksPage = regexp.artworksPage.exec(location.pathname);
const isUserPage = regexp.userPage.exec(location.pathname);
const isTagsPage = regexp.userPageTags.exec(location.pathname);
if (isArtworksPage) {
const id = isArtworksPage[1];
createMainBtn(id);
createMultyWorksBtn(id);
createPresentationBtn(id);
createPreviewModalBtn();
} else if (isUserPage) {
createDownloadBar(isUserPage[1]);
if (isTagsPage) {
createTagsBtn(isUserPage[1], isTagsPage[1]);
}
}
}
addStyle();
pixivHistory.updateHistory();
GM_registerMenuCommand("Apng", createSetFormatFn("png"), "a");
GM_registerMenuCommand("Gif", createSetFormatFn("gif"), "g");
GM_registerMenuCommand("Zip", createSetFormatFn("zip"), "z");
GM_registerMenuCommand("Webm", createSetFormatFn("webm"), "w");
GM_registerMenuCommand("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 }));
}
}
}
});
});
})();