Sleazy Fork is available in English.

Pixiv Downloader

Pixiv | Danbooru | Rule34. 一键下载各页面原图。批量下载画师作品,按作品标签下载。转换动图格式:Gif | Apng | Webp | Webm。自定义图片文件名,保存路径。保留 / 导出下载历史。

// ==UserScript==
// @name         Pixiv Downloader
// @namespace    https://greasyfork.org/zh-CN/scripts/432150
// @version      0.11.1
// @description:en  Pixiv | Danbooru | Rule34. Download artworks with one click. Batch download artworks or download by tags. Convert ugoira formats: Gif | Apng | Webp | Webm. Customize image file name, save path. Save / export download history.
// @description  Pixiv | Danbooru | Rule34. 一键下载各页面原图。批量下载画师作品,按作品标签下载。转换动图格式:Gif | Apng | Webp | Webm。自定义图片文件名,保存路径。保留 / 导出下载历史。
// @description:zh-TW  Pixiv | Danbooru | Rule34. 一鍵下載各頁面原圖。批次下載畫師作品,按作品標籤下載。轉換動圖格式:Gif | Apng | Webp | Webm。自定義圖片檔名,儲存路徑。保留 / 匯出下載歷史。
// @author       ruaruarua
// @match        https://www.pixiv.net/*
// @match        https://rule34.xxx/*
// @match        https://danbooru.donmai.us/*
// @icon         https://www.pixiv.net/favicon.ico
// @noframes
// @grant        unsafeWindow
// @grant        GM_xmlhttpRequest
// @grant        GM_addStyle
// @grant        GM_download
// @grant        GM_info
// @grant        GM_registerMenuCommand
// @grant        GM_getResourceText
// @connect      i.pximg.net
// @connect      rule34.xxx
// @connect      donmai.us
// @require      https://unpkg.com/jszip@3.7.1/dist/jszip.min.js
// @require      https://unpkg.com/gif.js@0.2.0/dist/gif.js
// @require      https://unpkg.com/dayjs@1.11.11/dayjs.min.js
// @require      https://unpkg.com/dexie@3.2.5/dist/dexie.min.js
// @require      https://greasyfork.org/scripts/455256-toanimatedwebp/code/toAnimatedWebp.js?version=1120088
// @resource pako https://unpkg.com/pako@2.0.4/dist/pako.min.js
// @resource upng https://unpkg.com/upng-js@2.1.0/UPNG.js
// @resource gifWorker https://unpkg.com/gif.js@0.2.0/dist/gif.worker.js
// ==/UserScript==


(function (Dexie, dayjs, JSZip, GIF, workerChunk) {
    'use strict';

    function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }

    var Dexie__default = /*#__PURE__*/_interopDefaultLegacy(Dexie);
    var dayjs__default = /*#__PURE__*/_interopDefaultLegacy(dayjs);
    var JSZip__default = /*#__PURE__*/_interopDefaultLegacy(JSZip);
    var GIF__default = /*#__PURE__*/_interopDefaultLegacy(GIF);
    var workerChunk__default = /*#__PURE__*/_interopDefaultLegacy(workerChunk);

    function getLogger() {
        const methods = ['info', 'warn', 'error'];
        const style = ['color: green;', 'color: orange;', 'color: red;'];
        const logLevel = 2 ;
        const namePrefix = '[Pixiv Downlaoder] ';
        function log(level, args) {
            if (logLevel <= level)
                console[methods[level]]('%c[Pixiv Downloader]', style[level], ...args);
        }
        return {
            info(...args) {
                log(0 , args);
            },
            warn(...args) {
                log(1 , args);
            },
            error(...args) {
                log(2 , args);
            },
            time(label) {
                console.time(namePrefix + label);
            },
            timeLog(label) {
                console.timeLog(namePrefix + label);
            },
            timeEnd(label) {
                console.timeEnd(namePrefix + label);
            }
        };
    }
    const logger = getLogger();

    function sleep(delay) {
        return new Promise((resolve) => {
            setTimeout(resolve, delay);
        });
    }
    function wakeableSleep(delay) {
        let wake = () => void {};
        const sleep = new Promise((r) => {
            setTimeout(r, delay);
            wake = r;
        });
        return {
            wake,
            sleep
        };
    }
    function replaceInvalidChar(str) {
        if (typeof str !== 'string')
            throw new TypeError('expect string but got ' + typeof str);
        if (!str)
            return '';
        return str
            .replace(/\p{C}/gu, '')
            .replace(/\\/g, '\')
            .replace(/\//g, '/')
            .replace(/:/g, ':')
            .replace(/\*/g, '*')
            .replace(/\?/g, '?')
            .replace(/\|/g, '|')
            .replace(/"/g, '"')
            .replace(/</g, '﹤')
            .replace(/>/g, '﹥')
            .replace(/~/g, '~')
            .trim()
            .replace(/^\.|\.$/g, '.');
    }
    function unescapeHtml(str) {
        if (typeof str !== 'string')
            throw new TypeError('expect string but got ' + typeof str);
        if (!str)
            return '';
        const el = document.createElement('p');
        el.innerHTML = str;
        return el.innerText;
    }
    function stringToFragment(string) {
        const renderer = document.createElement('template');
        renderer.innerHTML = string;
        return renderer.content;
    }
    function generateCsv(sheetData) {
        const sheetStr = sheetData
            .map((row) => {
            return row
                .map((cell) => {
                return '"' + cell.replace(/"/g, '""') + '"';
            })
                .join(',');
        })
            .join('\r\n');
        return new Blob(['\ufeff' + sheetStr], { type: 'text/csv' });
    }
    function evalScript(script) {
        const el = document.createElement('script');
        el.text = script;
        document.head.appendChild(el).parentNode.removeChild(el);
    }

    class RequestError extends Error {
        response;
        constructor(message, response) {
            super(message);
            this.name = 'RequestError';
            this.response = response;
        }
    }
    class CancelError extends Error {
        constructor() {
            super('User aborted');
            this.name = 'CancelError';
        }
    }
    class JsonDataError extends Error {
        constructor(msg) {
            super(msg);
            this.name = 'JsonDataError';
        }
    }

    const env = {
        isFirefox() {
            return navigator.userAgent.includes('Firefox');
        },
        isViolentmonkey() {
            return GM_info.scriptHandler === 'Violentmonkey';
        },
        isTampermonkey() {
            return GM_info.scriptHandler === 'Tampermonkey';
        },
        isBlobDlAvaliable() {
            return !this.isFirefox() || (this.isFirefox() && this.isTampermonkey() && parseFloat(GM_info.version ?? '') < 4.18);
        },
        isSupportSubpath() {
            return this.isBrowserDownloadMode();
        },
        isBrowserDownloadMode() {
            return GM_info.downloadMode === 'browser';
        },
        isConflictActionEnable() {
            return this.isTampermonkey() && parseFloat(GM_info.version ?? '') >= 4.18 && this.isBrowserDownloadMode();
        },
        isConflictActionPromptEnable() {
            return !this.isFirefox() && this.isConflictActionEnable();
        },
        isFileSystemAccessAvaliable() {
            return (typeof unsafeWindow.showDirectoryPicker === 'function' && typeof unsafeWindow.showSaveFilePicker === 'function');
        },
        isPixiv() {
            return location.hostname === 'www.pixiv.net';
        }
    };

    class FileSystemAccessHandler {
        filenameConflictAction = 'uniquify';
        updateDirHandleChannel;
        dirHandle = undefined;
        dirHandleStatus = 0 ;
        cachedTasks = [];
        duplicateFilenameCached = {};
        constructor(channelName) {
            this.updateDirHandleChannel = new BroadcastChannel(channelName);
            this.updateDirHandleChannel.onmessage = (evt) => {
                const data = evt.data;
                switch (data.kind) {
                    case 1 :
                        this.dirHandleStatus = 1 ;
                        logger.info('正在选择目录');
                        break;
                    case 0 :
                        logger.warn('取消更新dirHandle');
                        if (this.dirHandle) {
                            this.dirHandleStatus = 2 ;
                            this.processCachedTasks();
                        }
                        else {
                            this.dirHandleStatus = 0 ;
                            this.rejectCachedTasks();
                        }
                        break;
                    case 2 :
                        this.dirHandleStatus = 2 ;
                        this.dirHandle = data.handle;
                        logger.info('更新dirHandle', this.dirHandle);
                        this.processCachedTasks();
                        break;
                    case 'request':
                        if (this.dirHandle) {
                            this.updateDirHandleChannel.postMessage({
                                kind: 'response',
                                handle: this.dirHandle
                            });
                            logger.info('响应请求dirHandle');
                        }
                        break;
                    case 'response':
                        if (!this.dirHandle) {
                            if (this.dirHandleStatus === 0 )
                                this.dirHandleStatus = 2 ;
                            this.dirHandle = data.handle;
                            logger.info('首次获取dirHandle', this.dirHandle);
                        }
                        break;
                    default:
                        throw new Error('Invalid data kind.');
                }
            };
            this.updateDirHandleChannel.postMessage({ kind: 'request' });
        }
        async getDirHandleRecursive(dirs) {
            if (!this.dirHandle)
                throw new Error('未选择保存文件夹');
            let handler = this.dirHandle;
            if (typeof dirs === 'string') {
                if (dirs.indexOf('/') === -1)
                    return await handler.getDirectoryHandle(dirs, { create: true });
                dirs = dirs.split('/').filter((dir) => !!dir);
            }
            for await (const dir of dirs) {
                handler = await handler.getDirectoryHandle(dir, { create: true });
            }
            return handler;
        }
        processCachedTasks() {
            const { length } = this.cachedTasks;
            for (let i = 0; i < length; i++) {
                const [blob, downloadMeta, onSaveFullfilled, onSaveRejected] = this.cachedTasks[i];
                this.saveFile(blob, downloadMeta).then(onSaveFullfilled, onSaveRejected);
            }
            logger.info(`执行${length}个缓存任务`);
            if (this.cachedTasks.length > length) {
                this.cachedTasks = this.cachedTasks.slice(length);
            }
            else {
                this.cachedTasks.length = 0;
            }
        }
        rejectCachedTasks() {
            this.cachedTasks.forEach(([, , , onSaveRejected]) => onSaveRejected(new CancelError()));
            this.cachedTasks.length = 0;
            logger.info(`取消${this.cachedTasks.length}个缓存任务`);
        }
        async getFilenameHandle(dirHandle, filename) {
            if (this.filenameConflictAction === 'overwrite')
                return await dirHandle.getFileHandle(filename, { create: true });
            if (!(filename in this.duplicateFilenameCached)) {
                this.duplicateFilenameCached[filename] = [];
                try {
                    await dirHandle.getFileHandle(filename);
                    logger.warn('存在同名文件', filename);
                }
                catch (error) {
                    return await dirHandle.getFileHandle(filename, { create: true });
                }
            }
            const extIndex = filename.lastIndexOf('.');
            const ext = filename.slice(extIndex + 1);
            const name = filename.slice(0, extIndex);
            if (this.filenameConflictAction === 'prompt') {
                return await unsafeWindow.showSaveFilePicker({
                    suggestedName: filename,
                    types: [{ description: 'Image file', accept: { ['image/' + ext]: ['.' + ext] } }]
                });
            }
            else {
                for (let suffix = 1; suffix < 1000; suffix++) {
                    const newName = `${name} (${suffix}).${ext}`;
                    try {
                        await dirHandle.getFileHandle(newName);
                    }
                    catch (error) {
                        if (this.duplicateFilenameCached[filename].includes(newName)) {
                            continue;
                        }
                        else {
                            this.duplicateFilenameCached[filename].push(newName);
                        }
                        logger.info('使用文件名:', newName);
                        return await dirHandle.getFileHandle(newName, { create: true });
                    }
                }
                throw new RangeError('Oops, you have too many duplicate files.');
            }
        }
        clearFilenameCached(duplicateName, actualName) {
            if (!(duplicateName in this.duplicateFilenameCached))
                return;
            const usedNameArr = this.duplicateFilenameCached[duplicateName];
            logger.info('清理重名文件名', usedNameArr, actualName);
            if (usedNameArr.length === 0) {
                delete this.duplicateFilenameCached[duplicateName];
                return;
            }
            const index = usedNameArr.indexOf(actualName);
            if (index === -1)
                return;
            usedNameArr.splice(index, 1);
            if (usedNameArr.length === 0)
                delete this.duplicateFilenameCached[duplicateName];
        }
        async updateDirHandle() {
            try {
                this.dirHandleStatus = 1 ;
                this.updateDirHandleChannel.postMessage({ kind: 1  });
                this.dirHandle = await unsafeWindow.showDirectoryPicker({ id: 'pdl', mode: 'readwrite' });
                logger.info('更新dirHandle', this.dirHandle);
                this.dirHandleStatus = 2 ;
                this.updateDirHandleChannel.postMessage({
                    kind: 2 ,
                    handle: this.dirHandle
                });
                this.processCachedTasks();
                return true;
            }
            catch (error) {
                logger.warn(error);
                this.updateDirHandleChannel.postMessage({ kind: 0  });
                if (this.dirHandle) {
                    this.dirHandleStatus = 2 ;
                    this.processCachedTasks();
                }
                else {
                    this.dirHandleStatus = 0 ;
                    this.rejectCachedTasks();
                }
                return false;
            }
        }
        getCurrentDirName() {
            return this.dirHandle?.name ?? '';
        }
        isDirHandleNotSet() {
            return this.dirHandleStatus === 0 ;
        }
        setFilenameConflictAction(action) {
            this.filenameConflictAction = action;
        }
        async saveFile(blob, downloadMeta) {
            if (downloadMeta.isAborted)
                throw new CancelError();
            if (this.dirHandleStatus === 1 ) {
                let onSaveFullfilled;
                let onSaveRejected;
                const promiseExcutor = new Promise((resolve, reject) => {
                    onSaveFullfilled = resolve;
                    onSaveRejected = reject;
                });
                this.cachedTasks.push([blob, downloadMeta, onSaveFullfilled, onSaveRejected]);
                return promiseExcutor;
            }
            if (this.dirHandleStatus === 0 ) {
                const isSuccess = await this.updateDirHandle();
                if (!isSuccess)
                    throw new TypeError('Failed to get dir handle.');
            }
            let currenDirHandle;
            let filename;
            const path = downloadMeta.config.path;
            const index = path.lastIndexOf('/');
            if (index === -1) {
                filename = path;
                currenDirHandle = this.dirHandle;
            }
            else {
                filename = path.slice(index + 1);
                currenDirHandle = await this.getDirHandleRecursive(path.slice(0, index));
            }
            if (downloadMeta.isAborted)
                throw new CancelError();
            const fileHandle = await this.getFilenameHandle(currenDirHandle, filename);
            const writableStream = await fileHandle.createWritable();
            await writableStream.write(blob);
            await writableStream.close();
            this.clearFilenameCached(filename, fileHandle.name);
        }
    }
    const fsaHandler = new FileSystemAccessHandler('update_dir_channel');

    function gmDownload(blob, downloadMeta) {
        return new Promise((resolve, reject) => {
            if (downloadMeta.isAborted)
                return reject(new CancelError());
            const imgUrl = URL.createObjectURL(blob);
            const request = {
                url: URL.createObjectURL(blob),
                name: downloadMeta.config.path,
                onerror: (error) => {
                    URL.revokeObjectURL(imgUrl);
                    if (downloadMeta.isAborted) {
                        resolve();
                    }
                    else {
                        reject(new Error(`FileSave error: ${downloadMeta.config.path} because ${error.error} ${error.details ?? ''} `));
                    }
                },
                onload: () => {
                    URL.revokeObjectURL(imgUrl);
                    resolve();
                }
            };
            downloadMeta.abort = GM_download(request).abort;
        });
    }

    function aDownload(blob, downloadMeta) {
        if (downloadMeta.isAborted)
            return Promise.reject(new CancelError());
        let path = downloadMeta.config.path;
        const separaterIndex = path.lastIndexOf('/');
        if (separaterIndex !== -1)
            path = path.slice(separaterIndex + 1);
        const dlEle = document.createElement('a');
        dlEle.href = URL.createObjectURL(blob);
        dlEle.download = path;
        dlEle.click();
        URL.revokeObjectURL(dlEle.href);
        return Promise.resolve();
    }

    function loadConfig(siteConfig) {
        const defaultConfig = Object.freeze({
            version: "0.11.1",
            ugoiraFormat: 'zip',
            folderPattern: 'pixiv/{artist}',
            filenamePattern: '{artist}_{title}_{id}_p{page}',
            tagLang: 'ja',
            showMsg: true,
            filterExcludeDownloaded: false,
            filterIllusts: true,
            filterManga: true,
            filterUgoira: true,
            bundleIllusts: false,
            bundleManga: false,
            addBookmark: false,
            addBookmarkWithTags: false,
            privateR18: false,
            useFileSystemAccess: false,
            fileSystemFilenameConflictAction: 'uniquify',
            showPopupButton: true,
            ...siteConfig
        });
        let config;
        if (!localStorage.pdlSetting) {
            config = Object.assign({}, defaultConfig);
        }
        else {
            try {
                config = JSON.parse(localStorage.pdlSetting);
            }
            catch (error) {
                logger.error('Use default config because: ', error);
                config = Object.assign({}, defaultConfig);
            }
        }
        if (config.version !== defaultConfig.version) {
            config = {
                ...defaultConfig,
                ...config,
                version: defaultConfig.version,
                showMsg: true
            };
            localStorage.pdlSetting = JSON.stringify(config);
        }
        return {
            get(key) {
                return config[key] ?? defaultConfig[key];
            },
            set(key, value) {
                if (config[key] !== value) {
                    config[key] = value;
                    localStorage.pdlSetting = JSON.stringify(config);
                    logger.info('Config set:', key, value);
                }
            }
        };
    }
    const hostname = location.hostname;
    let siteConfig;
    if (hostname === 'rule34.xxx') {
        siteConfig = {
            folderPattern: 'rule34/{artist}',
            filenamePattern: '{id}_{artist}_{character}'
        };
    }
    else if (hostname === 'danbooru.donmai.us') {
        siteConfig = {
            folderPattern: 'danbooru/{artist}',
            filenamePattern: '{id}_{artist}_{character}'
        };
    }
    const config = loadConfig(siteConfig);

    let saveFile;
    if (env.isBlobDlAvaliable() && env.isSupportSubpath()) {
        saveFile = gmDownload;
    }
    else {
        saveFile = aDownload;
        logger.warn('Download function is not fully supported:', GM_info.scriptHandler, GM_info.version);
    }
    const fileSaveAdapters = {
        getAdapter() {
            if (this.isFileSystemAccessEnable()) {
                fsaHandler.setFilenameConflictAction(config.get('fileSystemFilenameConflictAction'));
                return fsaHandler.saveFile.bind(fsaHandler);
            }
            else {
                return saveFile;
            }
        },
        isFileSystemAccessEnable() {
            return env.isFileSystemAccessAvaliable() && config.get('useFileSystemAccess');
        },
        dirHandleCheck() {
            if (this.isFileSystemAccessEnable() && fsaHandler.isDirHandleNotSet())
                fsaHandler.updateDirHandle();
        }
    };

    function createDownloader() {
        const MAX_DOWNLOAD = 5;
        const MAX_RETRY = 3;
        const DOWNLOAD_INTERVAL = 500;
        let queue = [];
        let active = [];
        const cleanAndStartNext = (removeMeta, nextMeta) => {
            sleep(DOWNLOAD_INTERVAL).then(() => {
                active.splice(active.indexOf(removeMeta), 1);
                if (nextMeta) {
                    active.push(nextMeta);
                    dispatchDownload(nextMeta);
                }
                else if (queue.length) {
                    const meta = queue.shift();
                    active.push(meta);
                    dispatchDownload(meta);
                }
            });
        };
        const errorHandlerFactory = (downloadMeta) => {
            return {
                ontimeout(error) {
                    const { taskId, config, isAborted } = downloadMeta;
                    if (isAborted)
                        return;
                    downloadMeta.retry++;
                    logger.error('Download timeout', downloadMeta.retry, ':', config.src, error);
                    if (downloadMeta.retry > MAX_RETRY) {
                        const err = new Error(`Download failed: ${taskId} | ${config.src}`);
                        config.onError?.(err, config);
                        downloadMeta.reject(err);
                        cleanAndStartNext(downloadMeta);
                    }
                    else {
                        logger.info('Download retry:', downloadMeta.retry, config.src);
                        cleanAndStartNext(downloadMeta, downloadMeta);
                    }
                },
                onerror(error) {
                    const { taskId, config, isAborted } = downloadMeta;
                    if (isAborted)
                        return;
                    let err;
                    if ('status' in error && error.status === 429) {
                        err = new RequestError('Too many request', error);
                    }
                    else {
                        err = new Error(`Download failed. ID: ${taskId}. Reason: ${error.error}.`);
                        logger.error(error);
                    }
                    config.onError?.(err, config);
                    downloadMeta.reject(err);
                    cleanAndStartNext(downloadMeta);
                }
            };
        };
        const gmDownload = (downloadMeta, errHandler) => {
            const { taskId, config } = downloadMeta;
            const { ontimeout, onerror } = errHandler;
            return GM_download({
                url: config.src,
                name: config.path,
                headers: config.headers,
                ontimeout,
                onerror,
                onprogress(res) {
                    if (res.loaded > 0 && res.total > 0) {
                        const progress = Math.floor((res.loaded / res.total) * 100);
                        config.onProgress?.(progress, config);
                    }
                },
                onload() {
                    cleanAndStartNext(downloadMeta);
                    config.onFileSaved?.(config);
                    downloadMeta.resolve(taskId);
                    logger.info('Download complete:', taskId, config.path);
                }
            });
        };
        const xhr = (downloadMeta, errHandler) => {
            const { taskId, config, timeout } = downloadMeta;
            const { ontimeout, onerror } = errHandler;
            const saveFile = fileSaveAdapters.getAdapter();
            return GM_xmlhttpRequest({
                url: config.src,
                timeout,
                method: 'GET',
                headers: config.headers,
                responseType: 'blob',
                ontimeout,
                onerror,
                onprogress: (res) => {
                    if (res.loaded > 0 && res.total > 0) {
                        const progress = Math.floor((res.loaded / res.total) * 100);
                        config.onProgress?.(progress, config);
                    }
                },
                onload: async (e) => {
                    logger.info('Xhr complete:', config.src);
                    cleanAndStartNext(downloadMeta);
                    if (downloadMeta.isAborted)
                        return logger.warn('Download was canceled.', taskId, config.path);
                    config.onXhrLoaded?.(config);
                    try {
                        let modRes;
                        if (typeof config.beforeFileSave === 'function') {
                            modRes = await config.beforeFileSave(e.response, config);
                            if (modRes && !downloadMeta.isAborted) {
                                await saveFile(modRes, downloadMeta);
                                config.onFileSaved?.(config);
                                logger.info('Download complete:', config.path);
                            }
                        }
                        else {
                            await saveFile(e.response, downloadMeta);
                            config.onFileSaved?.(config);
                            logger.info('Download complete:', config.path);
                        }
                        downloadMeta.resolve(downloadMeta.taskId);
                    }
                    catch (error) {
                        config.onError?.(error, config);
                        downloadMeta.reject(error);
                    }
                }
            });
        };
        const isDirectSaveConfig = (downloadMeta) => {
            return !!downloadMeta.config.directSave;
        };
        const dispatchDownload = (downloadMeta) => {
            logger.info('Start download:', downloadMeta.config.src);
            let abortObj;
            const errHandler = errorHandlerFactory(downloadMeta);
            if (isDirectSaveConfig(downloadMeta)) {
                abortObj = gmDownload(downloadMeta, errHandler);
            }
            else {
                abortObj = xhr(downloadMeta, errHandler);
            }
            downloadMeta.abort = abortObj.abort;
        };
        return {
            get fileSystemAccessEnabled() {
                return fileSaveAdapters.isFileSystemAccessEnable();
            },
            dirHandleCheck() {
                fileSaveAdapters.dirHandleCheck();
            },
            async download(configs) {
                logger.info('Downloader add:', configs);
                if (!Array.isArray(configs))
                    configs = [configs];
                if (configs.length < 1)
                    return Promise.resolve([]);
                const promises = [];
                configs.forEach((config) => {
                    promises.push(new Promise((resolve, reject) => {
                        const downloadMeta = {
                            taskId: config.taskId,
                            config,
                            isAborted: false,
                            retry: 0,
                            timeout: config.timeout,
                            resolve,
                            reject
                        };
                        queue.push(downloadMeta);
                    }));
                });
                while (active.length < MAX_DOWNLOAD && queue.length) {
                    const meta = queue.shift();
                    active.push(meta);
                    dispatchDownload(meta);
                }
                return await Promise.all(promises);
            },
            abort(taskIds) {
                if (typeof taskIds === 'string')
                    taskIds = [taskIds];
                if (!taskIds.length)
                    return;
                logger.info('Downloader delete, active:', active.length, 'queue', queue.length);
                active = active.filter((downloadMeta) => {
                    if (taskIds.includes(downloadMeta.taskId) && !downloadMeta.isAborted) {
                        downloadMeta.isAborted = true;
                        downloadMeta.abort?.();
                        downloadMeta.config.onAbort?.(downloadMeta.config);
                        downloadMeta.reject(new CancelError());
                        logger.warn('Download aborted:', downloadMeta.config.path);
                    }
                    else {
                        return true;
                    }
                });
                queue = queue.filter((downloadMeta) => !taskIds.includes(downloadMeta.taskId));
                while (active.length < MAX_DOWNLOAD && queue.length) {
                    const meta = queue.shift();
                    active.push(meta);
                    dispatchDownload(meta);
                }
            }
        };
    }
    const downloader = createDownloader();

    class HistoryDb extends Dexie__default["default"] {
        history;
        constructor() {
            super('PdlHistory');
            this.version(2).stores({
                history: 'pid, userId, user, title, *tags'
            });
        }
    }
    function createHistoryDb() {
        const db = new HistoryDb();
        let record;
        function migrateFromLocalStorage() {
            if (localStorage.pixivDownloader) {
                const datas = JSON.parse(localStorage.pixivDownloader).map((pid) => ({
                    pid: Number(pid)
                }));
                const tempKeys = Object.keys(localStorage).filter((key) => /(?<=^pdlTemp-)\d+$/.test(key));
                if (tempKeys.length) {
                    tempKeys.forEach((key) => {
                        const [id] = /(?<=^pdlTemp-)\d+$/.exec(key);
                        datas.push({ pid: Number(id) });
                        localStorage.removeItem(key);
                    });
                }
                db.history.bulkPut(datas).then(() => {
                    localStorage.removeItem('pixivDownloader');
                });
            }
        }
        migrateFromLocalStorage();
        logger.time('loadDb');
        db.history.toArray().then((datas) => {
            record = new Set(datas.map((data) => data.pid));
            logger.timeEnd('loadDb');
        });
        return {
            async add(historyData) {
                if (!(await this.has(historyData.pid))) {
                    db.history.put(historyData);
                    record.add(historyData.pid);
                }
            },
            bulkAdd(historyDatas) {
                const result = db.history.bulkPut(historyDatas);
                historyDatas.forEach((data) => {
                    record.add(data.pid);
                });
                return result;
            },
            async has(pid) {
                if (typeof pid === 'string')
                    pid = Number(pid);
                if (record) {
                    return record.has(pid);
                }
                else {
                    return !!(await db.history.get(pid));
                }
            },
            getAll() {
                return db.history.toArray();
            },
            clear() {
                record && (record = new Set());
                return db.history.clear();
            }
        };
    }
    const historyDb = createHistoryDb();

    const rule34Parser = {
        async parse(id) {
            const res = await fetch('index.php?page=post&s=view&id=' + id);
            if (!res.ok)
                throw new RequestError('Request failed with status code ' + res.status, res);
            const html = await res.text();
            const doc = new DOMParser().parseFromString(html, 'text/html');
            const src = doc.querySelector('#gelcomVideoPlayer > source')?.src ||
                doc.querySelector('meta[property="og:image"]').getAttribute('content');
            const imageNameMatch = /(?<=\/)\w+\.\w+(?=\?)/.exec(src);
            if (!imageNameMatch)
                throw new Error('Can not parse image name from src.');
            const imageName = imageNameMatch[0];
            const [title, extendName] = imageName.split('.');
            const artists = [];
            const characters = [];
            const tags = [];
            const tagEls = doc.querySelectorAll('li[class*="tag-type"]');
            tagEls.forEach((tagEl) => {
                const tagTypeMatch = /(?<=tag-type-)\w+/.exec(tagEl.className);
                if (!tagTypeMatch)
                    throw new Error('Unknown tag: ' + tagEl.className);
                const tagType = tagTypeMatch[0];
                const tag = tagEl.querySelector('a[href*="page=post"]')?.textContent || '';
                if (tagType === 'artist') {
                    artists.push(tag);
                }
                else if (tagType === 'character') {
                    characters.push(tag);
                }
                tags.push(tagType + ':' + tag);
            });
            const uploaderEl = doc.querySelector('a[href*="page=account&s=profile"]');
            const postDateStr = uploaderEl?.parentElement?.firstChild?.nodeValue;
            const postDate = postDateStr ? postDateStr.split(': ')[1] : '';
            const sourceEl = uploaderEl?.parentElement?.nextElementSibling?.nextElementSibling;
            if (sourceEl && sourceEl.textContent?.toLowerCase().includes('source')) {
                const sourceLink = sourceEl.querySelector('a');
                if (sourceLink) {
                    tags.push('source:' + sourceLink.href);
                }
                else {
                    tags.push('source:' + sourceEl.textContent.replace('Source: ', ''));
                }
            }
            return {
                id,
                src,
                extendName,
                artist: artists.join(',') || 'UnknownArtist',
                character: characters.join(',') || 'UnknownCharacter',
                title,
                tags,
                createDate: postDate
            };
        }
    };

    class DownloadConfigBuilder {
        meta;
        constructor(meta) {
            this.meta = meta;
        }
        normalizeString(str) {
            return replaceInvalidChar(unescapeHtml(str));
        }
        getFolderPattern() {
            return config.get('folderPattern');
        }
        getFilenamePattern() {
            return config.get('filenamePattern');
        }
        getFullpathPattern() {
            const folder = this.getFolderPattern();
            const filename = this.getFilenamePattern() + '.' + this.meta.extendName;
            return folder ? folder + '/' + filename : filename;
        }
        isBrowserApi() {
            return env.isBrowserDownloadMode();
        }
        isFsaEnable() {
            return downloader.fileSystemAccessEnabled;
        }
        supportSubpath() {
            return this.isBrowserApi() || this.isFsaEnable();
        }
        isImage() {
            return /bmp|jp(e)?g|png|tif|gif|exif|svg|webp/i.test(this.meta.extendName);
        }
        buildFilePath() {
            const path = this.getFullpathPattern();
            const { id, createDate } = this.meta;
            let { artist, title, tags } = this.meta;
            artist = this.normalizeString(artist);
            title = this.normalizeString(title);
            tags = tags.map((tag) => this.normalizeString(tag));
            const replaceDate = (match, p1) => {
                const format = p1 || 'YYYY-MM-DD';
                return dayjs__default["default"](createDate).format(format);
            };
            return path
                .replaceAll(/\{date\((.*?)\)\}|\{date\}/g, replaceDate)
                .replaceAll('{artist}', artist)
                .replaceAll('{title}', title)
                .replaceAll('{tags}', tags.join('_'))
                .replaceAll('{id}', id);
        }
    }

    function artworkProgressFactory$1(btn) {
        if (!btn)
            return;
        return function onArtworkProgress(progress) {
            btn.setProgress(progress);
        };
    }
    class Rule34DownloadConfig extends DownloadConfigBuilder {
        meta;
        constructor(meta) {
            super(meta);
            this.meta = meta;
        }
        getDownloadConfig(btn) {
            return {
                taskId: Math.random().toString(36).slice(2),
                src: this.meta.src,
                path: this.buildFilePath(),
                source: this.meta,
                timeout: this.isImage() ? 60000 : undefined,
                directSave: downloader.fileSystemAccessEnabled ? false : true,
                onProgress: artworkProgressFactory$1(btn)
            };
        }
        buildFilePath() {
            const path = super.buildFilePath();
            return path.replaceAll('{character}', this.normalizeString(this.meta.character));
        }
    }

    function addBookmark$2(id) {
        unsafeWindow.addFav(id);
    }

    async function downloadArtwork$2(btn) {
        downloader.dirHandleCheck();
        const id = btn.getAttribute('pdl-id');
        const mediaMeta = await rule34Parser.parse(id);
        const { tags, artist, title } = mediaMeta;
        const downloadConfigs = new Rule34DownloadConfig(mediaMeta).getDownloadConfig(btn);
        config.get('addBookmark') && addBookmark$2(id);
        await downloader.download(downloadConfigs);
        const historyData = {
            pid: Number(id),
            user: artist,
            title,
            tags
        };
        historyDb.add(historyData);
    }

    const langZh = {
        button: {
            download_stop: '停止',
            download_works: '作品',
            download_bookmarks: '收藏',
            download_bookmarks_public: '公开',
            download_bookmarks_private: '不公开',
            download_all_one_page: '全部(单页)',
            download_all: '全部(批量)',
            download_r18_one_page: 'R-18(单页)',
            download_r18: 'R-18(批量)',
            fsa_change_dir: '更改',
            history_import: '导入记录(替换)',
            history_import_json: '导入记录(json)',
            history_import_txt: '导入旧记录(txt)',
            history_merge: '导入记录(合并)',
            history_export: '导出记录',
            history_export_csv: '导出CSV',
            history_clear: '清除记录'
        },
        checkbox: {
            filter_exclude_downloaded: '排除已下载图片',
            filter_illusts: '插画',
            filter_manga: '漫画',
            filter_ugoira: '动图'
        },
        radio: {
            filename_conflict_option_uniquify: '重命名',
            filename_conflict_option_overwrite: '覆盖',
            filename_conflict_option_prompt: '提示'
        },
        text: {
            feedback: '有问题or想建议?这里反馈',
            gm_menu: '设置',
            tab_title_filename: '文件名',
            tab_title_ugoira: '动图',
            tab_title_history: '历史记录',
            tab_title_button: '按钮',
            tab_title_others: '其它',
            tab_title_feedback: '反馈 / 赞赏',
            label_folder: '文件夹名:',
            label_filename: '文件名:',
            label_filename_conflict: '文件名重复时:',
            label_tag_lang: '标签语言:',
            label_fsa: '使用FileSystemAccess API',
            label_ugoira_format: '动图格式:',
            label_button_horizon: '水平:',
            label_button_vertical: '垂直:',
            title_button_preview: '预览图',
            title_button_preview_self_bookmark: '预览图(我的收藏)',
            placeholder_folder_subfolder_unused: '我不想保存到子文件夹',
            placeholder_folder_vm: 'Violentmonkey不支持',
            placeholder_folder_need_api: '需要Browser Api',
            placeholder_filename_requried: '你的名字?',
            placeholder_fsa_folder: '根文件夹名',
            tips_filename_pattern: '{artist}:作者, {artistID}:作者ID, {title}:作品标题, {id}:作品ID, {page}:页码, {tags}:作品标签,{date} / {date(占位符)}: 创建时间',
            tips_rule34_filename_pattern: '{artist}:作者, {character}:角色, {id}:作品ID, {date} / {date(占位符)}: 发布时间',
            tips_subfolder_unused: '如果不想保存到画师目录,文件夹名留空即可。',
            tips_tag_translation: '请注意:标签翻译不一定是你选择的语言,部分<a href="https://crowdin.com/project/pixiv-tags" target="_blank">无对应语言翻译的标签</a>仍可能是其他语言。',
            option_bundle_illusts: '将多页插图打包为.zip压缩包',
            option_bundle_manga: '将多页漫画作品打包为.zip压缩包',
            option_add_bookmark: '下载单个作品时收藏作品',
            option_add_bookmark_with_tags: '收藏时添加作品标签',
            option_add_bookmark_private_r18: '将R-18作品收藏到不公开类别',
            option_show_popup_button: '显示设置按钮',
            confirm_clear_history: '真的要清除历史记录吗?'
        }
    };
    const langEn = {
        button: {
            download_stop: 'Stop',
            download_works: 'Works',
            download_bookmarks: 'Bookmarks',
            download_bookmarks_public: 'Public',
            download_bookmarks_private: 'Private',
            download_all_one_page: 'All (one page)',
            download_all: 'All',
            download_r18_one_page: 'R-18 (one page)',
            download_r18: 'R-18',
            fsa_change_dir: 'Change',
            history_import: 'Import (Replace)',
            history_import_json: 'Import (json)',
            history_import_txt: 'Import (txt)',
            history_merge: 'Import (Merge)',
            history_export: 'Export',
            history_export_csv: 'Export as CSV',
            history_clear: 'Clear'
        },
        checkbox: {
            filter_exclude_downloaded: 'Exclude downloaded',
            filter_illusts: 'Illustrations',
            filter_manga: 'Manga',
            filter_ugoira: 'Ugoira'
        },
        radio: {
            filename_conflict_option_uniquify: 'Uniquify',
            filename_conflict_option_overwrite: 'Overwrite',
            filename_conflict_option_prompt: 'Prompt'
        },
        text: {
            feedback: 'Feedback',
            gm_menu: 'Setting',
            tab_title_filename: 'Filename',
            tab_title_ugoira: 'Ugoira',
            tab_title_history: 'History',
            tab_title_button: 'Button',
            tab_title_others: 'Others',
            tab_title_feedback: 'Feedback',
            label_folder: 'Folder:',
            label_filename: 'Filename:',
            label_filename_conflict: 'Conflict Action:',
            label_tag_lang: 'Tags language:',
            label_fsa: 'FileSystemAccess API',
            label_ugoira_format: 'Ugoira Format:',
            label_button_horizon: 'X:',
            label_button_vertical: 'Y:',
            title_button_preview: 'Thumbnail',
            title_button_preview_self_bookmark: 'Thumbnail(My bookmarks)',
            placeholder_folder_subfolder_unused: "I don't need subfolder",
            placeholder_folder_vm: "VM doesn't support",
            placeholder_folder_need_api: 'Need Browser Api',
            placeholder_filename_requried: 'Your Name?',
            placeholder_fsa_folder: 'Root directory',
            tips_filename_pattern: '{artist}, {artistID}, {title}, {id}, {page}, {tags}, {date} / {date(format)}',
            tips_rule34_filename_pattern: '{artist}, {character}, {id}, {date} / {date(format)}',
            tips_subfolder_unused: "If you don't need a subfolder, just leave the folder name blank.",
            tips_tag_translation: '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.',
            option_bundle_illusts: 'Pack multi-page illustrations into a .zip archive',
            option_bundle_manga: 'Pack manga into a .zip archive',
            option_add_bookmark: 'Bookmark work when downloading a single work',
            option_add_bookmark_with_tags: 'Add works tags',
            option_add_bookmark_private_r18: 'Bookmark R-18 works to private category',
            option_show_popup_button: 'Show setting button',
            confirm_clear_history: 'Do you really want to clear history?'
        }
    };
    const messages = {
        'zh-cn': langZh,
        'zh-tw': langZh,
        zh: langZh,
        en: langEn
    };
    const curLang = navigator.language.toLowerCase();
    const defaultLang = 'en';
    function t(key) {
        const lang = (curLang in messages ? curLang : defaultLang);
        const paths = key.split('.');
        let last = messages[lang];
        for (let i = 0; i < paths.length; i++) {
            const value = last[paths[i]];
            if (value === undefined || value === null)
                return null;
            last = value;
        }
        return last;
    }

    var css$9 = ".pdl-modal{background-color:rgba(0,0,0,.32);color:var(--pdl-text1);display:flex;font-family:'win-bug-omega, system-ui, -apple-system, \"Segoe UI\", Roboto, Ubuntu, Cantarell, \"Noto Sans\", \"Hiragino Kaku Gothic ProN\", Meiryo, sans-serif';height:100%;left:0;line-height:1.15;position:fixed;top:0;user-select:text;width:100%;z-index:99}.pdl-modal .pdl-dialog{background-color:var(--pdl-bg1);border-radius:24px;font-size:16px;margin:auto;max-width:1080px;min-width:680px;position:relative;width:50vw}.pdl-modal .pdl-dialog>.container{margin:20px 40px 30px;overflow:hidden;transition:all .2s}.pdl-modal .pdl-dialog>.container>div{display:flex;flex-direction:column}.pdl-modal .pdl-dialog .pdl-dialog-close{background-color:transparent;background:linear-gradient(#7d7d7d,#7d7d7d) 50%/18px 2px no-repeat,linear-gradient(#7d7d7d,#7d7d7d) 50%/2px 18px no-repeat;border:none;border-radius:50%;cursor:pointer;height:25px;margin:0;padding:0;position:absolute;right:10px;top:10px;transform:rotate(45deg);transition:background-color .25s;width:25px}.pdl-modal .pdl-dialog .pdl-dialog-close:hover{background-color:rgba(0,0,0,.05)}.pdl-modal .pdl-dialog .pdl-dialog-content a{color:#0096fa;text-decoration:underline}.pdl-modal .pdl-dialog .pdl-dialog-content input[type=radio],.pdl-modal .pdl-dialog .pdl-dialog-content input[type=radio]+label{cursor:pointer}.pdl-modal .pdl-dialog .pdl-dialog-content hr{border:none;border-top:1px solid var(--pdl-border1);height:0!important;margin:0}.pdl-modal .pdl-dialog .pdl-dialog-content hr.sub{margin-inline-start:1.5em}.pdl-modal .pdl-dialog .pdl-dialog-content hr.vertical{border:none;border-left:1px solid var(--pdl-border1);height:1.15em!important}.pdl-modal .pdl-dialog .pdl-dialog-content .pdl-dialog-button{background-color:#fff;border:1px solid #7d7d7d;border-radius:5px;cursor:pointer;font-size:16px;line-height:1.15;padding:.5em 1.5em;transition:opacity .2s}.pdl-modal .pdl-dialog .pdl-dialog-content .pdl-dialog-button[disabled]{background-color:#fff;border-color:#e4e7ed;color:#c0c4cc;cursor:not-allowed!important;opacity:1!important}.pdl-modal .pdl-dialog .pdl-dialog-content .pdl-dialog-button:hover{opacity:.7}.pdl-modal .pdl-dialog .pdl-dialog-content .pdl-dialog-button.primary{background-color:#0096fa;border-color:#0096fa;color:#fff}.pdl-modal .pdl-dialog .pdl-dialog-content .pdl-dialog-button.primary[disabled]{background-color:#a0cfff;border-color:#a0cfff}.pdl-modal .pdl-dialog .pdl-dialog-content .pdl-dialog-button.icon{padding:.5em .8em}.pdl-modal button,.pdl-modal input,.pdl-modal optgroup,.pdl-modal select,.pdl-modal textarea{font-family:inherit;font-size:100%;line-height:1.15;margin:0}";

    var css$8 = ".pdl-checkbox{appearance:none;background-color:#858585!important;border:2px solid transparent!important;border-radius:14px!important;box-sizing:border-box;cursor:pointer;height:14px;margin:0!important;padding:0!important;position:relative;transition:background-color .2s ease 0s,box-shadow .2s ease 0s;vertical-align:top;width:28px}.pdl-checkbox:hover{background-color:var(--pdl-bg3-hover)!important}.pdl-checkbox:checked{background-color:#0096fa!important}.pdl-checkbox:after{background-color:#fff;border-radius:10px;content:\"\";display:block;height:10px;left:0;position:absolute;top:0;transform:translateX(0);transition:transform .2s ease 0s;width:10px}.pdl-checkbox:checked:after{transform:translateX(14px)}";

    var css$7 = ".pdl-hide{display:none!important}.pdl-unavailable{cursor:not-allowed!important;opacity:.5!important;pointer-events:none!important}.pdl-spacer{flex:1;margin:0;padding:0}";

    class PdlDialog extends HTMLElement {
        ob;
        controller = null;
        constructor() {
            super();
            this.render();
        }
        render() {
            const shadowRoot = this.attachShadow({ mode: 'open' });
            shadowRoot.innerHTML = `
    <style>${css$7 + css$9 + css$8}</style>
    <div class="pdl-modal">
      <div class="pdl-dialog">
        <button class="pdl-dialog-close"></button>
        <div class="container">
          <div>
            <header class="pdl-dialog-header"></header>
            <div class="pdl-dialog-content"></div>
            <footer class="pdl-dialog-footer"></footer>
          </div>
        </div>
      </div> 
    </div>`;
        }
        connectedCallback() {
            const shadowRoot = this.shadowRoot;
            const modal = shadowRoot.querySelector('.pdl-modal');
            const closeBtn = modal.querySelector('.pdl-dialog-close');
            this.controller = new AbortController();
            const signal = this.controller.signal;
            window.addEventListener('keydown', (evt) => {
                evt.stopPropagation();
            }, { capture: true, signal });
            modal.addEventListener('click', (evt) => {
                const closeOnClickModal = this.hasAttribute('close-on-click-modal');
                closeOnClickModal && evt.target === modal && this.remove();
            }, { signal });
            closeBtn.addEventListener('click', () => {
                this.remove();
            }, { signal });
            const container = shadowRoot.querySelector('.pdl-dialog > .container');
            const obEl = container.firstElementChild;
            this.ob = new ResizeObserver(() => {
                const height = obEl.offsetHeight;
                container.style.height = height + 'px';
            });
            this.ob.observe(obEl);
        }
        disconnectedCallback() {
            this.controller?.abort();
            this.controller = null;
            this.ob.disconnect();
        }
    }
    customElements.define('pdl-dialog', PdlDialog);
    function createModal(args, option = { closeOnClickModal: false }) {
        const el = document.createElement('pdl-dialog');
        if (option.closeOnClickModal) {
            el.setAttribute('close-on-click-modal', '');
        }
        const shadowRoot = el.shadowRoot;
        const header = shadowRoot.querySelector('.pdl-dialog-header');
        const content = shadowRoot.querySelector('.pdl-dialog-content');
        const footer = shadowRoot.querySelector('.pdl-dialog-footer');
        args.header && header.appendChild(args.header);
        args.footer && footer.appendChild(args.footer);
        content.appendChild(args.content);
        return el;
    }

    function createTabUgoira() {
        const tabHtml = `<div class="pdl-tab-item ${!env.isPixiv() && 'pdl-hide'}">${t('text.tab_title_ugoira')}</div>`;
        const paneHtml = `
<div class="pdl-tab-pane">
  <div id="pdl-setting-ugoira">
    <p class="option-header">${t('text.label_ugoira_format')}</p>
    <div id="pdl-ugoira-format-wrap">
      <div class="pdl-ugoira-format-item">
        <input type="radio" id="pdl-ugoira-zip" value="zip" name="format" /><label for="pdl-ugoira-zip">Zip</label>
      </div>
      <div class="pdl-ugoira-format-item">
        <input type="radio" id="pdl-ugoira-gif" value="gif" name="format" /><label for="pdl-ugoira-gif">Gif</label>
      </div>
      <div class="pdl-ugoira-format-item">
        <input type="radio" id="pdl-ugoira-apng" value="png" name="format" /><label for="pdl-ugoira-apng">Png</label>
      </div>
      <div class="pdl-ugoira-format-item">
        <input type="radio" id="pdl-ugoira-webm" value="webm" name="format" /><label for="pdl-ugoira-webm">Webm</label>
      </div>
      <div class="pdl-ugoira-format-item">
        <input type="radio" id="pdl-ugoira-webp" value="webp" name="format" /><label for="pdl-ugoira-webp">Webp</label>
      </div>
    </div>
  </div>
</div>`;
        const tab = stringToFragment(tabHtml);
        const pane = stringToFragment(paneHtml);
        const ugoiraFormat = config.get('ugoiraFormat');
        pane.querySelectorAll('.pdl-ugoira-format-item input[type="radio"]').forEach((el) => {
            if (ugoiraFormat === el.value)
                el.checked = true;
            el.addEventListener('change', (ev) => {
                config.set('ugoiraFormat', ev.currentTarget.value);
            });
        });
        return {
            tab,
            pane
        };
    }

    function createTabFilename() {
        const tabHtml = `<div class="pdl-tab-item">${t('text.tab_title_filename')}</div>`;
        const paneHtml = `
<div class="pdl-tab-pane">
  <div id="pdl-setting-filename">
    <div>
      <div class="pdl-input-wrap">
        <label for="pdlfolder">${t('text.label_folder')}</label>
        <input type="text" id="pdlfolder" maxlength="100" />
        <button id="pdl-filename-folder-reset" class="pdl-dialog-button icon" disabled>↺</button>
        <button id="pdl-filename-folder-confirm" class="pdl-dialog-button icon primary" disabled>✓</button>
      </div>
      <div class="pdl-input-wrap">
        <label for="pdlfilename">${t('text.label_filename')}</label>
        <input type="text" id="pdlfilename" placeholder="${t('text.placeholder_folder_subfolder_unused')}" required maxlength="100" />
        <button id="pdl-filename-filename-reset" class="pdl-dialog-button icon" disabled>↺</button>
        <button id="pdl-filename-filename-confirm" class="pdl-dialog-button icon primary" disabled>✓</button>
      </div>
    </div>
    ${env.isPixiv()
        ? `<div class="tags-option">
    <span class="tags-title">${t('text.label_tag_lang')}</span>
    <div class="tags-content">
      <div class="tags-item">
        <input class="pdl-option-tag" type="radio" name="lang" id="lang_ja" value="ja" />
        <label for="lang_ja">日本語(default)</label>
      </div>
      <div class="tags-item">
        <input class="pdl-option-tag" type="radio" name="lang" id="lang_zh" value="zh" />
        <label for="lang_zh">简中</label>
      </div>
      <div class="tags-item">
        <input class="pdl-option-tag" type="radio" name="lang" id="lang_zh_tw" value="zh_tw" />
        <label for="lang_zh_tw">繁中</label>
      </div>
      <div class="tags-item">
        <input class="pdl-option-tag" type="radio" name="lang" id="lang_en" value="en" />
        <label for="lang_en">English</label>
      </div>
    </div>
  </div>
  <p style="font-size: 14px; margin: 0.5em 0">${t('text.tips_filename_pattern')}</p>
  <p style="font-size: 14px; margin: 0.5em 0">${t('text.tips_tag_translation')}</p>`
        : `<p style="font-size: 14px; margin: 0.5em 0">${t('text.tips_rule34_filename_pattern')}</p>`}
    <p style="font-size: 14px; margin: 0.5em 0">${t('text.tips_subfolder_unused')}</p>
    <hr />
    <div ${env.isFileSystemAccessAvaliable() ? '' : 'class="pdl-unavailable"'}>
      <div style="display: flex; justify-content: space-between; align-items: center; margin: 12px 0; gap: 12px">
        <label class="pdl-options" style="padding: 0.6em 4px">
          <span style="font-weight: 700; margin-right: 8px">${t('text.label_fsa')}</span>
          <input id="pdl-options-file-system-access" type="checkbox" class="pdl-checkbox"/>
        </label>
        <hr class="vertical" />
        <div class="pdl-input-wrap" style="flex: 1; margin: 0">
          <input id="pdl-fsa-show-directory" type="text" placeholder="${t('text.placeholder_fsa_folder')}" style="font-size: 14px; padding: 8px 0.5em; line-height: 1.15" disabled/>
        </div>
        <button id="pdl-fsa-change-directory" class="pdl-dialog-button primary">${t('button.fsa_change_dir')}</button>
      </div>
      <div class="tags-option">
        <span class="tags-title">${t('text.label_filename_conflict')}</span>
        <div class="tags-content">
          <div class="tags-item">
            <input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_uniquify" value="uniquify"/>
            <label for="action_uniquify">${t('radio.filename_conflict_option_uniquify')}</label>
          </div>
          <div class="tags-item">
            <input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_overwrite" value="overwrite"/>
            <label for="action_overwrite">${t('radio.filename_conflict_option_overwrite')}</label>
          </div>
          <div class="tags-item">
            <input class="pdl-option-conflict" type="radio" name="conflict_action" id="action_prompt" value="prompt"/>
            <label for="action_prompt">${t('radio.filename_conflict_option_prompt')}</label>
          </div>
        </div>
      </div>
    </div>
  </div>
</div>`;
        const tab = stringToFragment(tabHtml);
        const pane = stringToFragment(paneHtml);
        const folder = pane.querySelector('#pdlfolder');
        const folderReset = pane.querySelector('#pdl-filename-folder-reset');
        const folderUpdate = pane.querySelector('#pdl-filename-folder-confirm');
        const filename = pane.querySelector('#pdlfilename');
        const filenameReset = pane.querySelector('#pdl-filename-filename-reset');
        const filenameUpdate = pane.querySelector('#pdl-filename-filename-confirm');
        const filenamePattern = config.get('filenamePattern');
        const folderPattern = config.get('folderPattern');
        if (!folder || !filename)
            throw new Error('[Error]Can not create modal.');
        filename.value = filenamePattern;
        if (!env.isSupportSubpath()) {
            folder.setAttribute('disabled', '');
            folder.value = '';
        }
        else {
            folder.value = folderPattern;
        }
        folder.placeholder = env.isViolentmonkey()
            ? t('text.placeholder_folder_vm')
            : !env.isSupportSubpath()
                ? t('text.placeholder_folder_need_api')
                : t('text.placeholder_folder_subfolder_unused');
        folder.addEventListener('input', () => {
            folderReset?.removeAttribute('disabled');
            folderUpdate?.removeAttribute('disabled');
        });
        folderReset?.addEventListener('click', () => {
            folder.value = config.get('folderPattern');
            folderReset?.setAttribute('disabled', '');
            folderUpdate?.setAttribute('disabled', '');
        });
        folderUpdate?.addEventListener('click', () => {
            const folderPattern = folder.value
                .split('/')
                .map(replaceInvalidChar)
                .filter((path) => !!path)
                .join('/');
            config.set('folderPattern', folderPattern);
            folder.value = folderPattern;
            folderReset?.setAttribute('disabled', '');
            folderUpdate?.setAttribute('disabled', '');
        });
        filename.addEventListener('input', () => {
            filenameReset?.removeAttribute('disabled');
            filenameUpdate?.removeAttribute('disabled');
        });
        filenameReset?.addEventListener('click', () => {
            filename.value = config.get('filenamePattern');
            filenameReset?.setAttribute('disabled', '');
            filenameUpdate?.setAttribute('disabled', '');
        });
        filenameUpdate?.addEventListener('click', () => {
            const filenamePattern = replaceInvalidChar(filename.value);
            if (filenamePattern === '')
                return filenameReset?.click();
            config.set('filenamePattern', filenamePattern);
            filename.value = filenamePattern;
            filenameReset?.setAttribute('disabled', '');
            filenameUpdate?.setAttribute('disabled', '');
        });
        const tagLang = config.get('tagLang');
        pane.querySelectorAll('.tags-content .tags-item input.pdl-option-tag').forEach((input) => {
            if (tagLang === input.value)
                input.checked = true;
            input.addEventListener('change', (ev) => {
                config.set('tagLang', ev.currentTarget.value);
            });
        });
        if (env.isFileSystemAccessAvaliable()) {
            const enableFSA = pane.querySelector('#pdl-options-file-system-access');
            const showDir = pane.querySelector('#pdl-fsa-show-directory');
            const changeDirBtn = pane.querySelector('#pdl-fsa-change-directory');
            const actionInput = pane.querySelectorAll('.tags-content .tags-item input.pdl-option-conflict');
            const interactElems = [changeDirBtn, ...actionInput];
            const isUseFSA = config.get('useFileSystemAccess');
            const conflictAction = config.get('fileSystemFilenameConflictAction');
            if (isUseFSA) {
                folder.placeholder = t('text.placeholder_folder_subfolder_unused');
                folder.removeAttribute('disabled');
                folder.value = folderPattern;
            }
            enableFSA.checked = isUseFSA;
            if (!isUseFSA) {
                interactElems.forEach((el) => el.setAttribute('disabled', ''));
            }
            enableFSA.addEventListener('change', (ev) => {
                const isEnabled = ev.target.checked;
                config.set('useFileSystemAccess', isEnabled);
                if (isEnabled) {
                    folder.placeholder = t('text.placeholder_folder_subfolder_unused');
                    if (folder.hasAttribute('disabled')) {
                        folder.removeAttribute('disabled');
                        folder.value = config.get('folderPattern');
                    }
                    interactElems.forEach((el) => el.removeAttribute('disabled'));
                }
                else {
                    if (env.isViolentmonkey()) {
                        folder.placeholder = t('text.placeholder_folder_vm');
                        folder.setAttribute('disabled', '');
                        folder.value = '';
                    }
                    else if (!env.isSupportSubpath()) {
                        folder.placeholder = t('text.placeholder_folder_need_api');
                        folder.setAttribute('disabled', '');
                        folder.value = '';
                    }
                    interactElems.forEach((el) => el.setAttribute('disabled', ''));
                }
            });
            showDir.value = fsaHandler.getCurrentDirName();
            changeDirBtn.addEventListener('click', async () => {
                await fsaHandler.updateDirHandle();
                showDir.value = fsaHandler.getCurrentDirName();
            });
            actionInput.forEach((input) => {
                if (conflictAction === input.value)
                    input.checked = true;
                input.addEventListener('change', (ev) => {
                    config.set('fileSystemFilenameConflictAction', ev.currentTarget.value);
                });
            });
        }
        return {
            tab,
            pane
        };
    }

    var css$6 = "@property --pdl-progress{syntax:\"<percentage>\";inherits:true;initial-value:0}@keyframes pdl_loading{to{transform:translate(-50%,-50%) rotate(1turn)}}.pdl-btn{background:no-repeat 50%/85%;background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\");border:none;border-radius:4px;color:#01b468;cursor:pointer;display:inline-block;font-family:'win-bug-omega, system-ui, -apple-system, \"Segoe UI\", Roboto, Ubuntu, Cantarell, \"Noto Sans\", \"Hiragino Kaku Gothic ProN\", Meiryo, sans-serif';font-size:13px;font-weight:700;height:32px;line-height:32px;margin:0;overflow:hidden;padding:0;position:relative;text-align:center;text-decoration:none!important;text-overflow:ellipsis;user-select:none;white-space:nowrap;width:32px}.pdl-btn.pdl-btn-main{margin:0 0 0 10px}.pdl-btn.pdl-btn-sub{background-color:hsla(0,0%,100%,.5);left:calc((100% - 32px)*var(--pdl-btn-left)/100);position:absolute;top:calc((100% - 32px)*var(--pdl-btn-top)/100);z-index:1}.pdl-btn.pdl-btn-sub.presentation{border-radius:8px;left:auto;position:fixed;right:20px;top:50px}.pdl-btn.pdl-btn-sub.manga-viewer{border-radius:8px;left:auto;right:4px;top:80%}.pdl-btn.pdl-btn-sub.self-bookmark{left:calc((100% - 32px)*var(--pdl-btn-self-bookmark-left)/100);top:calc((100% - 32px)*var(--pdl-btn-self-bookmark-top)/100)}._history-item>.pdl-btn.pdl-btn-sub{z-index:auto}.pdl-btn.pdl-error{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")!important}.pdl-btn.pdl-complete{background-image:url(\"data:image/svg+xml;charset=utf-8,%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.267-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/svg%3E\")!important}.pdl-btn.pdl-progress{background-image:none!important;cursor:default!important}.pdl-btn.pdl-progress:after{border-radius:50%;content:\"\";display:inline-block;height:27px;left:50%;-webkit-mask:radial-gradient(transparent,transparent 54%,#000 57%,#000);mask:radial-gradient(transparent,transparent 54%,#000 57%,#000);position:absolute;top:50%;transform:translate(-50%,-50%);width:27px}.pdl-btn.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-btn.pdl-progress:empty:after{animation:pdl_loading 1.5s linear infinite;background:conic-gradient(#01b468 0,#01b468 25%,rgba(1,180,104,.2) 25%,rgba(1,180,104,.2))}.pdl-btn.pdl-tag{background-color:var(--pdl-btn1);border-bottom-right-radius:4px;border-top-right-radius:4px;height:auto;left:-1px;transition:background-image .5s}.pdl-btn.pdl-tag:hover{background-color:var(--pdl-btn1-hover)}.pdl-btn.pdl-modal-tag{background-color:var(--pdl-btn1);background-origin:content-box;border-radius:4px;height:50px;padding:5px;position:absolute;right:65px;top:6px;transition:background-color .25s;width:42px}.pdl-btn.pdl-modal-tag:hover{background-color:var(--pdl-btn1-hover)}.pdl-btn.pdl-tag-hide{background-image:url(\"data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'/%3E\")!important;pointer-events:none!important}.pdl-wrap-artworks{bottom:0;margin-top:40px;position:absolute;right:8px;top:0;z-index:1}.pdl-wrap-artworks.rule34{bottom:calc(22px + 1em)}.pdl-wrap-artworks .pdl-btn-sub.artworks{left:0;position:sticky;top:40px}:root .pdl-btn-main,:root[data-theme=default] .pdl-btn-main{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}@media (prefers-color-scheme:light){:root .pdl-btn-main{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}}:root[data-theme=dark] .pdl-btn-main{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}@media (prefers-color-scheme:dark){:root .pdl-btn-main{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}}";

    function createTabAdjustButtonPosition() {
        const style = getComputedStyle(document.documentElement);
        const tabHtml = `<div class="pdl-tab-item">${t('text.tab_title_button')}</div>`;
        const paneHtml = `
<div class="pdl-tab-pane">
  <style>${css$6}</style>
  <div class="pdl-adjust-button">
    <div class="pdl-adjust-content">
      <datalist id="pdl-adjust-tickmarks">
        <option value="0"></option>
        <option value="25"></option>
        <option value="50"></option>
        <option value="75"></option>
        <option value="100"></option>
      </datalist>
      ${env.isPixiv()
        ? `<div class="pdl-adjust-item">
      <p class="pdl-adjust-title">${t('text.title_button_preview_self_bookmark')}</p>
      <div class="pdl-adjust-select">
        <span>${t('text.label_button_horizon')}</span
        ><input
          id="pdl-btn-self-bookmark-left"
          type="range"
          max="100"
          min="0"
          step="1"
          list="pdl-adjust-tickmarks"
          value="${style.getPropertyValue('--pdl-btn-self-bookmark-left')}"
        />
      </div>
      <div class="pdl-adjust-select">
        <span>${t('text.label_button_vertical')}</span
        ><input
          id="pdl-btn-self-bookmark-top"
          type="range"
          max="100"
          min="0"
          step="1"
          list="pdl-adjust-tickmarks"
          value="${style.getPropertyValue('--pdl-btn-self-bookmark-top')}"
        />
      </div>
    </div>`
        : ''}
      <div class="pdl-adjust-item">
        <p class="pdl-adjust-title">${t('text.title_button_preview')}</p>
        <div class="pdl-adjust-select">
          <span>${t('text.label_button_horizon')}</span
          ><input
            id="pdl-btn-left"
            type="range"
            max="100"
            min="0"
            step="1"
            list="pdl-adjust-tickmarks"
            value="${style.getPropertyValue('--pdl-btn-left')}"
          />
        </div>
        <div class="pdl-adjust-select">
          <span>${t('text.label_button_vertical')}</span
          ><input
            id="pdl-btn-top"
            type="range"
            max="100"
            min="0"
            step="1"
            list="pdl-adjust-tickmarks"
            value="${style.getPropertyValue('--pdl-btn-top')}"
          />
        </div>
      </div>
    </div>
    <div class="pdl-adjust-preview">
      <div class="pdl-spacer"></div>
      <div class="pdl-thumbnail-sample">
        <button class="pdl-btn pdl-btn-sub"></button>
        ${env.isPixiv() ? `<button class="pdl-btn pdl-btn-sub self-bookmark pdl-complete"></button>` : ''}
      </div>
      <div class="pdl-spacer"></div>
    </div>
  </div>
</div>`;
        const tab = stringToFragment(tabHtml);
        const pane = stringToFragment(paneHtml);
        pane.querySelectorAll('.pdl-adjust-select input[type="range"]').forEach((el) => {
            el.addEventListener('input', (ev) => {
                const el = ev.target;
                const val = el.value;
                document.documentElement.style.setProperty('--' + el.id, val);
            });
            el.addEventListener('change', (ev) => {
                const el = ev.target;
                const key = el.id;
                config.set(key, el.value);
            });
        });
        return {
            tab,
            pane
        };
    }

    function createTabHistory() {
        const tabHtml = `<div class="pdl-tab-item">${t('text.tab_title_history')}</div>`;
        const paneHtml = `
<div class="pdl-tab-pane">
  <div id="pdl-setting-history">
    <div>
      <button id="pdl-export" class="btn-history pdl-dialog-button primary">
        ${t('button.history_export')}
      </button>
    </div>
    <div>
      <button id="pdl-export-csv" class="btn-history pdl-dialog-button primary">
        ${t('button.history_export_csv')}
      </button>
    </div>
    <div>
      <input type="file" id="pdl-import-json" accept=".json" style="display: none" />
      <button id="pdl-import-json-btn" class="btn-history pdl-dialog-button primary">
        ${t('button.history_import_json')}
      </button>
    </div>
    <div>
      <input type="file" id="pdl-import-txt" accept=".txt" style="display: none" />
      <button id="pdl-import-txt-btn" class="btn-history pdl-dialog-button primary">
        ${t('button.history_import_txt')}
      </button>
    </div>
    <div>
      <button id="pdl-clear-history" class="btn-history pdl-dialog-button primary">
        ${t('button.history_clear')}
      </button>
    </div>
  </div>
</div>`;
        const tab = stringToFragment(tabHtml);
        const pane = stringToFragment(paneHtml);
        function readHistoryFile(type, file) {
            return new Promise((resolve) => {
                if (file.type !== type)
                    throw new Error('Invalid file');
                const reader = new FileReader();
                reader.readAsText(file);
                reader.onload = (readEvt) => {
                    const text = readEvt.target?.result;
                    if (typeof text !== 'string')
                        throw new Error('Invalid file');
                    const history = JSON.parse(text);
                    if (!(history instanceof Array))
                        throw new Error('Invalid file');
                    resolve(history);
                };
            });
        }
        const importJSON = pane.querySelector('#pdl-import-json');
        importJSON?.addEventListener('change', (evt) => {
            const file = evt.currentTarget.files?.[0];
            if (!file)
                return;
            readHistoryFile('application/json', file)
                .then((data) => historyDb.bulkAdd(data))
                .then(() => location.reload())
                .catch((err) => alert(err?.message));
        });
        const importTxt = pane.querySelector('#pdl-import-txt');
        importTxt?.addEventListener('change', (evt) => {
            const file = evt.currentTarget.files?.[0];
            if (!file)
                return;
            readHistoryFile('text/plain', file)
                .then((data) => historyDb.bulkAdd(data.map((pid) => ({ pid: Number(pid) }))))
                .then(() => location.reload())
                .catch((err) => alert(err?.message));
        });
        const importJsonBtn = pane.querySelector('#pdl-import-json-btn');
        importJsonBtn?.addEventListener('click', () => importJSON?.click());
        const importTxtBtn = pane.querySelector('#pdl-import-txt-btn');
        importTxtBtn?.addEventListener('click', () => importTxt?.click());
        const exportBtn = pane.querySelector('#pdl-export');
        exportBtn?.addEventListener('click', () => {
            historyDb.getAll().then((datas) => {
                const str = JSON.stringify(datas);
                const dlEle = document.createElement('a');
                dlEle.href = URL.createObjectURL(new Blob([str], { type: 'application/json' }));
                dlEle.download = 'Pixiv Downloader ' + new Date().toLocaleString() + '.json';
                dlEle.click();
                URL.revokeObjectURL(dlEle.href);
            });
        });
        const exportCsvBtn = pane.querySelector('#pdl-export-csv');
        exportCsvBtn?.addEventListener('click', () => {
            historyDb.getAll().then((datas) => {
                const csvData = datas.map((historyData) => {
                    const { pid, userId = '', user = '', title = '', tags = '' } = historyData;
                    return [String(pid), String(userId), user, title, tags ? tags.join(',') : tags];
                });
                csvData.unshift(['id', 'userId', 'user', 'title', 'tags']);
                const csv = generateCsv(csvData);
                const dlEle = document.createElement('a');
                dlEle.href = URL.createObjectURL(csv);
                dlEle.download = 'Pixiv Downloader ' + new Date().toLocaleString() + '.csv';
                dlEle.click();
                URL.revokeObjectURL(dlEle.href);
            });
        });
        const clearBtn = pane.querySelector('#pdl-clear-history');
        clearBtn?.addEventListener('click', () => {
            const isConfirm = confirm(t('text.confirm_clear_history'));
            if (!isConfirm)
                return;
            historyDb.clear().then(() => location.reload());
        });
        return {
            tab,
            pane
        };
    }

    function createTabOthers() {
        const tabHtml = `<div class="pdl-tab-item">${t('text.tab_title_others')}</div>`;
        const paneHtml = `
<div class="pdl-tab-pane">
  <div id="pdl-setting-others">
  ${env.isPixiv()
        ? `<div>
  <label class="pdl-options">
    <input id="pdl-options-bundle-illusts" type="checkbox" class="pdl-checkbox" />
    <span>${t('text.option_bundle_illusts')}</span>
  </label>
</div>
<hr />
<div>
  <label class="pdl-options">
    <input id="pdl-options-bundle-manga" type="checkbox" class="pdl-checkbox" />
    <span>${t('text.option_bundle_manga')}</span>
  </label>
</div>
<hr />`
        : ''}
    <div>
      <label class="pdl-options">
        <input id="pdl-options-add-bookmark" type="checkbox" class="pdl-checkbox" />
        <span>${t('text.option_add_bookmark')}</span>
      </label>
    </div>
    <hr />
    ${env.isPixiv()
        ? `<div>
    <label class="pdl-options sub-option">
      <input id="pdl-options-add-bookmark-tags" type="checkbox" class="pdl-checkbox" />
      <span>${t('text.option_add_bookmark_with_tags')}</span>
    </label>
  </div>
  <hr class="sub" />
  <div>
    <label class="pdl-options sub-option">
      <input id="pdl-options-add-bookmark-private-r18" type="checkbox" class="pdl-checkbox" />
      <span>${t('text.option_add_bookmark_private_r18')}</span>
    </label>
  </div>
  <hr />`
        : ''}
    <div>
      <label class="pdl-options">
        <input id="pdl-options-show-popup-button" type="checkbox" class="pdl-checkbox" />
        <span>${t('text.option_show_popup_button')}</span>
      </label>
    </div>
  </div>
</div>`;
        const tab = stringToFragment(tabHtml);
        const pane = stringToFragment(paneHtml);
        [
            { selector: '#pdl-options-bundle-illusts', settingKey: 'bundleIllusts' },
            { selector: '#pdl-options-bundle-manga', settingKey: 'bundleManga' },
            { selector: '#pdl-options-add-bookmark', settingKey: 'addBookmark' },
            {
                selector: '#pdl-options-add-bookmark-tags',
                settingKey: 'addBookmarkWithTags'
            },
            {
                selector: '#pdl-options-add-bookmark-private-r18',
                settingKey: 'privateR18'
            },
            { selector: '#pdl-options-show-popup-button', settingKey: 'showPopupButton' }
        ].forEach(({ selector, settingKey }) => {
            const optionEl = pane.querySelector(selector);
            if (!optionEl)
                return;
            optionEl.checked = config.get(settingKey);
            optionEl.addEventListener('change', (ev) => {
                config.set(settingKey, ev.currentTarget.checked);
            });
        });
        pane.querySelector('#pdl-options-show-popup-button').addEventListener('change', (ev) => {
            if (ev.currentTarget.checked) {
                dispatchEvent(new CustomEvent("popupBtn.show" ));
            }
            else {
                dispatchEvent(new CustomEvent("popupBtn.hide" ));
            }
        });
        return {
            tab,
            pane
        };
    }

    const regexp = {
        preloadData: /"meta-preload-data" content='(.*?)'>/,
        globalData: /"meta-global-data" content='(.*?)'>/,
        artworksPage: /artworks\/(\d+)$/,
        userPage: /\/users\/(\d+)$|\/users\/(\d+)\/(?!following|mypixiv|followers)/,
        bookmarkPage: /users\/(\d+)\/bookmarks\/artworks/,
        userPageTags: /users\/\d+\/(artworks|illustrations|manga|bookmarks(?=\/artworks))/,
        searchPage: /\/tags\/.*\/(artworks|illustrations|manga)/,
        suscribePage: /bookmark_new_illust/,
        activityHref: /illust_id=(\d+)/,
        originSrcPageNum: /(?<=_p)\d+/,
        followLatest: /\/bookmark_new_illust(?:_r18)?\.php/,
        historyPage: /\/history\.php/,
        historyThumbnailsId: /\d+(?=_)/
    };
    const creditCode = `<img style="display: block; margin: 1em auto; width: 200px"
src=""
/>`;

    function createTabFeedback() {
        const tabHtml = `<div class="pdl-tab-item">${t('text.tab_title_feedback')}</div>`;
        const paneHtml = `
<div class="pdl-tab-pane">
  <div id="pdl-setting-donate">
    ${creditCode}
    <p>如果脚本有帮助到你,欢迎扫码请我喝杯可乐 ^_^</p>
    <p>
      <a
        target="_blank"
        style="color: #0096fa; text-decoration: underline"
        href="https://sleazyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback"
        >${t('text.feedback')}</a
      >
    </p>
  </div>
</div>`;
        return {
            tab: stringToFragment(tabHtml),
            pane: stringToFragment(paneHtml)
        };
    }

    var css$5 = ".pdl-tabs-nav{align-items:center;border-bottom:1px solid #dcdfe6;display:flex;position:relative}.pdl-tabs-nav .pdl-tabs__active-bar{background-color:#0096fa;bottom:-1px;height:2px;left:0;position:absolute;transition:width .2s,transform .2s;z-index:1}.pdl-tabs-nav .pdl-tab-item{cursor:pointer;line-height:2.5;padding:0 16px;transition:color .2s}.pdl-tabs-nav .pdl-tab-item:hover{color:#0096fa}.pdl-tabs-nav .pdl-tab-item.active{color:#0096fa;font-weight:700}.pdl-tabs-nav .pdl-tab-item:nth-child(2){padding-left:0}.pdl-tabs-nav .pdl-tab-item:last-child{padding-right:0}.pdl-tabs-content{min-height:200px;padding:16px}.pdl-tabs-content .option-header{font-weight:700}.pdl-tabs-content #pdl-setting-filename .pdl-input-wrap,.pdl-tabs-content #pdl-setting-filename .tags-option{align-items:center;display:flex;gap:12px;margin:12px 0}.pdl-tabs-content #pdl-setting-filename .pdl-input-wrap label,.pdl-tabs-content #pdl-setting-filename .tags-option .tags-title{cursor:default;font-weight:700;min-width:5em}.pdl-tabs-content #pdl-setting-filename .pdl-input-wrap input[type=text]{border:1px solid #333;flex:1;font-size:16px;height:auto;line-height:1.5;padding:.5em}.pdl-tabs-content #pdl-setting-filename .pdl-input-wrap input[type=text]:focus{background-color:#fff!important}.pdl-tabs-content #pdl-setting-filename .pdl-input-wrap button{line-height:1.5}.pdl-tabs-content #pdl-setting-filename .tags-option .tags-content{display:flex;flex:1;gap:20px}.pdl-tabs-content #pdl-setting-filename .pdl-options{align-items:center;cursor:pointer;display:flex;justify-content:space-between;padding:.6em 0}.pdl-tabs-content #pdl-setting-filename .pdl-options:hover{background-color:var(--pdl-bg2-hover)}.pdl-tabs-content #pdl-setting-ugoira #pdl-ugoira-format-wrap{display:flex;flex-wrap:nowrap;justify-content:space-between;margin:1.5em 1em}.pdl-tabs-content #pdl-setting-ugoira #pdl-ugoira-format-wrap .pdl-ugoira-format-item label{padding-left:4px}.pdl-tabs-content #pdl-setting-history div{margin:1em 0;text-align:center}.pdl-tabs-content #pdl-setting-history div .btn-history{width:80%}.pdl-tabs-content #pdl-setting-others .pdl-options{align-items:center;border-radius:4px;cursor:pointer;display:flex;gap:20px;padding:1em .5em;transition:background-color .2s}.pdl-tabs-content #pdl-setting-others .pdl-options:hover{background-color:var(--pdl-bg2-hover)}.pdl-tabs-content #pdl-setting-others .pdl-options.sub-option{padding:.5em;padding-inline-start:2em}.pdl-tabs-content #pdl-setting-donate{text-align:center}.pdl-tabs-content #pdl-setting-donate p{margin:.5em 0}.pdl-tabs-content .pdl-adjust-button{display:flex;gap:32px;justify-content:space-between}.pdl-tabs-content .pdl-adjust-button .pdl-adjust-content{flex:2}.pdl-tabs-content .pdl-adjust-button .pdl-adjust-content .pdl-adjust-item{margin-bottom:1em}.pdl-tabs-content .pdl-adjust-button .pdl-adjust-content .pdl-adjust-item .pdl-adjust-title{font-weight:700;margin-bottom:.8em}.pdl-tabs-content .pdl-adjust-button .pdl-adjust-content .pdl-adjust-item .pdl-adjust-select{align-items:center;display:flex;gap:20px;margin:.6em 0;padding:0 .4em}.pdl-tabs-content .pdl-adjust-button .pdl-adjust-content .pdl-adjust-item .pdl-adjust-select input[type=range]{flex:1 1;max-width:450px}.pdl-tabs-content .pdl-adjust-button .pdl-adjust-preview{align-self:center;display:flex;flex:1}.pdl-tabs-content .pdl-adjust-button .pdl-adjust-preview .pdl-thumbnail-sample{background-color:rgba(0,150,250,.15);border-radius:4px;height:184px;position:relative;width:184px}";

    function showSettings() {
        if (document.querySelector('pdl-dialog'))
            return;
        const contentHtml = `
<style>${css$5}</style>
<div>
  <div class="pdl-tabs-nav">
    <div class="pdl-tabs__active-bar"></div>
  </div>
  <div class="pdl-tabs-content"></div>
</div>`;
        const content = stringToFragment(contentHtml);
        const tabsNav = content.querySelector('.pdl-tabs-nav');
        const tabContent = content.querySelector('.pdl-tabs-content');
        [
            createTabFilename(),
            createTabUgoira(),
            createTabHistory(),
            createTabAdjustButtonPosition(),
            createTabOthers(),
            createTabFeedback()
        ].forEach(({ tab, pane }) => {
            tabsNav.appendChild(tab);
            tabContent.appendChild(pane);
        });
        const panes = Array.from(tabContent.querySelectorAll('.pdl-tab-pane'));
        panes.forEach((el) => {
            el.style.setProperty('display', 'none');
        });
        const activeBar = tabsNav.querySelector('.pdl-tabs__active-bar');
        const tabs = Array.from(content.querySelectorAll('.pdl-tabs-nav .pdl-tab-item'));
        tabs.forEach((el) => {
            el.addEventListener('click', (ev) => {
                const tab = ev.currentTarget;
                if (!tab)
                    return;
                tabs.forEach((tab) => tab.classList.remove('active'));
                tab.classList.add('active');
                activeBar.style.width = getComputedStyle(tab).width;
                activeBar.style.transform = `translateX(${tab.offsetLeft + parseFloat(getComputedStyle(tab).paddingLeft)}px)`;
                panes.forEach((pane) => pane.style.setProperty('display', 'none'));
                panes[tabs.indexOf(tab)].style.removeProperty('display');
            });
        });
        tabs[0].classList.add('active');
        panes[0].style.removeProperty('display');
        const el = createModal({
            content
        });
        document.body.appendChild(el);
        activeBar.style.width = getComputedStyle(tabs[0]).width;
        activeBar.style.transform = `translateX(${tabs[0].offsetLeft + parseFloat(getComputedStyle(tabs[0]).paddingLeft)}px)`;
    }

    function createPopupBtn(isShow = false) {
        const btn = document.createElement('button');
        btn.className = 'pdl-popup-button';
        btn.innerHTML = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12,15.5A3.5,3.5 0 0,1 8.5,12A3.5,3.5 0 0,1 12,8.5A3.5,3.5 0 0,1 15.5,12A3.5,3.5 0 0,1 12,15.5M19.43,12.97C19.47,12.65 19.5,12.33 19.5,12C19.5,11.67 19.47,11.34 19.43,11L21.54,9.37C21.73,9.22 21.78,8.95 21.66,8.73L19.66,5.27C19.54,5.05 19.27,4.96 19.05,5.05L16.56,6.05C16.04,5.66 15.5,5.32 14.87,5.07L14.5,2.42C14.46,2.18 14.25,2 14,2H10C9.75,2 9.54,2.18 9.5,2.42L9.13,5.07C8.5,5.32 7.96,5.66 7.44,6.05L4.95,5.05C4.73,4.96 4.46,5.05 4.34,5.27L2.34,8.73C2.21,8.95 2.27,9.22 2.46,9.37L4.57,11C4.53,11.34 4.5,11.67 4.5,12C4.5,12.33 4.53,12.65 4.57,12.97L2.46,14.63C2.27,14.78 2.21,15.05 2.34,15.27L4.34,18.73C4.46,18.95 4.73,19.03 4.95,18.95L7.44,17.94C7.96,18.34 8.5,18.68 9.13,18.93L9.5,21.58C9.54,21.82 9.75,22 10,22H14C14.25,22 14.46,21.82 14.5,21.58L14.87,18.93C15.5,18.67 16.04,18.34 16.56,17.94L19.05,18.95C19.27,19.03 19.54,18.95 19.66,18.73L21.66,15.27C21.78,15.05 21.73,14.78 21.54,14.63L19.43,12.97Z" /></svg>`;
        !isShow && (btn.style.display = 'none');
        const popupBtn = {
            show() {
                btn.style.display = 'block';
            },
            hide() {
                btn.style.display = 'none';
            }
        };
        btn.addEventListener('click', () => {
            dispatchEvent(new CustomEvent("popupBtn.click" ));
        });
        addEventListener("popupBtn.show" , popupBtn.show);
        addEventListener("popupBtn.hide" , popupBtn.hide);
        document.body.appendChild(btn);
    }

    function showUpgradeMsg() {
        const headerHtml = `<h3>Pixiv Downloader ${config.get('version')}</h3>`;
        const contentHtml = `
<div class="pdl-changelog">
<style>
  li {
    line-height: 2;
  }
  p {
    margin: 0.5em 0;
  }
</style>
  <ul>
    <li>现在可以在Danbooru上使用插件:<a href="https://danbooru.donmai.us/">Danbooru</a>。</li>
    <li>支持在Rule34的comment页和pool页上下载图片。</li>
    <li>对转换ugoira做了一些优化。</li>
    <li>修正打开插件设置时键盘按键会触发页面跳转的问题。</li>
  </ul>
</div>`;
        const footerHtml = `
<style>
  .pdl-dialog-footer {
    position: relative;
    font-size: 12px;
    margin-top: 1.5em;
  }
</style>
<details>
  <summary style="display: inline-block; list-style: none; cursor: pointer; color: #0096fa; text-decoration: underline">
    脚本还行?请我喝杯可乐吧!
  </summary>
  ${creditCode}
  <p style="text-align: center">愿你每天都能找到对的色图,就像我每天都能喝到香草味可乐</p>
</details>
<a
  target="_blank"
  style="position: absolute; right: 0px; top: 0px; color: #0096fa; text-decoration: underline"
  href="https://sleazyfork.org/zh-CN/scripts/432150-pixiv-downloader/feedback"
  >${t('text.feedback')}
</a>`;
        document.body.appendChild(createModal({
            header: stringToFragment(headerHtml),
            content: stringToFragment(contentHtml),
            footer: stringToFragment(footerHtml)
        }));
    }

    var css$4 = ":root{--pdl-btn-top:100;--pdl-btn-left:0;--pdl-btn-self-bookmark-top:75;--pdl-btn-self-bookmark-left:100}";

    var css$3 = ":root,:root body[data-current-user-theme=light],:root[data-theme=default]{--pdl-bg1:#fff;--pdl-bg2-hover:rgba(0,0,0,.05);--pdl-bg3-hover:#1f1f1f;--pdl-btn1:rgba(0,0,0,.04);--pdl-btn1-hover:rgba(0,0,0,.12);--pdl-border1:rgba(0,0,0,.1);--pdl-text1:#1f1f1f}@media (prefers-color-scheme:light){:root{--pdl-bg1:#fff;--pdl-bg2-hover:rgba(0,0,0,.05);--pdl-bg3-hover:#1f1f1f;--pdl-btn1:rgba(0,0,0,.04);--pdl-btn1-hover:rgba(0,0,0,.12);--pdl-border1:rgba(0,0,0,.1);--pdl-text1:#1f1f1f}}:root body[data-current-user-theme=dark],:root[data-theme=dark]{--pdl-bg1:#1f1f1f;--pdl-bg2-hover:hsla(0,0%,100%,.1);--pdl-bg3-hover:#9b9b9b;--pdl-btn1:hsla(0,0%,100%,.4);--pdl-btn1-hover:hsla(0,0%,100%,.6);--pdl-border1:hsla(0,0%,100%,.3);--pdl-text1:#f5f5f5}@media (prefers-color-scheme:dark){:root{--pdl-bg1:#1f1f1f;--pdl-bg2-hover:hsla(0,0%,100%,.1);--pdl-bg3-hover:#9b9b9b;--pdl-btn1:hsla(0,0%,100%,.4);--pdl-btn1-hover:hsla(0,0%,100%,.6);--pdl-border1:hsla(0,0%,100%,.3);--pdl-text1:#f5f5f5}}pdl-button{--pdl-green1:#01b468;--pdl-black1:#3c3c3c;--pdl-red1:#ea0000}pdl-button,pdl-button[status=init]{--pdl-fill-svg:var(--pdl-black1)}pdl-button[type=pixiv-toolbar]{--pdl-fill-svg:var(--pdl-text1)}pdl-button[status]{--pdl-fill-svg:var(--pdl-green1)}pdl-button[status=error]{--pdl-fill-svg:var(--pdl-red1)}";

    var css$2 = ".pdl-popup-button{background-color:rgba(0,150,250,.5)!important;border:none;border-radius:50%;bottom:100px;color:#fff!important;cursor:pointer;line-height:0;margin:0;opacity:.32!important;padding:12px;position:fixed;right:28px;transition:opacity .3s ease 0s;z-index:1}.pdl-popup-button:hover{opacity:1!important}.pdl-popup-button svg{fill:currentColor;height:24px;width:24px}";

    class SiteInject {
        constructor() {
            this.addStyle();
            this.init();
        }
        init() {
            GM_registerMenuCommand(t('text.gm_menu'), showSettings, 's');
            addEventListener("popupBtn.click" , showSettings);
            if (config.get('showMsg')) {
                showUpgradeMsg();
                config.set('showMsg', false);
            }
            createPopupBtn(config.get('showPopupButton'));
            ['pdl-btn-self-bookmark-left', 'pdl-btn-self-bookmark-top', 'pdl-btn-left', 'pdl-btn-top'].forEach((key) => {
                let val;
                if ((val = config.get(key)) !== undefined) {
                    document.documentElement.style.setProperty('--' + key, val);
                }
            });
        }
        addStyle() {
            [css$7, css$4, css$3, css$6, css$2].forEach((style) => GM_addStyle(style));
        }
    }

    var css$1 = ".pdl-thumbnail{align-items:center;background-color:hsla(0,0%,100%,.5);border:none;border-radius:4px;color:var(--pdl-green1);cursor:pointer;display:flex;font-family:'win-bug-omega, system-ui, -apple-system, \"Segoe UI\", Roboto, Ubuntu, Cantarell, \"Noto Sans\", \"Hiragino Kaku Gothic ProN\", Meiryo, sans-serif';font-size:13px;font-weight:700;height:32px;justify-content:center;left:calc((100% - 32px)*var(--pdl-btn-left)/100);margin:0;overflow:hidden;padding:0;position:absolute;top:calc((100% - 32px)*var(--pdl-btn-top)/100);user-select:none;white-space:nowrap;width:32px;z-index:1}.pdl-thumbnail:disabled{cursor:default}.pdl-thumbnail>svg{fill:var(--pdl-fill-svg);stroke:var(--pdl-fill-svg);height:85%;position:absolute;width:85%}.pdl-thumbnail>span{opacity:0;transition:opacity .2s}.pdl-thumbnail>span.show{opacity:1}:host([type=gallery])::part(button){left:0;position:sticky;top:40px}:host([type=pixiv-my-bookmark])::part(button){left:calc((100% - 32px)*var(--pdl-btn-self-bookmark-left)/100);top:calc((100% - 32px)*var(--pdl-btn-self-bookmark-top)/100)}:host([type=pixiv-history])::part(button){z-index:auto}:host([type=pixiv-presentation])::part(button){left:auto;position:fixed;right:20px;top:50px}:host([type=pixiv-toolbar])::part(button){background-color:transparent;left:auto;position:relative;top:auto}:host([type=pixiv-manga-viewer])::part(button){left:auto;right:4px;top:80%}";

    var svgGroup = "<svg xmlns=\"http://www.w3.org/2000/svg\" style=\"display: none\">\n  <symbol id=\"pdl-download\" viewBox=\"0 0 512 512\">\n    <path\n      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\"\n    ></path>\n  </symbol>\n\n  <symbol id=\"pdl-loading\" viewBox=\"0 0 512 512\">\n    <style>\n      @keyframes pdl-loading {\n        0% {\n          transform: rotate3d(0, 0, 1, -90deg) rotate3d(1, 0, 0, 0deg);\n          stroke-dashoffset: 1407.43;\n        }\n\n        49.99% {\n          transform: rotate3d(0, 0, 1, 90deg) rotate3d(1, 0, 0, 0deg);\n        }\n\n        50% {\n          transform: rotate3d(0, 0, 1, 90deg) rotate3d(1, 0, 0, 180deg);\n          stroke-dashoffset: 0;\n        }\n\n        100% {\n          transform: rotate3d(0, 0, 1, 270deg) rotate3d(1, 0, 0, 180deg);\n          stroke-dashoffset: 1407.43;\n        }\n      }\n\n      circle.rotate {\n        transform-origin: 50% 50%;\n        animation: 2.5s infinite ease-in-out pdl-loading;\n      }\n    </style>\n    <circle\n      class=\"rotate\"\n      cx=\"256\"\n      cy=\"256\"\n      r=\"224\"\n      stroke-width=\"48\"\n      fill=\"none\"\n      stroke-dasharray=\"1407.43\"\n      stroke-dashoffset=\"1055.57\"\n      stroke-linecap=\"round\"\n    ></circle>\n  </symbol>\n\n  <symbol id=\"pdl-progress\" viewBox=\"0 0 512 512\">\n    <style>\n      circle.progress {\n        transition: stroke-dashoffset 0.2s ease;\n      }\n    </style>\n    <circle\n      class=\"progress\"\n      cx=\"256\"\n      cy=\"256\"\n      r=\"224\"\n      stroke-width=\"48\"\n      fill=\"none\"\n      stroke-dasharray=\"1407.43\"\n      stroke-linecap=\"round\"\n      transform=\"rotate(-90 256 256)\"\n    ></circle>\n  </symbol>\n\n  <symbol id=\"pdl-error\" viewBox=\"0 0 512 512\">\n    <path\n      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\"\n    ></path>\n  </symbol>\n\n  <symbol id=\"pdl-complete\" viewBox=\"0 0 512 512\">\n    <path\n      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\"\n    ></path>\n  </symbol>\n</svg>\n";

    const iconTypeMap = {
        init: '#pdl-download',
        loading: '#pdl-loading',
        progress: '#pdl-progress',
        complete: '#pdl-complete',
        error: '#pdl-error'
    };
    class ThumbnailButton extends HTMLElement {
        status;
        mediaId;
        page;
        type;
        onClick;
        constructor(props) {
            super();
            this.status = "init" ;
            this.mediaId = props.id;
            this.page = props.page;
            this.type = props.type;
            this.onClick = props.onClick;
            this.render();
        }
        static get observedAttributes() {
            return ['status', 'page', 'disabled'];
        }
        attributeChangedCallback(name, oldValue, newValue) {
            if (name === 'status') {
                this.updateIcon(newValue);
            }
            else if (name === 'page') {
                this.updatePage(newValue);
            }
            else {
                this.updateDisableStatus(newValue);
            }
        }
        updateDisableStatus(val) {
            const btn = this.shadowRoot.querySelector('button');
            if (typeof val === 'string') {
                btn.setAttribute('disabled', '');
            }
            else {
                btn.removeAttribute('disabled');
            }
        }
        updatePage(page) {
            const pageNum = Number(page);
            if (!Number.isNaN(pageNum) && pageNum >= 0) {
                this.page = pageNum;
            }
        }
        updateIcon(status) {
            if (status === null || !(status in iconTypeMap))
                return;
            const useEl = this.shadowRoot.querySelector('use');
            this.status = status;
            useEl.setAttribute('xlink:href', iconTypeMap[status]);
            useEl.animate([
                {
                    opacity: 0.5
                },
                {
                    opactiy: 1
                }
            ], {
                duration: 200
            });
        }
        render() {
            const shadowRoot = this.attachShadow({ mode: 'open' });
            shadowRoot.innerHTML = `    <style>${css$1}</style>${svgGroup}<button part="button" class="pdl-thumbnail">
      <svg xmlns="http://www.w3.org/2000/svg" class="pdl-icon">
        <use xlink:href="#pdl-download"></use>
      </svg>
      <span></span>
    </button>`;
            this.type !== "danbooru-pool"  &&
                historyDb.has(this.mediaId).then((downloaded) => {
                    downloaded && this.setStatus("complete" );
                });
            this.setAttribute('pdl-id', this.mediaId);
            this.page !== undefined && !Number.isNaN(this.page) && this.setAttribute('page', String(this.page));
            this.type && this.setAttribute('type', this.type);
        }
        connectedCallback() {
            this.shadowRoot.lastElementChild.addEventListener('click', (evt) => {
                evt.preventDefault();
                evt.stopPropagation();
                this.setAttribute('disabled', '');
                this.setStatus("loading" );
                Promise.resolve(this.onClick(this))
                    .then(() => {
                    this.setStatus("complete" );
                }, (err) => {
                    if (err)
                        logger.error(err);
                    this.setStatus("error" );
                })
                    .finally(() => {
                    this.removeAttribute('disabled');
                });
            });
        }
        setProgress(progress, updateProgressbar = true) {
            if (progress < 0 || progress > 100)
                throw new RangeError('Value "progress" must between 0-100');
            const shadowRoot = this.shadowRoot;
            const span = shadowRoot.querySelector('span');
            if (this.status !== "progress" ) {
                this.setAttribute('status', "progress" );
                span.classList.toggle('show');
            }
            span.textContent = String(Math.floor(progress));
            if (!updateProgressbar)
                return;
            const svg = shadowRoot.querySelector('svg.pdl-icon');
            const radius = 224;
            const circumference = 2 * Math.PI * radius;
            const offset = circumference - (progress / 100) * circumference;
            svg.style.strokeDashoffset = String(offset);
        }
        removeProgress() {
            const shadowRoot = this.shadowRoot;
            const span = shadowRoot.querySelector('span');
            const svg = shadowRoot.querySelector('svg.pdl-icon');
            span.classList.toggle('show');
            span.addEventListener('transitionend', () => {
                span.textContent = '';
            }, { once: true });
            svg.style.removeProperty('stroke-dashoffset');
            if (this.status === "progress" )
                this.setAttribute('status', "init" );
        }
        setStatus(status) {
            if (status !== this.status) {
                if (status === "progress" ) {
                    this.setProgress(0);
                    return;
                }
                if (this.status === "progress" ) {
                    this.removeProgress();
                }
                this.setAttribute('status', status);
            }
        }
    }
    customElements.define('pdl-button', ThumbnailButton);

    class Rule34 extends SiteInject {
        init() {
            super.init();
            this.pageAction();
        }
        createThumbnailBtn() {
            const btnContainers = document.querySelectorAll('.thumb:not(.blacklisted-image) > a:first-child');
            if (!btnContainers.length)
                return;
            btnContainers.forEach((el) => {
                el.style.display = 'inline-block';
                el.style.position = 'relative';
                const imgEl = el.querySelector('img');
                imgEl.style.boxSizing = 'border-box';
                let aspectRatio = imgEl.naturalHeight / imgEl.naturalWidth;
                aspectRatio > 1 && (el.style.height = 'inherit');
                imgEl.onload = () => {
                    aspectRatio = imgEl.naturalHeight / imgEl.naturalWidth;
                    aspectRatio > 1 && (el.style.height = 'inherit');
                };
                const idMathch = /(?<=&id=)\d+/.exec(el.href);
                if (!idMathch)
                    return;
                const id = idMathch[0];
                el.appendChild(new ThumbnailButton({
                    id,
                    onClick: downloadArtwork$2
                }));
            });
        }
        createArtworkBtn(id) {
            const btnContainer = document.querySelector('div.flexi > div');
            btnContainer.style.position = 'relative';
            const wrapper = document.createElement('div');
            wrapper.classList.add('pdl-wrap-artworks', 'rule34');
            const btn = new ThumbnailButton({
                id,
                type: "gallery" ,
                onClick: downloadArtwork$2
            });
            wrapper.appendChild(btn);
            btnContainer.appendChild(wrapper);
        }
        pageAction() {
            const query = location.search;
            if (!query)
                return;
            const searchParams = new URLSearchParams(query);
            const page = searchParams.get('page');
            const s = searchParams.get('s');
            if (page === 'post' && s === 'view') {
                if (!document.querySelector('#image, #gelcomVideoPlayer'))
                    return;
                const id = searchParams.get('id');
                this.createArtworkBtn(id);
            }
            else {
                this.createThumbnailBtn();
            }
        }
    }

    async function addBookmark$1(id) {
        try {
            const token = document.head.querySelector('meta[name="csrf-token"]')?.content;
            if (!token)
                throw new Error('Can not get csrf-token');
            const res = await fetch('/favorites?post_id=' + id, {
                method: 'POST',
                headers: {
                    'X-Csrf-Token': token
                }
            });
            if (!res.ok)
                throw new Error(res.status + ' ' + res.statusText);
            const galleryMatch = /(?<=^\/posts\/)\d+/.exec(location.pathname);
            if (galleryMatch && id !== galleryMatch[0]) {
                unsafeWindow.Danbooru.Utility.notice('You have favorited ' + id);
            }
            else {
                const script = await res.text();
                evalScript(script);
            }
        }
        catch (error) {
            logger.error(error);
        }
    }

    const danbooruParser = {
        async getDoc(url) {
            const res = await fetch(url);
            if (!res.ok)
                throw new RequestError('Request failed with status code ' + res.status, res);
            const html = await res.text();
            return new DOMParser().parseFromString(html, 'text/html');
        },
        async parse(id) {
            const doc = await this.getDoc('/posts/' + id);
            const src = doc.querySelector('a[download]')?.href;
            if (!src)
                throw new Error('Can not get media src');
            const ogImageMeta = doc.querySelector('meta[property="og:image"]');
            const mediaSrc = ogImageMeta.getAttribute('content');
            const title = mediaSrc.slice(mediaSrc.lastIndexOf('/') + 1).split('.')[0];
            const ogTypeMeta = doc.querySelector('meta[property="og:video:type"]') || doc.querySelector('meta[property="og:image:type"]');
            const mimeType = ogTypeMeta.getAttribute('content');
            const extendName = mimeType.slice(mimeType.lastIndexOf('/') + 1);
            const artists = [];
            const characters = [];
            const tags = [];
            const tagLists = doc.querySelectorAll('section#tag-list  ul[class*="-tag-list"]');
            if (tagLists.length) {
                tagLists.forEach((ul) => {
                    const tagTypeMatch = /[a-zA-Z]+(?=-tag-list)/.exec(ul.className);
                    if (!tagTypeMatch)
                        throw new Error('Unknown tag: ' + ul.className);
                    const tagType = tagTypeMatch[0];
                    const liEls = ul.children;
                    let tagRef;
                    if (tagType === 'artist') {
                        tagRef = artists;
                    }
                    else if (tagType === 'character') {
                        tagRef = characters;
                    }
                    for (let i = 0; i < liEls.length; i++) {
                        const tag = liEls[i].getAttribute('data-tag-name');
                        if (!tag)
                            continue;
                        tagRef && tagRef.push(tag);
                        tags.push(tagType + ':' + tag);
                    }
                });
            }
            const postDate = doc.querySelector('time')?.getAttribute('datetime') ?? '';
            const source = doc.querySelector('li#post-info-source > a')?.href;
            if (source)
                tags.push('source:' + source);
            return {
                id,
                src,
                extendName,
                artist: artists.join(',') || 'UnknownArtist',
                character: characters.join(',') || 'UnknownCharacter',
                title,
                tags,
                createDate: postDate
            };
        },
        async getPoolPostCount(poolId) {
            const doc = await this.getDoc(`/pools/${poolId}`);
            const nextEl = doc.querySelector('a.paginator-next');
            if (nextEl) {
                const lastPageEl = nextEl.previousElementSibling;
                const poolPageCount = Number(lastPageEl.textContent);
                const lastPageDoc = await this.getDoc(lastPageEl.href);
                const postPerPage = Number(lastPageDoc.body.getAttribute('data-current-user-per-page'));
                const lastPagePostCount = lastPageDoc.querySelectorAll('.posts-container article').length;
                return (poolPageCount - 1) * postPerPage + lastPagePostCount;
            }
            else {
                const imageContainers = doc.querySelectorAll('.posts-container article');
                return imageContainers.length;
            }
        },
        async *genIdByPool(poolId, filter) {
            let page = 0;
            let nextUrl;
            do {
                ++page > 1 && (await sleep(1000));
                const doc = await this.getDoc(`/pools/${poolId}?page=${page}`);
                const nextEl = doc.querySelector('a.paginator-next');
                nextUrl = nextEl?.getAttribute('href') ?? '';
                const imageContainers = doc.querySelectorAll('.posts-container article');
                const ids = Array.from(imageContainers).map((el) => el.getAttribute('data-id'));
                for (let i = 0; i < ids.length; i++) {
                    const id = ids[i];
                    const isValid = (await filter?.(id)) ?? true;
                    if (isValid) {
                        yield id;
                        i !== id.length - 1 && (await sleep(1000));
                    }
                }
            } while (nextUrl);
        }
    };

    function artworkProgressFactory(btn) {
        if (!btn)
            return;
        return function onArtworkProgress(progress) {
            btn.setProgress(progress);
        };
    }
    class DanbooruDownloadConfig extends DownloadConfigBuilder {
        meta;
        constructor(meta) {
            super(meta);
            this.meta = meta;
        }
        getDownloadConfig(btn) {
            return {
                taskId: Math.random().toString(36).slice(2),
                src: this.meta.src,
                path: this.buildFilePath(),
                source: this.meta,
                timeout: this.isImage() ? 60000 : undefined,
                directSave: downloader.fileSystemAccessEnabled ? false : true,
                onProgress: artworkProgressFactory(btn)
            };
        }
        buildFilePath() {
            const path = super.buildFilePath();
            return path.replaceAll('{character}', this.normalizeString(this.meta.character));
        }
    }

    async function downloadArtwork$1(btn) {
        downloader.dirHandleCheck();
        const id = btn.getAttribute('pdl-id');
        const mediaMeta = await danbooruParser.parse(id);
        const { tags, artist, title } = mediaMeta;
        const downloadConfigs = new DanbooruDownloadConfig(mediaMeta).getDownloadConfig(btn);
        config.get('addBookmark') && addBookmark$1(id);
        await downloader.download(downloadConfigs);
        const historyData = {
            pid: Number(id),
            user: artist,
            title,
            tags
        };
        historyDb.add(historyData);
    }

    async function downloadPoolArtwork(btn) {
        downloader.dirHandleCheck();
        const poolId = btn.getAttribute('pdl-id');
        const promises = [];
        const postCount = await danbooruParser.getPoolPostCount(poolId);
        let completed = 0;
        const filter = async (id) => !(await historyDb.has(id));
        const idGen = danbooruParser.genIdByPool(poolId, filter);
        for await (const id of idGen) {
            const mediaMeta = await danbooruParser.parse(id);
            const downloadConfigs = new DanbooruDownloadConfig(mediaMeta).getDownloadConfig();
            const p = downloader
                .download(downloadConfigs)
                .then(() => {
                completed++;
                btn.setProgress((completed / postCount) * 100);
            })
                .then(() => {
                const { tags, artist, title } = mediaMeta;
                const historyData = {
                    pid: Number(id),
                    user: artist,
                    title,
                    tags
                };
                historyDb.add(historyData);
            });
            promises.push(p);
        }
        const results = await Promise.allSettled(promises);
        const rejectedTasks = results.filter((result) => result.status === 'rejected');
        if (rejectedTasks.length) {
            rejectedTasks.length > 1 && logger.error(rejectedTasks);
            throw rejectedTasks[0].reason;
        }
    }

    class Danbooru extends SiteInject {
        init() {
            super.init();
            this.pageAction();
        }
        createThumbnailBtn() {
            const btnContainers = document.querySelectorAll('article a.post-preview-link');
            if (!btnContainers.length)
                return;
            btnContainers.forEach((el) => {
                const id = /(?<=\/posts\/)\d+/.exec(el.href)?.[0];
                if (!id)
                    return;
                const btn = new ThumbnailButton({
                    id,
                    onClick: downloadArtwork$1
                });
                el.appendChild(btn);
            });
        }
        createArtworkBtn(id) {
            const btnContainer = document.querySelector('section.image-container');
            const btn = new ThumbnailButton({
                id,
                type: "gallery" ,
                onClick: downloadArtwork$1
            });
            const wrapper = document.createElement('div');
            wrapper.classList.add('pdl-wrap-artworks');
            wrapper.appendChild(btn);
            btnContainer.appendChild(wrapper);
        }
        createPoolThumbnailBtn() {
            const btnContainers = document.querySelectorAll('article a.post-preview-link');
            if (!btnContainers.length)
                return;
            btnContainers.forEach((el) => {
                const poolId = /(?<=\/pools\/)\d+/.exec(el.href)?.[0];
                if (!poolId)
                    return;
                const btn = new ThumbnailButton({
                    id: poolId,
                    type: "danbooru-pool" ,
                    onClick: downloadPoolArtwork
                });
                el.appendChild(btn);
            });
        }
        pageAction() {
            const path = location.pathname;
            if (/^\/posts\/\d+/.test(path)) {
                const imageContainer = document.querySelector('section.image-container:not(.blacklisted-active)');
                if (!imageContainer)
                    return;
                const id = imageContainer.getAttribute('data-id');
                this.createArtworkBtn(id);
                this.createThumbnailBtn();
            }
            else if (/^\/pools\/gallery/.test(path)) {
                this.createPoolThumbnailBtn();
            }
            else {
                this.createThumbnailBtn();
            }
        }
    }

    function getSelfId() {
        return document.querySelector('#qualtrics_user-id')?.textContent ?? '';
    }

    function getIllustId(node) {
        const isLinkToArtworksPage = regexp.artworksPage.exec(node.getAttribute('href') || '');
        if (isLinkToArtworksPage) {
            if (node.getAttribute('data-gtm-value') ||
                [
                    'gtm-illust-recommend-node-node',
                    'gtm-discover-user-recommend-node',
                    'work',
                    '_history-item',
                    '_history-related-item'
                ].some((className) => node.classList.contains(className))) {
                return isLinkToArtworksPage[1];
            }
        }
        else if (node.className.includes('_history-item')) {
            const result = regexp.historyThumbnailsId.exec(node.getAttribute('style') || '');
            if (result)
                return result[0];
        }
        else {
            const isActivityThumb = regexp.activityHref.exec(node.getAttribute('href') || '');
            if (isActivityThumb && node.classList.contains('work')) {
                return isActivityThumb[1];
            }
        }
        return '';
    }

    var IllustType;
    (function (IllustType) {
        IllustType[IllustType["illusts"] = 0] = "illusts";
        IllustType[IllustType["manga"] = 1] = "manga";
        IllustType[IllustType["ugoira"] = 2] = "ugoira";
    })(IllustType || (IllustType = {}));
    var BookmarkRestrict;
    (function (BookmarkRestrict) {
        BookmarkRestrict[BookmarkRestrict["public"] = 0] = "public";
        BookmarkRestrict[BookmarkRestrict["private"] = 1] = "private";
    })(BookmarkRestrict || (BookmarkRestrict = {}));

    function createService() {
        async function _requestJson(url, init) {
            logger.info('fetch url:', url);
            const res = await fetch(url, init);
            if (!res.ok)
                throw new RequestError('Request ' + url + ' failed with status code ' + res.status, res);
            const data = await res.json();
            if (data.error)
                throw new JsonDataError(data.message);
            return data.body;
        }
        return {
            async getJson(url) {
                return await _requestJson(url);
            },
            async getArtworkHtml(illustId) {
                logger.info('Fetch illust:', illustId);
                let params = '';
                const tagLang = config.get('tagLang');
                if (tagLang !== 'ja')
                    params = '?lang=' + tagLang;
                const res = await fetch('https://www.pixiv.net/artworks/' + illustId + params);
                if (!res.ok)
                    throw new RequestError('Request failed with status code ' + res.status, res);
                return await res.text();
            },
            addBookmark(illustId, token, tags = [], restrict = BookmarkRestrict.public) {
                return _requestJson('/ajax/illusts/bookmarks/add', {
                    method: 'POST',
                    headers: {
                        accept: 'application/json',
                        'content-type': 'application/json; charset=utf-8',
                        'x-csrf-token': token
                    },
                    body: JSON.stringify({
                        illust_id: illustId,
                        restrict,
                        comment: '',
                        tags
                    })
                });
            },
            getFollowLatestWorks(page, mode = 'all') {
                return _requestJson(`/ajax/follow_latest/illust?p=${page}&mode=${mode}&lang=jp`);
            },
            getUserAllProfile(userId) {
                return _requestJson('/ajax/user/' + userId + '/profile/all');
            },
            getUgoiraMeta(illustId) {
                return _requestJson('/ajax/illust/' + illustId + '/ugoira_meta');
            },
            getArtworkDetail(illustId) {
                let params = '';
                const tagLang = config.get('tagLang');
                if (tagLang !== 'ja')
                    params = '?lang=' + tagLang;
                return _requestJson('/ajax/illust/' + illustId + params);
            }
        };
    }
    const api = createService();

    function addBookmark(btn, illustId, token, tags) {
        if (!config.get('addBookmark'))
            return;
        api
            .addBookmark(illustId, token, config.get('addBookmarkWithTags') ? tags : [], config.get('privateR18') && tags.includes('R-18') ? BookmarkRestrict.private : BookmarkRestrict.public)
            .then(() => {
            const bookmarkBtnRef = findBookmarkBtn(btn);
            if (!bookmarkBtnRef)
                return;
            switch (bookmarkBtnRef.kind) {
                case "main" : {
                    const pathBorder = bookmarkBtnRef.button.querySelector('svg g path');
                    pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)');
                    break;
                }
                case "sub" : {
                    const pathBorder = bookmarkBtnRef.button.querySelector('path');
                    pathBorder && (pathBorder.style.color = 'rgb(255, 64, 96)');
                    break;
                }
                case "rank" : {
                    bookmarkBtnRef.button.style.backgroundColor = 'rgb(255, 64, 96)';
                    break;
                }
            }
        })
            .catch((reason) => {
            logger.error(reason.message);
        });
    }
    function findBookmarkBtn(btn) {
        const bookmarkBtnRef = {};
        if (!btn.getAttribute('type')) {
            const favBtn = btn.parentElement?.nextElementSibling?.querySelector('button[type="button"]');
            if (favBtn) {
                bookmarkBtnRef.kind = "sub" ;
                bookmarkBtnRef.button = favBtn;
            }
            else {
                const favBtn = btn.parentElement?.querySelector('div._one-click-bookmark');
                if (favBtn) {
                    bookmarkBtnRef.kind = "rank" ;
                    bookmarkBtnRef.button = favBtn;
                }
            }
        }
        else if (btn.getAttribute('type') === "pixiv-toolbar" ) {
            const favBtn = btn.parentElement?.parentElement?.querySelector('button.gtm-main-bookmark');
            if (favBtn) {
                bookmarkBtnRef.kind = "main" ;
                bookmarkBtnRef.button = favBtn;
            }
        }
        else {
            return logger.warn(new Error('Can not find bookmark button.'));
        }
        return bookmarkBtnRef;
    }

    function isValidIllustType(illustType, option) {
        switch (illustType) {
            case IllustType.illusts:
                if (option.filterIllusts)
                    return true;
                break;
            case IllustType.manga:
                if (option.filterManga)
                    return true;
                break;
            case IllustType.ugoira:
                if (option.filterUgoira)
                    return true;
                break;
            default:
                throw new Error('Invalid filter type');
        }
        return false;
    }
    async function filterWorks(works, option) {
        const obj = {
            unavaliable: [],
            avaliable: [],
            invalid: []
        };
        for (const work of works) {
            if (!work.isBookmarkable) {
                obj.unavaliable.push(work.id);
            }
            else if (option.filterExcludeDownloaded && (await historyDb.has(work.id))) {
                obj.invalid.push(work.id);
            }
            else if (!isValidIllustType(work.illustType, option)) {
                obj.invalid.push(work.id);
            }
            else {
                obj.avaliable.push(work.id);
            }
        }
        return obj;
    }
    const pixivParser = {
        async parse(illustId) {
            const htmlText = await api.getArtworkHtml(illustId);
            const preloadDataText = htmlText.match(regexp.preloadData);
            if (!preloadDataText)
                throw new Error('Fail to parse preload data.');
            const globalDataText = htmlText.match(regexp.globalData);
            if (!globalDataText)
                throw new Error('Fail to parse global data.');
            const preloadData = JSON.parse(preloadDataText[1]);
            const globalData = JSON.parse(globalDataText[1]);
            const illustData = preloadData.illust[illustId];
            const { illustType, userName, userId, illustTitle, tags, pageCount, createDate, urls, bookmarkData } = illustData;
            const { token } = globalData;
            const tagsArr = [];
            const tagsTranslatedArr = [];
            tags.tags.forEach((tagData) => {
                tagsArr.push(tagData.tag);
                tagsTranslatedArr.push(tagData.translation?.en || tagData.tag);
            });
            const meta = {
                id: illustId,
                src: urls.original,
                extendName: urls.original.slice(-3),
                artist: userName,
                title: illustTitle,
                tags: tagsArr,
                tagsTranslated: tagsTranslatedArr,
                userId,
                pageCount,
                bookmarkData,
                createDate,
                token
            };
            if (illustType === IllustType.ugoira) {
                return {
                    ...meta,
                    illustType,
                    ugoiraMeta: await api.getUgoiraMeta(illustId)
                };
            }
            else {
                return {
                    ...meta,
                    illustType
                };
            }
        },
        async getFollowLatestGenerator(filterOption, mode, page) {
            const MAX_PAGE = 34;
            const MAX_ILLUSTS_PER_PAGE = 60;
            let lastId;
            let total;
            let data;
            let cache;
            function findLastId(ids) {
                return Math.min(...ids.map((id) => Number(id)));
            }
            if (page === undefined) {
                data = await api.getFollowLatestWorks(1, mode);
                const ids = data.page.ids;
                total = ids.length;
                lastId = findLastId(ids);
                if (total === MAX_ILLUSTS_PER_PAGE) {
                    const secondPageData = await api.getFollowLatestWorks(2, mode);
                    const secondIds = secondPageData.page.ids;
                    const secondLastId = findLastId(secondIds);
                    if (secondLastId < lastId) {
                        lastId = secondLastId;
                        cache = secondPageData;
                        total += secondIds.length;
                    }
                }
            }
            else {
                data = await api.getFollowLatestWorks(page, mode);
                total = data.page.ids.length;
            }
            async function* generateIds() {
                yield await filterWorks(data.thumbnails.illust, filterOption);
                if (page === undefined) {
                    if (total === MAX_ILLUSTS_PER_PAGE)
                        return;
                    if (total < MAX_ILLUSTS_PER_PAGE * 2) {
                        yield await filterWorks(cache.thumbnails.illust, filterOption);
                        return;
                    }
                    let currentPage = 3;
                    while (currentPage <= MAX_PAGE) {
                        const data = await api.getFollowLatestWorks(currentPage, mode);
                        const ids = data.page.ids;
                        const pageLastId = findLastId(ids);
                        if (pageLastId >= lastId) {
                            logger.info('getFollowLatestGenerator: got duplicate works');
                            yield await filterWorks(cache.thumbnails.illust, filterOption);
                            break;
                        }
                        lastId = pageLastId;
                        total += ids.length;
                        yield { ...(await filterWorks(cache.thumbnails.illust, filterOption)), total };
                        cache = data;
                        currentPage++;
                        await sleep(3000);
                    }
                }
            }
            return {
                total,
                generator: generateIds()
            };
        },
        async getChunksGenerator(userId, category, tag, rest, filterOption) {
            const OFFSET = 48;
            let requestUrl;
            if (category === 'bookmarks') {
                requestUrl = `https://www.pixiv.net/ajax/user/${userId}/illusts/bookmarks?tag=${tag}&offset=0&limit=${OFFSET}&rest=${rest}&lang=ja`;
            }
            else {
                requestUrl = `https://www.pixiv.net/ajax/user/${userId}/${category}/tag?tag=${tag}&offset=0&limit=${OFFSET}&lang=ja`;
            }
            let head = 0;
            const firstPageData = await api.getJson(requestUrl);
            const total = firstPageData.total;
            async function* generateIds() {
                yield await filterWorks(firstPageData.works, filterOption);
                head += OFFSET;
                while (head < total) {
                    const data = await api.getJson(requestUrl.replace('offset=0', 'offset=' + head));
                    head += OFFSET;
                    await sleep(3000);
                    yield await filterWorks(data.works, filterOption);
                }
            }
            return {
                total,
                generator: generateIds()
            };
        },
        async getAllWorksGenerator(userId, filterOption) {
            const profile = await api.getUserAllProfile(userId);
            let illustIds = [];
            let mangaIds = [];
            if ((filterOption.filterIllusts || filterOption.filterUgoira) && typeof profile.illusts === 'object') {
                illustIds.push(...Object.keys(profile.illusts).reverse());
            }
            if (filterOption.filterManga && typeof profile.manga === 'object') {
                mangaIds.push(...Object.keys(profile.manga).reverse());
            }
            if (filterOption.filterExcludeDownloaded) {
                const filteredIllustIds = [];
                for (const id of illustIds) {
                    const isDownloaded = await historyDb.has(id);
                    !isDownloaded && filteredIllustIds.push(id);
                }
                illustIds = filteredIllustIds;
                const filteredMangaIds = [];
                for (const id of mangaIds) {
                    const isDownloaded = await historyDb.has(id);
                    !isDownloaded && filteredMangaIds.push(id);
                }
                mangaIds = filteredMangaIds;
            }
            async function* generateIds() {
                const OFFSET = 48;
                const baseUrl = 'https://www.pixiv.net/ajax/user/' + userId + '/profile/illusts';
                let workCategory = 'illust';
                while (illustIds.length > 0) {
                    let searchStr = '?';
                    const chunk = illustIds.splice(0, OFFSET);
                    searchStr +=
                        chunk.map((id) => 'ids[]=' + id).join('&') + `&work_category=${workCategory}&is_first_page=0&lang=ja`;
                    const data = await api.getJson(baseUrl + searchStr);
                    await sleep(3000);
                    yield await filterWorks(Object.values(data.works).reverse(), filterOption);
                }
                workCategory = 'manga';
                while (mangaIds.length > 0) {
                    let searchStr = '?';
                    const chunk = mangaIds.splice(0, OFFSET);
                    searchStr +=
                        chunk.map((id) => 'ids[]=' + id).join('&') + `&work_category=${workCategory}&is_first_page=0&lang=ja`;
                    const data = await api.getJson(baseUrl + searchStr);
                    await sleep(3000);
                    yield await filterWorks(Object.values(data.works).reverse(), filterOption);
                }
            }
            return {
                total: illustIds.length + mangaIds.length,
                generator: generateIds()
            };
        }
    };

    function createCompressor() {
        const zip = new JSZip__default["default"]();
        return {
            add(id, name, data) {
                zip.folder(id)?.file(name, data);
            },
            bundle(id) {
                const folder = zip.folder(id);
                if (!folder)
                    throw new TypeError('no such folder:' + id);
                return folder.generateAsync({ type: 'blob' });
            },
            remove(ids) {
                if (typeof ids === 'string') {
                    zip.remove(ids);
                }
                else {
                    const dirs = zip.filter((_, file) => file.dir).map((dir) => dir.name);
                    const dirsToDel = ids.filter((id) => dirs.some((dir) => dir.includes(id)));
                    dirsToDel.forEach((dir) => zip.remove(dir));
                    logger.info('Compressor: Remove', zip);
                }
            },
            fileCount(id) {
                let count = 0;
                zip.folder(id)?.forEach(() => count++);
                return count;
            },
            async unzip(data) {
                const id = Math.random().toString(36);
                let folder = zip.folder(id);
                if (!folder)
                    throw TypeError('Can not get new root folder');
                const filesPromises = [];
                folder = await folder.loadAsync(data);
                folder.forEach((_, file) => {
                    filesPromises.push(file.async('blob'));
                });
                const files = await Promise.all(filesPromises);
                zip.remove(id);
                return files;
            }
        };
    }
    const compressor = createCompressor();

    const workerUrl$2 = URL.createObjectURL(new Blob([GM_getResourceText('gifWorker')], { type: 'text/javascript' }));
    function gif(frames, convertMeta) {
        return Promise.all(frames.map((frame) => createImageBitmap(frame))).then((bitmaps) => {
            return new Promise((resolve, reject) => {
                logger.info('Start convert:', convertMeta.id);
                logger.time(convertMeta.id);
                const canvas = document.createElement('canvas');
                const width = (canvas.width = bitmaps[0].width);
                const height = (canvas.height = bitmaps[0].height);
                const ctx = canvas.getContext('2d', { willReadFrequently: true });
                const gif = new GIF__default["default"]({
                    workers: 2,
                    quality: 10,
                    width,
                    height,
                    workerScript: workerUrl$2
                });
                convertMeta.abort = () => {
                    gif.abort();
                };
                bitmaps.forEach((bitmap, i) => {
                    ctx.drawImage(bitmap, 0, 0);
                    gif.addFrame(ctx, {
                        copy: true,
                        delay: convertMeta.source.delays[i]
                    });
                });
                gif.on('progress', (progress) => {
                    convertMeta.onProgress?.(progress * 100);
                });
                gif.on('finished', (gifBlob) => {
                    logger.timeEnd(convertMeta.id);
                    resolve(gifBlob);
                });
                gif.on('abort', () => {
                    logger.timeEnd(convertMeta.id);
                    logger.warn('Convert stop manually. ' + convertMeta.id);
                    convertMeta.isAborted = true;
                    reject(new CancelError());
                });
                gif.render();
            });
        });
    }

    var pngWorkerFragment = "onmessage = async (evt) => {\n  const { frames, delay, cnum = 0 } = evt.data;\n  const bitmaps = await Promise.all(frames.map((blob) => createImageBitmap(blob)));\n\n  const width = bitmaps[0].width;\n  const height = bitmaps[0].height;\n  const canvas = new OffscreenCanvas(width, height);\n  const ctx = canvas.getContext('2d', { willReadFrequently: true });\n  const u8arrs = [];\n\n  for (let i = 0; i < bitmaps.length; i++) {\n    ctx?.drawImage(bitmaps[i], 0, 0);\n    u8arrs.push(ctx?.getImageData(0, 0, width, height).data);\n  }\n\n  const png = UPNG.encode(u8arrs, width, height, cnum, delay, { loop: 0 });\n  if (!png) console.error('Convert Apng failed.');\n  postMessage(png, [png]);\n};\n";

    const pako = GM_getResourceText('pako');
    const upng = GM_getResourceText('upng');
    const workerUrl$1 = URL.createObjectURL(new Blob([pngWorkerFragment + pako + upng.replace('window.UPNG', 'self.UPNG').replace('window.pako', 'self.pako')], {
        type: 'text/javascript'
    }));
    const freeApngWorkers = [];
    function png(frames, convertMeta) {
        return new Promise((resolve, reject) => {
            logger.info('Start convert:', convertMeta.id);
            logger.time(convertMeta.id);
            let worker;
            if (freeApngWorkers.length) {
                worker = freeApngWorkers.shift();
                logger.info('Reuse apng workers.');
            }
            else {
                worker = new Worker(workerUrl$1);
            }
            convertMeta.abort = () => {
                logger.timeEnd(convertMeta.id);
                logger.warn('Convert stop manually. ' + convertMeta.id);
                reject(new CancelError());
                convertMeta.isAborted = true;
                worker.terminate();
            };
            worker.onmessage = function (e) {
                freeApngWorkers.push(worker);
                logger.timeEnd(convertMeta.id);
                if (!e.data) {
                    return reject(new TypeError('Failed to get png data. ' + convertMeta.id));
                }
                const pngBlob = new Blob([e.data], { type: 'image/png' });
                resolve(pngBlob);
            };
            const delay = convertMeta.source.delays;
            const cfg = { frames, delay };
            worker.postMessage(cfg);
        });
    }

    var webpWorkerFragment = "// Lossless encoding (0=lossy(default), 1=lossless).\n// quality: between 0 and 100. For lossy, 0 gives the smallest\n// size and 100 the largest. For lossless, this\n// parameter is the amount of effort put into the\n// compression: 0 is the fastest but gives larger\n// files compared to the slowest, but best, 100.\n// method: quality/speed trade-off (0=fast, 6=slower-better)\n\nlet webpApi = {};\nModule.onRuntimeInitialized = () => {\n  webpApi = {\n    init: Module.cwrap('init', '', ['number', 'number', 'number']),\n    createBuffer: Module.cwrap('createBuffer', 'number', ['number']),\n    addFrame: Module.cwrap('addFrame', 'number', ['number', 'number', 'number']),\n    generate: Module.cwrap('generate', 'number', []),\n    freeResult: Module.cwrap('freeResult', '', []),\n    getResultPointer: Module.cwrap('getResultPointer', 'number', []),\n    getResultSize: Module.cwrap('getResultSize', 'number', [])\n  };\n\n  postMessage('ok');\n};\n\nonmessage = async (evt) => {\n  const { frames, delays, lossless = 1, quality = 75, method = 4 } = evt.data;\n\n  webpApi.init(lossless, quality, method);\n\n  const bitmaps = await Promise.all(frames.map((blob) => createImageBitmap(blob)));\n  const width = bitmaps[0].width;\n  const height = bitmaps[0].height;\n  const canvas = new OffscreenCanvas(width, height);\n  const ctx = canvas.getContext('2d');\n\n  for (let i = 0; i < bitmaps.length; i++) {\n    ctx?.drawImage(bitmaps[i], 0, 0);\n    const webpBlob = await canvas.convertToBlob({ type: 'image/webp', quality: 1 });\n    const buffer = await webpBlob.arrayBuffer();\n    const u8a = new Uint8Array(buffer);\n    const pointer = webpApi.createBuffer(u8a.length);\n\n    Module.HEAPU8.set(u8a, pointer);\n    webpApi.addFrame(pointer, u8a.length, delays[i]);\n    postMessage(((i + 1) / bitmaps.length) * 100);\n  }\n\n  webpApi.generate();\n  const resultPointer = webpApi.getResultPointer();\n  const resultSize = webpApi.getResultSize();\n  const result = new Uint8Array(Module.HEAP8.buffer, resultPointer, resultSize);\n  postMessage(result);\n  webpApi.freeResult();\n};\n";

    const workerUrl = URL.createObjectURL(new Blob([workerChunk__default["default"] + webpWorkerFragment], { type: 'text/javascript' }));
    const freeWebpWorkers = [];
    function webp(frames, convertMeta) {
        return new Promise((resolve, reject) => {
            logger.time(convertMeta.id);
            let worker;
            if (freeWebpWorkers.length) {
                logger.info('Reuse webp workers.');
                worker = freeWebpWorkers.shift();
                resolve(worker);
            }
            else {
                worker = new Worker(workerUrl);
                worker.onmessage = (evt) => {
                    if (evt.data === 'ok') {
                        logger.info('Webp worker loaded.');
                        resolve(worker);
                    }
                    else {
                        reject(evt.data);
                    }
                };
            }
        }).then((worker) => {
            if (convertMeta.isAborted) {
                freeWebpWorkers.push(worker);
                logger.timeEnd(convertMeta.id);
                logger.warn('Convert stop manually.' + convertMeta.id);
                throw new CancelError();
            }
            return new Promise((resolve, reject) => {
                worker.onmessage = (evt) => {
                    if (convertMeta.isAborted) {
                        worker.terminate();
                        logger.timeEnd(convertMeta.id);
                        logger.warn('Convert stop manually.' + convertMeta.id);
                        reject(new CancelError());
                    }
                    else {
                        const data = evt.data;
                        if (typeof data !== 'object') {
                            convertMeta.onProgress?.(evt.data);
                        }
                        else {
                            logger.timeEnd(convertMeta.id);
                            freeWebpWorkers.push(worker);
                            resolve(new Blob([evt.data], { type: 'image/webp' }));
                        }
                    }
                };
                const delays = convertMeta.source.delays;
                worker.postMessage({ frames, delays });
            });
        });
    }

    function getDefaultExportFromCjs (x) {
    	return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
    }

    var WebMWriter$1 = {exports: {}};

    WebMWriter$1.exports;
    (function (module) {
    	(function() {
    	    function extend(base, top) {
    	        let
    	            target = {};
    	        [base, top].forEach(function(obj) {
    	            for (let prop in obj) {
    	                if (Object.prototype.hasOwnProperty.call(obj, prop)) {
    	                    target[prop] = obj[prop];
    	                }
    	            }
    	        });
    	        return target;
    	    }
    	    function decodeBase64WebPDataURL(url) {
    	        if (typeof url !== "string" || !url.match(/^data:image\/webp;base64,/i)) {
    	            throw new Error("Failed to decode WebP Base64 URL");
    	        }
    	        return window.atob(url.substring("data:image\/webp;base64,".length));
    	    }
    	    function renderAsWebP(canvas, quality) {
    	        let
    	            frame = typeof canvas === 'string' && /^data:image\/webp/.test(canvas)
    	                ? canvas
    	                : canvas.toDataURL('image/webp', quality);
    	        return decodeBase64WebPDataURL(frame);
    	    }
    	    function byteStringToUint32LE(string) {
    	        let
    	            a = string.charCodeAt(0),
    	            b = string.charCodeAt(1),
    	            c = string.charCodeAt(2),
    	            d = string.charCodeAt(3);
    	        return (a | (b << 8) | (c << 16) | (d << 24)) >>> 0;
    	    }
    	    function extractKeyframeFromWebP(webP) {
    	        let
    	            cursor = webP.indexOf('VP8', 12);
    	        if (cursor === -1) {
    	            throw new Error("Bad image format, does this browser support WebP?");
    	        }
    	        let
    	            hasAlpha = false;
    	        while (cursor < webP.length - 8) {
    	            let
    	                chunkLength, fourCC;
    	            fourCC = webP.substring(cursor, cursor + 4);
    	            cursor += 4;
    	            chunkLength = byteStringToUint32LE(webP.substring(cursor, cursor + 4));
    	            cursor += 4;
    	            switch (fourCC) {
    	                case "VP8 ":
    	                    return {
    	                        frame: webP.substring(cursor, cursor + chunkLength),
    	                        hasAlpha: hasAlpha
    	                    };
    	                case "ALPH":
    	                    hasAlpha = true;
    	                    break;
    	            }
    	            cursor += chunkLength;
    	            if ((chunkLength & 0x01) !== 0) {
    	                cursor++;
    	            }
    	        }
    	        throw new Error("Failed to find VP8 keyframe in WebP image, is this image mistakenly encoded in the Lossless WebP format?");
    	    }
    	    const
    	        EBML_SIZE_UNKNOWN = -1,
    	        EBML_SIZE_UNKNOWN_5_BYTES = -2;
    	    function EBMLFloat32(value) {
    	        this.value = value;
    	    }
    	    function EBMLFloat64(value) {
    	        this.value = value;
    	    }
    	    function writeEBML(buffer, bufferFileOffset, ebml) {
    	        if (Array.isArray(ebml)) {
    	            for (let i = 0; i < ebml.length; i++) {
    	                writeEBML(buffer, bufferFileOffset, ebml[i]);
    	            }
    	        } else if (typeof ebml === "string") {
    	            buffer.writeString(ebml);
    	        } else if (ebml instanceof Uint8Array) {
    	            buffer.writeBytes(ebml);
    	        } else if (ebml.id){
    	            ebml.offset = buffer.pos + bufferFileOffset;
    	            buffer.writeUnsignedIntBE(ebml.id);
    	            if (Array.isArray(ebml.data)) {
    	                let
    	                    sizePos, dataBegin, dataEnd;
    	                if (ebml.size === EBML_SIZE_UNKNOWN) {
    	                    buffer.writeByte(0xFF);
    	                } else if (ebml.size === EBML_SIZE_UNKNOWN_5_BYTES) {
    	                    sizePos = buffer.pos;
    	                    buffer.writeBytes([0x0F, 0xFF, 0xFF, 0xFF, 0xFF]);
    	                } else {
    	                    sizePos = buffer.pos;
    	                    buffer.writeBytes([0, 0, 0, 0]);
    	                }
    	                dataBegin = buffer.pos;
    	                ebml.dataOffset = dataBegin + bufferFileOffset;
    	                writeEBML(buffer, bufferFileOffset, ebml.data);
    	                if (ebml.size !== EBML_SIZE_UNKNOWN && ebml.size !== EBML_SIZE_UNKNOWN_5_BYTES) {
    	                    dataEnd = buffer.pos;
    	                    ebml.size = dataEnd - dataBegin;
    	                    buffer.seek(sizePos);
    	                    buffer.writeEBMLVarIntWidth(ebml.size, 4);
    	                    buffer.seek(dataEnd);
    	                }
    	            } else if (typeof ebml.data === "string") {
    	                buffer.writeEBMLVarInt(ebml.data.length);
    	                ebml.dataOffset = buffer.pos + bufferFileOffset;
    	                buffer.writeString(ebml.data);
    	            } else if (typeof ebml.data === "number") {
    	                if (!ebml.size) {
    	                    ebml.size = buffer.measureUnsignedInt(ebml.data);
    	                }
    	                buffer.writeEBMLVarInt(ebml.size);
    	                ebml.dataOffset = buffer.pos + bufferFileOffset;
    	                buffer.writeUnsignedIntBE(ebml.data, ebml.size);
    	            } else if (ebml.data instanceof EBMLFloat64) {
    	                buffer.writeEBMLVarInt(8);
    	                ebml.dataOffset = buffer.pos + bufferFileOffset;
    	                buffer.writeDoubleBE(ebml.data.value);
    	            } else if (ebml.data instanceof EBMLFloat32) {
    	                buffer.writeEBMLVarInt(4);
    	                ebml.dataOffset = buffer.pos + bufferFileOffset;
    	                buffer.writeFloatBE(ebml.data.value);
    	            } else if (ebml.data instanceof Uint8Array) {
    	                buffer.writeEBMLVarInt(ebml.data.byteLength);
    	                ebml.dataOffset = buffer.pos + bufferFileOffset;
    	                buffer.writeBytes(ebml.data);
    	            } else {
    	                throw new Error("Bad EBML datatype " + typeof ebml.data);
    	            }
    	        } else {
    	            throw new Error("Bad EBML datatype " + typeof ebml.data);
    	        }
    	    }
    	    let WebMWriter = function(ArrayBufferDataStream, BlobBuffer) {
    	        return function(options) {
    	            let
    	                MAX_CLUSTER_DURATION_MSEC = 5000,
    	                DEFAULT_TRACK_NUMBER = 1,
    	                writtenHeader = false,
    	                videoWidth = 0, videoHeight = 0,
    	                alphaBuffer = null,
    	                alphaBufferContext = null,
    	                alphaBufferData = null,
    	                clusterFrameBuffer = [],
    	                clusterStartTime = 0,
    	                clusterDuration = 0,
    	                optionDefaults = {
    	                    quality: 0.95,
    	                    transparent: false,
    	                    alphaQuality: undefined,
    	                    fileWriter: null,
    	                    fd: null,
    	                    frameDuration: null,
    	                    frameRate: null,
    	                },
    	                seekPoints = {
    	                    Cues: {id: new Uint8Array([0x1C, 0x53, 0xBB, 0x6B]), positionEBML: null},
    	                    SegmentInfo: {id: new Uint8Array([0x15, 0x49, 0xA9, 0x66]), positionEBML: null},
    	                    Tracks: {id: new Uint8Array([0x16, 0x54, 0xAE, 0x6B]), positionEBML: null},
    	                },
    	                ebmlSegment,
    	                segmentDuration = {
    	                    "id": 0x4489,
    	                    "data": new EBMLFloat64(0)
    	                },
    	                seekHead,
    	                cues = [],
    	                blobBuffer = new BlobBuffer(options.fileWriter || options.fd);
    	            function fileOffsetToSegmentRelative(fileOffset) {
    	                return fileOffset - ebmlSegment.dataOffset;
    	            }
    	            function convertAlphaToGrayscaleImage(source) {
    	                if (alphaBuffer === null || alphaBuffer.width !== source.width || alphaBuffer.height !== source.height) {
    	                    alphaBuffer = document.createElement("canvas");
    	                    alphaBuffer.width = source.width;
    	                    alphaBuffer.height = source.height;
    	                    alphaBufferContext = alphaBuffer.getContext("2d");
    	                    alphaBufferData = alphaBufferContext.createImageData(alphaBuffer.width, alphaBuffer.height);
    	                }
    	                let
    	                    sourceContext = source.getContext("2d"),
    	                    sourceData = sourceContext.getImageData(0, 0, source.width, source.height).data,
    	                    destData = alphaBufferData.data,
    	                    dstCursor = 0,
    	                    srcEnd = source.width * source.height * 4;
    	                for (let srcCursor = 3 ; srcCursor < srcEnd; srcCursor += 4) {
    	                    let
    	                        alpha = sourceData[srcCursor];
    	                    destData[dstCursor++] = alpha;
    	                    destData[dstCursor++] = alpha;
    	                    destData[dstCursor++] = alpha;
    	                    destData[dstCursor++] = 255;
    	                }
    	                alphaBufferContext.putImageData(alphaBufferData, 0, 0);
    	                return alphaBuffer;
    	            }
    	            function createSeekHead() {
    	                let
    	                    seekPositionEBMLTemplate = {
    	                        "id": 0x53AC,
    	                        "size": 5,
    	                        "data": 0
    	                    },
    	                    result = {
    	                        "id": 0x114D9B74,
    	                        "data": []
    	                    };
    	                for (let name in seekPoints) {
    	                    let
    	                        seekPoint = seekPoints[name];
    	                    seekPoint.positionEBML = Object.create(seekPositionEBMLTemplate);
    	                    result.data.push({
    	                         "id": 0x4DBB,
    	                         "data": [
    	                              {
    	                                  "id": 0x53AB,
    	                                  "data": seekPoint.id
    	                              },
    	                              seekPoint.positionEBML
    	                         ]
    	                    });
    	                }
    	                return result;
    	            }
    	            function writeHeader() {
    	                seekHead = createSeekHead();
    	                let
    	                    ebmlHeader = {
    	                        "id": 0x1a45dfa3,
    	                        "data": [
    	                            {
    	                                "id": 0x4286,
    	                                "data": 1
    	                            },
    	                            {
    	                                "id": 0x42f7,
    	                                "data": 1
    	                            },
    	                            {
    	                                "id": 0x42f2,
    	                                "data": 4
    	                            },
    	                            {
    	                                "id": 0x42f3,
    	                                "data": 8
    	                            },
    	                            {
    	                                "id": 0x4282,
    	                                "data": "webm"
    	                            },
    	                            {
    	                                "id": 0x4287,
    	                                "data": 2
    	                            },
    	                            {
    	                                "id": 0x4285,
    	                                "data": 2
    	                            }
    	                        ]
    	                    },
    	                    segmentInfo = {
    	                        "id": 0x1549a966,
    	                        "data": [
    	                            {
    	                                "id": 0x2ad7b1,
    	                                "data": 1e6
    	                            },
    	                            {
    	                                "id": 0x4d80,
    	                                "data": "webm-writer-js",
    	                            },
    	                            {
    	                                "id": 0x5741,
    	                                "data": "webm-writer-js"
    	                            },
    	                            segmentDuration
    	                        ]
    	                    },
    	                    videoProperties = [
    	                        {
    	                            "id": 0xb0,
    	                            "data": videoWidth
    	                        },
    	                        {
    	                            "id": 0xba,
    	                            "data": videoHeight
    	                        }
    	                    ];
    	                if (options.transparent) {
    	                    videoProperties.push(
    	                        {
    	                            "id": 0x53C0,
    	                            "data": 1
    	                        }
    	                    );
    	                }
    	                let
    	                    tracks = {
    	                        "id": 0x1654ae6b,
    	                        "data": [
    	                            {
    	                                "id": 0xae,
    	                                "data": [
    	                                    {
    	                                        "id": 0xd7,
    	                                        "data": DEFAULT_TRACK_NUMBER
    	                                    },
    	                                    {
    	                                        "id": 0x73c5,
    	                                        "data": DEFAULT_TRACK_NUMBER
    	                                    },
    	                                    {
    	                                        "id": 0x9c,
    	                                        "data": 0
    	                                    },
    	                                    {
    	                                        "id": 0x22b59c,
    	                                        "data": "und"
    	                                    },
    	                                    {
    	                                        "id": 0x86,
    	                                        "data": "V_VP8"
    	                                    },
    	                                    {
    	                                        "id": 0x258688,
    	                                        "data": "VP8"
    	                                    },
    	                                    {
    	                                        "id": 0x83,
    	                                        "data": 1
    	                                    },
    	                                    {
    	                                        "id": 0xe0,
    	                                        "data": videoProperties
    	                                    }
    	                                ]
    	                            }
    	                        ]
    	                    };
    	                ebmlSegment = {
    	                    "id": 0x18538067,
    	                    "size": EBML_SIZE_UNKNOWN_5_BYTES,
    	                    "data": [
    	                        seekHead,
    	                        segmentInfo,
    	                        tracks,
    	                    ]
    	                };
    	                let
    	                    bufferStream = new ArrayBufferDataStream(256);
    	                writeEBML(bufferStream, blobBuffer.pos, [ebmlHeader, ebmlSegment]);
    	                blobBuffer.write(bufferStream.getAsDataArray());
    	                seekPoints.SegmentInfo.positionEBML.data = fileOffsetToSegmentRelative(segmentInfo.offset);
    	                seekPoints.Tracks.positionEBML.data = fileOffsetToSegmentRelative(tracks.offset);
    		            writtenHeader = true;
    	            }
    	            function createBlockGroupForTransparentKeyframe(keyframe) {
    	                let
    	                    block, blockAdditions,
    	                    bufferStream = new ArrayBufferDataStream(1 + 2 + 1);
    	                if (!(keyframe.trackNumber > 0 && keyframe.trackNumber < 127)) {
    	                    throw new Error("TrackNumber must be > 0 and < 127");
    	                }
    	                bufferStream.writeEBMLVarInt(keyframe.trackNumber);
    	                bufferStream.writeU16BE(keyframe.timecode);
    	                bufferStream.writeByte(0);
    	                block = {
    	                    "id": 0xA1,
    	                    "data": [
    	                        bufferStream.getAsDataArray(),
    	                        keyframe.frame
    	                    ]
    	                };
    	                blockAdditions = {
    	                    "id": 0x75A1,
    	                    "data": [
    	                        {
    	                            "id": 0xA6,
    	                            "data": [
    	                                {
    	                                    "id": 0xEE,
    	                                    "data": 1
    	                                },
    	                                {
    	                                    "id": 0xA5,
    	                                    "data": keyframe.alpha
    	                                }
    	                            ]
    	                        }
    	                    ]
    	                };
    	                return {
    	                    "id": 0xA0,
    	                    "data": [
    	                        block,
    	                        blockAdditions
    	                    ]
    	                };
    	            }
    	            function createSimpleBlockForKeyframe(keyframe) {
    	                let
    	                    bufferStream = new ArrayBufferDataStream(1 + 2 + 1);
    	                if (!(keyframe.trackNumber > 0 && keyframe.trackNumber < 127)) {
    	                    throw new Error("TrackNumber must be > 0 and < 127");
    	                }
    	                bufferStream.writeEBMLVarInt(keyframe.trackNumber);
    	                bufferStream.writeU16BE(keyframe.timecode);
    	                bufferStream.writeByte(
    	                    1 << 7
    	                );
    	                return {
    	                    "id": 0xA3,
    	                    "data": [
    	                         bufferStream.getAsDataArray(),
    	                         keyframe.frame
    	                    ]
    	                };
    	            }
    	            function createContainerForKeyframe(keyframe) {
    	                if (keyframe.alpha) {
    	                    return createBlockGroupForTransparentKeyframe(keyframe);
    	                }
    	                return createSimpleBlockForKeyframe(keyframe);
    	            }
    	            function createCluster(cluster) {
    	                return {
    	                    "id": 0x1f43b675,
    	                    "data": [
    	                         {
    	                            "id": 0xe7,
    	                            "data": Math.round(cluster.timecode)
    	                         }
    	                    ]
    	                };
    	            }
    	            function addCuePoint(trackIndex, clusterTime, clusterFileOffset) {
    	                cues.push({
    	                    "id": 0xBB,
    	                    "data": [
    	                         {
    	                             "id": 0xB3,
    	                             "data": clusterTime
    	                         },
    	                         {
    	                             "id": 0xB7,
    	                             "data": [
    	                                  {
    	                                      "id": 0xF7,
    	                                      "data": trackIndex
    	                                  },
    	                                  {
    	                                      "id": 0xF1,
    	                                      "data": fileOffsetToSegmentRelative(clusterFileOffset)
    	                                  }
    	                             ]
    	                         }
    	                    ]
    	                });
    	            }
    	            function writeCues() {
    	                let
    	                    ebml = {
    	                        "id": 0x1C53BB6B,
    	                        "data": cues
    	                    },
    	                    cuesBuffer = new ArrayBufferDataStream(16 + cues.length * 32);
    	                writeEBML(cuesBuffer, blobBuffer.pos, ebml);
    	                blobBuffer.write(cuesBuffer.getAsDataArray());
    	                seekPoints.Cues.positionEBML.data = fileOffsetToSegmentRelative(ebml.offset);
    	            }
    	            function flushClusterFrameBuffer() {
    	                if (clusterFrameBuffer.length === 0) {
    	                    return;
    	                }
    	                let
    	                    rawImageSize = 0;
    	                for (let i = 0; i < clusterFrameBuffer.length; i++) {
    	                    rawImageSize += clusterFrameBuffer[i].frame.length + (clusterFrameBuffer[i].alpha ? clusterFrameBuffer[i].alpha.length : 0);
    	                }
    	                let
    	                    buffer = new ArrayBufferDataStream(rawImageSize + clusterFrameBuffer.length * 64),
    	                    cluster = createCluster({
    	                        timecode: Math.round(clusterStartTime),
    	                    });
    	                for (let i = 0; i < clusterFrameBuffer.length; i++) {
    	                    cluster.data.push(createContainerForKeyframe(clusterFrameBuffer[i]));
    	                }
    	                writeEBML(buffer, blobBuffer.pos, cluster);
    	                blobBuffer.write(buffer.getAsDataArray());
    	                addCuePoint(DEFAULT_TRACK_NUMBER, Math.round(clusterStartTime), cluster.offset);
    	                clusterFrameBuffer = [];
    	                clusterStartTime += clusterDuration;
    	                clusterDuration = 0;
    	            }
    	            function validateOptions() {
    	                if (!options.frameDuration) {
    	                    if (options.frameRate) {
    	                        options.frameDuration = 1000 / options.frameRate;
    	                    } else {
    	                        throw new Error("Missing required frameDuration or frameRate setting");
    	                    }
    	                }
    	                options.quality = Math.max(Math.min(options.quality, 0.99999), 0);
    	                if (options.alphaQuality === undefined) {
    	                    options.alphaQuality = options.quality;
    	                } else {
    	                    options.alphaQuality = Math.max(Math.min(options.alphaQuality, 0.99999), 0);
    	                }
    	            }
    	            function addFrameToCluster(frame) {
    	                frame.trackNumber = DEFAULT_TRACK_NUMBER;
    	                frame.timecode = Math.round(clusterDuration);
    	                clusterFrameBuffer.push(frame);
    	                clusterDuration += frame.duration;
    	                if (clusterDuration >= MAX_CLUSTER_DURATION_MSEC) {
    	                    flushClusterFrameBuffer();
    	                }
    	            }
    	            function rewriteSeekHead() {
    	                let
    	                    seekHeadBuffer = new ArrayBufferDataStream(seekHead.size),
    	                    oldPos = blobBuffer.pos;
    	                writeEBML(seekHeadBuffer, seekHead.dataOffset, seekHead.data);
    	                blobBuffer.seek(seekHead.dataOffset);
    	                blobBuffer.write(seekHeadBuffer.getAsDataArray());
    	                blobBuffer.seek(oldPos);
    	            }
    	            function rewriteDuration() {
    	                let
    	                    buffer = new ArrayBufferDataStream(8),
    	                    oldPos = blobBuffer.pos;
    	                buffer.writeDoubleBE(clusterStartTime);
    	                blobBuffer.seek(segmentDuration.dataOffset);
    	                blobBuffer.write(buffer.getAsDataArray());
    	                blobBuffer.seek(oldPos);
    	            }
    	            function rewriteSegmentLength() {
    	                let
    	                    buffer = new ArrayBufferDataStream(10),
    	                    oldPos = blobBuffer.pos;
    	                buffer.writeUnsignedIntBE(ebmlSegment.id);
    	                buffer.writeEBMLVarIntWidth(blobBuffer.pos - ebmlSegment.dataOffset, 5);
    	                blobBuffer.seek(ebmlSegment.offset);
    	                blobBuffer.write(buffer.getAsDataArray());
    	                blobBuffer.seek(oldPos);
    	            }
    	            this.addFrame = function(frame, alpha, overrideFrameDuration) {
    	                if (!writtenHeader) {
    	                    videoWidth = frame.width || 0;
    	                    videoHeight = frame.height || 0;
    	                    writeHeader();
    	                }
    	                let
    	                    keyframe = extractKeyframeFromWebP(renderAsWebP(frame, options.quality)),
    	                    frameDuration, frameAlpha = null;
    	                if (overrideFrameDuration) {
    	                    frameDuration = overrideFrameDuration;
    	                } else if (typeof alpha == "number") {
    	                    frameDuration = alpha;
    	                } else {
    	                    frameDuration = options.frameDuration;
    	                }
    	                if (options.transparent) {
    	                    if (alpha instanceof HTMLCanvasElement || typeof alpha === "string") {
    	                        frameAlpha = alpha;
    	                    } else if (keyframe.hasAlpha) {
    	                        frameAlpha = convertAlphaToGrayscaleImage(frame);
    	                    }
    	                }
    	                addFrameToCluster({
    	                    frame: keyframe.frame,
    	                    duration: frameDuration,
    	                    alpha: frameAlpha ? extractKeyframeFromWebP(renderAsWebP(frameAlpha, options.alphaQuality)).frame : null
    	                });
    	            };
    	            this.complete = function() {
    	            	if (!writtenHeader) {
    			            writeHeader();
    		            }
    	                flushClusterFrameBuffer();
    	                writeCues();
    	                rewriteSeekHead();
    	                rewriteDuration();
    	                rewriteSegmentLength();
    	                return blobBuffer.complete('video/webm');
    	            };
    	            this.getWrittenSize = function() {
    	                return blobBuffer.length;
    	            };
    	            options = extend(optionDefaults, options || {});
    	            validateOptions();
    	        };
    	    };
    	    {
    		    module.exports = WebMWriter;
    	    }
    	})();
    } (WebMWriter$1));
    var WebMWriterExports = WebMWriter$1.exports;
    getDefaultExportFromCjs(WebMWriterExports);

    var ArrayBufferDataStream = {exports: {}};

    ArrayBufferDataStream.exports;
    (function (module) {
    	(function(){
    	    let ArrayBufferDataStream = function(length) {
    	        this.data = new Uint8Array(length);
    	        this.pos = 0;
    	    };
    	    ArrayBufferDataStream.prototype.seek = function(toOffset) {
    	        this.pos = toOffset;
    	    };
    	    ArrayBufferDataStream.prototype.writeBytes = function(arr) {
    	        for (let i = 0; i < arr.length; i++) {
    	            this.data[this.pos++] = arr[i];
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.writeByte = function(b) {
    	        this.data[this.pos++] = b;
    	    };
    	    ArrayBufferDataStream.prototype.writeU8 = ArrayBufferDataStream.prototype.writeByte;
    	    ArrayBufferDataStream.prototype.writeU16BE = function(u) {
    	        this.data[this.pos++] = u >> 8;
    	        this.data[this.pos++] = u;
    	    };
    	    ArrayBufferDataStream.prototype.writeDoubleBE = function(d) {
    	        let
    	            bytes = new Uint8Array(new Float64Array([d]).buffer);
    	        for (let i = bytes.length - 1; i >= 0; i--) {
    	            this.writeByte(bytes[i]);
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.writeFloatBE = function(d) {
    	        let
    	            bytes = new Uint8Array(new Float32Array([d]).buffer);
    	        for (let i = bytes.length - 1; i >= 0; i--) {
    	            this.writeByte(bytes[i]);
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.writeString = function(s) {
    	        for (let i = 0; i < s.length; i++) {
    	            this.data[this.pos++] = s.charCodeAt(i);
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.writeEBMLVarIntWidth = function(i, width) {
    	        switch (width) {
    	            case 1:
    	                this.writeU8((1 << 7) | i);
    	            break;
    	            case 2:
    	                this.writeU8((1 << 6) | (i >> 8));
    	                this.writeU8(i);
    	            break;
    	            case 3:
    	                this.writeU8((1 << 5) | (i >> 16));
    	                this.writeU8(i >> 8);
    	                this.writeU8(i);
    	            break;
    	            case 4:
    	                this.writeU8((1 << 4) | (i >> 24));
    	                this.writeU8(i >> 16);
    	                this.writeU8(i >> 8);
    	                this.writeU8(i);
    	            break;
    	            case 5:
    	                this.writeU8((1 << 3) | ((i / 4294967296) & 0x7));
    	                this.writeU8(i >> 24);
    	                this.writeU8(i >> 16);
    	                this.writeU8(i >> 8);
    	                this.writeU8(i);
    	            break;
    	            default:
    	                throw new Error("Bad EBML VINT size " + width);
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.measureEBMLVarInt = function(val) {
    	        if (val < (1 << 7) - 1) {
    	            return 1;
    	        } else if (val < (1 << 14) - 1) {
    	            return 2;
    	        } else if (val < (1 << 21) - 1) {
    	            return 3;
    	        } else if (val < (1 << 28) - 1) {
    	            return 4;
    	        } else if (val < 34359738367) {
    	            return 5;
    	        } else {
    	            throw new Error("EBML VINT size not supported " + val);
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.writeEBMLVarInt = function(i) {
    	        this.writeEBMLVarIntWidth(i, this.measureEBMLVarInt(i));
    	    };
    	    ArrayBufferDataStream.prototype.writeUnsignedIntBE = function(u, width) {
    	        if (width === undefined) {
    	            width = this.measureUnsignedInt(u);
    	        }
    	        switch (width) {
    	            case 5:
    	                this.writeU8(Math.floor(u / 4294967296));
    	            case 4:
    	                this.writeU8(u >> 24);
    	            case 3:
    	                this.writeU8(u >> 16);
    	            case 2:
    	                this.writeU8(u >> 8);
    	            case 1:
    	                this.writeU8(u);
    	            break;
    	            default:
    	                throw new Error("Bad UINT size " + width);
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.measureUnsignedInt = function(val) {
    	        if (val < (1 << 8)) {
    	            return 1;
    	        } else if (val < (1 << 16)) {
    	            return 2;
    	        } else if (val < (1 << 24)) {
    	            return 3;
    	        } else if (val < 4294967296) {
    	            return 4;
    	        } else {
    	            return 5;
    	        }
    	    };
    	    ArrayBufferDataStream.prototype.getAsDataArray = function() {
    	        if (this.pos < this.data.byteLength) {
    	            return this.data.subarray(0, this.pos);
    	        } else if (this.pos == this.data.byteLength) {
    	            return this.data;
    	        } else {
    	            throw new Error("ArrayBufferDataStream's pos lies beyond end of buffer");
    	        }
    	    };
    		{
    			module.exports = ArrayBufferDataStream;
    		}
    	}());
    } (ArrayBufferDataStream));
    var ArrayBufferDataStreamExports = ArrayBufferDataStream.exports;
    getDefaultExportFromCjs(ArrayBufferDataStreamExports);

    var BlobBuffer = {exports: {}};

    BlobBuffer.exports;
    (function (module) {
    	(function() {
    		let BlobBuffer = function(fs) {
    			return function(destination) {
    				let
    					buffer = [],
    					writePromise = Promise.resolve(),
    					fileWriter = null,
    					fd = null;
    				if (destination && destination.constructor.name === "FileWriter") {
    					fileWriter = destination;
    				} else if (fs && destination) {
    					fd = destination;
    				}
    				this.pos = 0;
    				this.length = 0;
    				function readBlobAsBuffer(blob) {
    					return new Promise(function (resolve, reject) {
    						let
    							reader = new FileReader();
    						reader.addEventListener("loadend", function () {
    							resolve(reader.result);
    						});
    						reader.readAsArrayBuffer(blob);
    					});
    				}
    				function convertToUint8Array(thing) {
    					return new Promise(function (resolve, reject) {
    						if (thing instanceof Uint8Array) {
    							resolve(thing);
    						} else if (thing instanceof ArrayBuffer || ArrayBuffer.isView(thing)) {
    							resolve(new Uint8Array(thing));
    						} else if (thing instanceof Blob) {
    							resolve(readBlobAsBuffer(thing).then(function (buffer) {
    								return new Uint8Array(buffer);
    							}));
    						} else {
    							resolve(readBlobAsBuffer(new Blob([thing])).then(function (buffer) {
    								return new Uint8Array(buffer);
    							}));
    						}
    					});
    				}
    				function measureData(data) {
    					let
    						result = data.byteLength || data.length || data.size;
    					if (!Number.isInteger(result)) {
    						throw new Error("Failed to determine size of element");
    					}
    					return result;
    				}
    				this.seek = function (offset) {
    					if (offset < 0) {
    						throw new Error("Offset may not be negative");
    					}
    					if (isNaN(offset)) {
    						throw new Error("Offset may not be NaN");
    					}
    					if (offset > this.length) {
    						throw new Error("Seeking beyond the end of file is not allowed");
    					}
    					this.pos = offset;
    				};
    				this.write = function (data) {
    					let
    						newEntry = {
    							offset: this.pos,
    							data: data,
    							length: measureData(data)
    						},
    						isAppend = newEntry.offset >= this.length;
    					this.pos += newEntry.length;
    					this.length = Math.max(this.length, this.pos);
    					writePromise = writePromise.then(function () {
    						if (fd) {
    							return new Promise(function(resolve, reject) {
    								convertToUint8Array(newEntry.data).then(function(dataArray) {
    									let
    										totalWritten = 0,
    										buffer = Buffer.from(dataArray.buffer),
    										handleWriteComplete = function(err, written, buffer) {
    											totalWritten += written;
    											if (totalWritten >= buffer.length) {
    												resolve();
    											} else {
    												fs.write(fd, buffer, totalWritten, buffer.length - totalWritten, newEntry.offset + totalWritten, handleWriteComplete);
    											}
    										};
    									fs.write(fd, buffer, 0, buffer.length, newEntry.offset, handleWriteComplete);
    								});
    							});
    						} else if (fileWriter) {
    							return new Promise(function (resolve, reject) {
    								fileWriter.onwriteend = resolve;
    								fileWriter.seek(newEntry.offset);
    								fileWriter.write(new Blob([newEntry.data]));
    							});
    						} else if (!isAppend) {
    							for (let i = 0; i < buffer.length; i++) {
    								let
    									entry = buffer[i];
    								if (!(newEntry.offset + newEntry.length <= entry.offset || newEntry.offset >= entry.offset + entry.length)) {
    									if (newEntry.offset < entry.offset || newEntry.offset + newEntry.length > entry.offset + entry.length) {
    										throw new Error("Overwrite crosses blob boundaries");
    									}
    									if (newEntry.offset == entry.offset && newEntry.length == entry.length) {
    										entry.data = newEntry.data;
    										return;
    									} else {
    										return convertToUint8Array(entry.data)
    											.then(function (entryArray) {
    												entry.data = entryArray;
    												return convertToUint8Array(newEntry.data);
    											}).then(function (newEntryArray) {
    												newEntry.data = newEntryArray;
    												entry.data.set(newEntry.data, newEntry.offset - entry.offset);
    											});
    									}
    								}
    							}
    						}
    						buffer.push(newEntry);
    					});
    				};
    				this.complete = function (mimeType) {
    					if (fd || fileWriter) {
    						writePromise = writePromise.then(function () {
    							return null;
    						});
    					} else {
    						writePromise = writePromise.then(function () {
    							let
    								result = [];
    							for (let i = 0; i < buffer.length; i++) {
    								result.push(buffer[i].data);
    							}
    							return new Blob(result, {type: mimeType});
    						});
    					}
    					return writePromise;
    				};
    			};
    		};
    		{
    			module.exports = BlobBuffer;
    		}
    	})();
    } (BlobBuffer));
    var BlobBufferExports = BlobBuffer.exports;
    getDefaultExportFromCjs(BlobBufferExports);

    var browser = WebMWriterExports(ArrayBufferDataStreamExports, BlobBufferExports(null));
    var WebMWriter = getDefaultExportFromCjs(browser);

    function readBlobAsDataUrl(blob) {
        return new Promise((resolve) => {
            const reader = new FileReader();
            reader.onload = () => {
                resolve(reader.result);
            };
            reader.readAsDataURL(blob);
        });
    }
    function webm(frames, convertMeta) {
        return Promise.all(frames.map((frame) => createImageBitmap(frame)))
            .then((bitmaps) => {
            if (convertMeta.isAborted)
                throw new CancelError();
            const width = bitmaps[0].width;
            const height = bitmaps[0].height;
            const canvas = new OffscreenCanvas(width, height);
            const ctx = canvas.getContext('2d');
            const dataUrls = [];
            for (let i = 0; i < frames.length; i++) {
                ctx.drawImage(bitmaps[i], 0, 0);
                const url = canvas.convertToBlob({ type: 'image/webp', quality: 0.95 }).then(readBlobAsDataUrl);
                dataUrls.push(url);
            }
            return Promise.all(dataUrls);
        })
            .then((dataUrls) => {
            if (convertMeta.isAborted)
                throw new CancelError();
            const videoWriter = new WebMWriter({
                quality: 0.95,
                frameRate: 30,
                transparent: false
            });
            const delays = convertMeta.source.delays;
            for (let i = 0; i < dataUrls.length; i++) {
                videoWriter.addFrame(dataUrls[i], delays[i]);
            }
            return videoWriter.complete();
        })
            .then((blob) => {
            if (convertMeta.isAborted)
                throw new CancelError();
            return blob;
        });
    }

    const adapter = {
        gif,
        png,
        webp,
        webm
    };
    const convertAdapter = {
        getAdapter(format) {
            return adapter[format];
        }
    };

    function createConverter() {
        const MAX_CONVERT = 2;
        const framesData = {};
        let isStop = false;
        let queue = [];
        let active = [];
        const doConvert = (convertMeta) => {
            const { id, format, source, resolve, reject } = convertMeta;
            active.push(convertMeta);
            convertMeta.onProgress?.(0);
            delete framesData[id];
            const adapter = convertAdapter.getAdapter(format);
            adapter(source.data, convertMeta)
                .then(resolve, reject)
                .finally(() => {
                active.splice(active.indexOf(convertMeta), 1);
                if (queue.length)
                    doConvert(queue.shift());
            });
        };
        return {
            del: (taskIds) => {
                if (typeof taskIds === 'string')
                    taskIds = [taskIds];
                if (!taskIds.length)
                    return;
                logger.info('Converter del, active:', active.map((meta) => meta.id), 'queue:', queue.map((meta) => meta.id));
                isStop = true;
                taskIds.forEach((taskId) => {
                    if (taskId in framesData)
                        delete framesData[taskId];
                });
                active = active.filter((convertMeta) => {
                    if (taskIds.includes(convertMeta.id)) {
                        convertMeta.abort();
                    }
                    else {
                        return true;
                    }
                });
                queue = queue.filter((convertMeta) => !taskIds.includes(convertMeta.id));
                isStop = false;
                while (active.length < MAX_CONVERT && queue.length) {
                    doConvert(queue.shift());
                }
            },
            addFrame(taskId, data, delay, order) {
                if (!(taskId in framesData)) {
                    framesData[taskId] = {
                        id: taskId,
                        data: [],
                        delays: []
                    };
                }
                if (order === undefined) {
                    framesData[taskId]['data'].push(data);
                    framesData[taskId]['delays'].push(delay);
                }
                else {
                    framesData[taskId]['data'][order] = data;
                    framesData[taskId]['delays'][order] = delay;
                }
            },
            framesCount(taskId) {
                return taskId in framesData ? framesData[taskId]['delays'].filter((delay) => delay !== undefined).length : 0;
            },
            convert(taskId, format, onProgress) {
                return new Promise((resolve, reject) => {
                    const meta = {
                        id: taskId,
                        format,
                        source: framesData[taskId],
                        isAborted: false,
                        onProgress,
                        resolve,
                        reject,
                        abort() {
                            this.isAborted = true;
                        }
                    };
                    logger.info('Converter add', taskId);
                    queue.push(meta);
                    while (active.length < MAX_CONVERT && queue.length && !isStop) {
                        doConvert(queue.shift());
                    }
                });
            }
        };
    }
    const converter = createConverter();

    const pixivHooks = {
        download: {
            singleArtworkProgressFactory(btn, pageCount) {
                if (!btn || !pageCount)
                    return;
                return function onSingleArtworkProgress(progress) {
                    if (pageCount === 1) {
                        btn.setProgress(progress);
                    }
                };
            },
            mulityArtworksProgressFactory(btn, pageCount) {
                if (!btn || !pageCount)
                    return;
                let pageComplete = 0;
                return function onMulityArtworksProgress() {
                    if (pageCount < 2)
                        return;
                    const progress = Math.floor((++pageComplete / pageCount) * 100);
                    btn.setProgress(progress);
                };
            }
        },
        bundle: {
            async beforeFileSave(imgBlob, config) {
                const { taskId, source } = config;
                compressor.add(taskId, source.filename, imgBlob);
                if (compressor.fileCount(taskId) === source.pageCount) {
                    const zipData = await compressor.bundle(taskId);
                    compressor.remove(taskId);
                    return zipData;
                }
            },
            onError(err, config) {
                compressor.remove(config.taskId);
            },
            onAbort(config) {
                compressor.remove(config.taskId);
            }
        },
        convert: {
            convertProgressFactory(btn) {
                return function onConvertProgress(progress) {
                    if (progress > 0) {
                        btn.setProgress(progress, false);
                    }
                    else {
                        btn.setStatus("loading" );
                    }
                };
            },
            beforeFileSaveFactory(btn) {
                const onProgress = btn ? this.convertProgressFactory(btn) : undefined;
                return async function beforeFileSave(imgBlob, config) {
                    const { taskId, source } = config;
                    if (source.illustType === IllustType.ugoira) {
                        converter.addFrame(taskId, imgBlob, source.ugoiraMeta.frames[source.order]['delay'], source.order);
                        if (converter.framesCount(taskId) === source.pageCount) {
                            return await converter.convert(taskId, source.extendName, onProgress);
                        }
                    }
                };
            },
            onError(err, config) {
                converter.del(config.taskId);
            },
            onAbort(config) {
                converter.del(config.taskId);
            }
        }
    };
    const downloaderHooks = {
        getHooks(meta, downloadType, button) {
            switch (downloadType) {
                case 'download':
                    return {
                        onProgress: pixivHooks.download.singleArtworkProgressFactory(button, meta.pageCount),
                        onFileSaved: pixivHooks.download.mulityArtworksProgressFactory(button, meta.pageCount)
                    };
                case 'bundle':
                    return {
                        onXhrLoaded: pixivHooks.download.mulityArtworksProgressFactory(button, meta.pageCount),
                        beforeFileSave: pixivHooks.bundle.beforeFileSave,
                        onError: pixivHooks.bundle.onError
                    };
                case 'convert':
                    return {
                        onXhrLoaded: pixivHooks.download.mulityArtworksProgressFactory(button, meta.pageCount),
                        beforeFileSave: pixivHooks.convert.beforeFileSaveFactory(button),
                        onError: pixivHooks.convert.onError
                    };
            }
        }
    };
    class PixivDownloadConfig extends DownloadConfigBuilder {
        meta;
        downloadAll = true;
        headers = {
            referer: 'https://www.pixiv.net'
        };
        timeout = 60000;
        getImgSrc = () => '';
        constructor(meta) {
            super(meta);
            this.meta = meta;
            this.getImgSrc =
                this.meta.illustType === IllustType.ugoira
                    ? (page) => this.meta.src.replace('ugoira0', 'ugoira' + page)
                    : (page) => this.meta.src.replace('_p0', '_p' + page);
        }
        getConvertFormat() {
            return config.get('ugoiraFormat');
        }
        needConvert() {
            return this.meta.illustType === IllustType.ugoira && config.get('ugoiraFormat') !== 'zip';
        }
        needBundle() {
            const { pageCount, illustType } = this.meta;
            return (this.downloadAll &&
                ((illustType === IllustType.ugoira && this.getConvertFormat() === 'zip') ||
                    (pageCount > 1 &&
                        ((illustType === IllustType.manga && config.get('bundleManga')) ||
                            (illustType === IllustType.illusts && config.get('bundleIllusts'))))));
        }
        useTranslatedTags() {
            return config.get('tagLang') !== 'ja';
        }
        useDirectSave() {
            return (!!this.getFolderPattern() &&
                !this.needBundle() &&
                !this.needConvert() &&
                (!env.isBlobDlAvaliable() || (env.isViolentmonkey() && !this.isFsaEnable())));
        }
        supportSubpath() {
            return ((this.isBrowserApi() && ((!this.needConvert() && !this.needBundle()) || env.isBlobDlAvaliable())) ||
                this.isFsaEnable());
        }
        buildPattern(pattern, page) {
            const { id, userId, artist, title, tags, tagsTranslated, createDate, pageCount } = this.meta;
            const currPage = page === undefined ? pageCount : page;
            const useTags = this.useTranslatedTags() ? tagsTranslated : tags;
            const fArtist = this.normalizeString(artist);
            const fTitle = this.normalizeString(title);
            const fTags = this.normalizeString(useTags.join('_'));
            const replaceDate = (match, p1) => {
                const format = p1 || 'YYYY-MM-DD';
                return dayjs__default["default"](createDate).format(format);
            };
            return pattern
                .replaceAll(/\{date\((.*?)\)\}|\{date\}/g, replaceDate)
                .replaceAll('{artist}', fArtist)
                .replaceAll('{artistID}', userId)
                .replaceAll('{title}', fTitle)
                .replaceAll('{tags}', fTags)
                .replaceAll('{page}', String(currPage))
                .replaceAll('{id}', id);
        }
        getDownloadConfig(btn) {
            const { illustType, src, id, pageCount, extendName } = this.meta;
            const pageAttr = btn?.getAttribute('page');
            const downloadPage = pageAttr ? Number(pageAttr) : undefined;
            if (downloadPage && (downloadPage > pageCount - 1 || downloadPage < 0))
                throw new Error('Invalid downloadPage.');
            if (downloadPage !== undefined)
                this.downloadAll = false;
            const taskId = id + '_' + Math.random().toString(36).slice(2);
            const headers = this.headers;
            const directSave = this.useDirectSave();
            const supportSubPath = this.supportSubpath();
            const downloadConfigs = [];
            if ((pageCount === 1 || downloadPage !== undefined) && illustType !== IllustType.ugoira) {
                const imgSrc = downloadPage ? this.getImgSrc(downloadPage) : src;
                const pathPattern = supportSubPath ? this.getFullpathPattern() : this.getFilenamePattern() + '.' + extendName;
                const path = this.buildPattern(pathPattern, downloadPage ? downloadPage : 0);
                const filename = path.slice(path.lastIndexOf('/') + 1);
                const hooks = downloaderHooks.getHooks({ ...this.meta, pageCount: 1 }, 'download', btn);
                const source = {
                    ...this.meta,
                    pageCount: 1,
                    filename,
                    order: downloadPage ?? 0
                };
                const downloadConfig = {
                    taskId,
                    src: imgSrc,
                    path,
                    source,
                    directSave,
                    headers,
                    timeout: this.timeout,
                    ...hooks
                };
                downloadConfigs.push(downloadConfig);
            }
            else {
                const pathPatternNoExt = supportSubPath ? this.getFullpathPattern().slice(0, -4) : this.getFilenamePattern();
                if (this.needBundle()) {
                    const pathPattern = pathPatternNoExt + '.zip';
                    const filenamePattern = this.getFilenamePattern().includes('{page}')
                        ? this.getFilenamePattern() + '.' + extendName
                        : this.getFilenamePattern() + '_{page}' + '.' + extendName;
                    let path;
                    let imgCount;
                    if (illustType === IllustType.ugoira) {
                        path = this.buildPattern(pathPattern, 0);
                        imgCount = this.meta.ugoiraMeta.frames.length;
                    }
                    else {
                        path = this.buildPattern(pathPattern);
                        imgCount = pageCount;
                    }
                    const hooks = downloaderHooks.getHooks({ ...this.meta, pageCount: imgCount }, 'bundle', btn);
                    for (let page = 0; page < imgCount; page++) {
                        const filename = this.buildPattern(filenamePattern, page);
                        const imgSrc = this.getImgSrc(page);
                        const source = {
                            ...this.meta,
                            pageCount: imgCount,
                            extendName: 'zip',
                            filename,
                            order: page
                        };
                        const downloadConfig = {
                            taskId,
                            src: imgSrc,
                            path,
                            source,
                            headers,
                            timeout: this.timeout,
                            ...hooks
                        };
                        downloadConfigs.push(downloadConfig);
                    }
                }
                else if (this.needConvert()) {
                    const ext = this.getConvertFormat();
                    const pathPattern = pathPatternNoExt + '.' + ext;
                    const path = this.buildPattern(pathPattern, 0);
                    const filename = path.slice(path.lastIndexOf('/') + 1);
                    const imgCount = this.meta.ugoiraMeta.frames.length;
                    const hooks = downloaderHooks.getHooks({ ...this.meta, pageCount: imgCount }, 'convert', btn);
                    for (let page = 0; page < imgCount; page++) {
                        const imgSrc = this.getImgSrc(page);
                        const source = {
                            ...this.meta,
                            pageCount: imgCount,
                            extendName: ext,
                            filename,
                            order: page
                        };
                        const downloadConfig = {
                            taskId,
                            src: imgSrc,
                            path,
                            source,
                            headers,
                            timeout: this.timeout,
                            ...hooks
                        };
                        downloadConfigs.push(downloadConfig);
                    }
                }
                else {
                    const pathPattern = pathPatternNoExt + '.' + extendName;
                    const hooks = downloaderHooks.getHooks(this.meta, 'download', btn);
                    for (let page = 0; page < pageCount; page++) {
                        const path = this.buildPattern(pathPattern, page);
                        const filename = path.slice(path.lastIndexOf('/') + 1);
                        const imgSrc = this.getImgSrc(page);
                        const source = {
                            ...this.meta,
                            filename,
                            order: page
                        };
                        const downloadConfig = {
                            taskId,
                            src: imgSrc,
                            path,
                            source,
                            directSave,
                            headers,
                            timeout: this.timeout,
                            ...hooks
                        };
                        downloadConfigs.push(downloadConfig);
                    }
                }
            }
            !this.downloadAll && (this.downloadAll = true);
            return downloadConfigs;
        }
    }

    async function downloadArtwork(btn) {
        downloader.dirHandleCheck();
        const id = btn.getAttribute('pdl-id');
        const pixivMeta = await pixivParser.parse(id);
        const { bookmarkData, token, tags, artist, userId, title } = pixivMeta;
        if (!bookmarkData) {
            addBookmark(btn, id, token, tags);
        }
        const downloadConfigs = new PixivDownloadConfig(pixivMeta).getDownloadConfig(btn);
        await downloader.download(downloadConfigs);
        const historyData = {
            pid: Number(id),
            user: artist,
            userId: Number(userId),
            title,
            tags
        };
        historyDb.add(historyData);
    }

    function createThumbnailBtn(nodes) {
        let isSelfBookmark = false;
        const inBookmarkPage = regexp.bookmarkPage.exec(location.pathname);
        inBookmarkPage && inBookmarkPage[1] === getSelfId() && (isSelfBookmark = true);
        nodes.forEach((e) => {
            let illustId;
            let type;
            if ((e.childElementCount !== 0 ||
                e.className.includes('_history-item') ||
                e.className.includes('_history-related-item')) &&
                !e.querySelector('pdl-button') &&
                (illustId = getIllustId(e))) {
                if (isSelfBookmark) {
                    type = "pixiv-my-bookmark" ;
                }
                else if (e.className.includes('_history-related-item')) {
                    e.style.position = 'relative';
                    type = "pixiv-history" ;
                }
                else if (e.className.includes('_history-item')) {
                    type = "pixiv-history" ;
                }
                const btn = new ThumbnailButton({
                    id: illustId,
                    type,
                    onClick: downloadArtwork
                });
                e.appendChild(btn);
            }
        });
    }

    function fixPixivPreviewer(nodes) {
        const isPpSearchPage = regexp.searchPage.test(location.pathname);
        if (!isPpSearchPage)
            return;
        nodes.forEach((node) => {
            const pdlEle = node.querySelector('.pdl-btn');
            if (!pdlEle)
                return false;
            pdlEle.remove();
        });
    }

    function createPdlBtn(option) {
        const { attrs, classList, textContent, downloadArtwork } = option;
        const ele = document.createElement('button');
        textContent && (ele.textContent = textContent);
        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 (downloadArtwork) {
            ele.addEventListener('click', (evt) => {
                evt.preventDefault();
                evt.stopPropagation();
                const btn = evt.currentTarget;
                if (!btn.classList.contains('pdl-progress')) {
                    btn.classList.add('pdl-progress');
                    const setProgress = (progress, updateProgressbar = true) => {
                        if (progress !== null) {
                            progress = Math.floor(progress);
                            btn.textContent = String(progress);
                            updateProgressbar && btn.style.setProperty('--pdl-progress', progress + '%');
                        }
                        else {
                            btn.textContent = '';
                            updateProgressbar && btn.style.removeProperty('--pdl-progress');
                        }
                    };
                    downloadArtwork(btn, setProgress)
                        .then(() => {
                        btn.classList.remove('pdl-error');
                        btn.classList.add('pdl-complete');
                    })
                        .catch((err) => {
                        if (err)
                            logger.error(err);
                        btn.classList.remove('pdl-complete');
                        btn.classList.add('pdl-error');
                    })
                        .finally(() => {
                        btn.innerHTML = '';
                        btn.style.removeProperty('--pdl-progress');
                        btn.classList.remove('pdl-progress');
                    });
                }
            });
        }
        return ele;
    }

    const dlBarRef = {
        filter: {
            filterExcludeDownloaded: undefined,
            filterIllusts: undefined,
            filterManga: undefined,
            filterUgoira: undefined
        },
        statusBar: undefined,
        abortBtn: undefined
    };
    function updateStatus(str) {
        dlBarRef.statusBar && (dlBarRef.statusBar.textContent = str);
    }
    function createFilterEl(id, filterType, text) {
        const checkbox = document.createElement('input');
        const label = document.createElement('label');
        checkbox.id = id;
        checkbox.type = 'checkbox';
        checkbox.classList.add('pdl-checkbox');
        checkbox.setAttribute('category', String(filterType));
        checkbox.checked = config.get(filterType);
        label.setAttribute('for', id);
        label.setAttribute('category', String(filterType));
        label.textContent = text;
        checkbox.addEventListener('change', (evt) => {
            const checkbox = evt.currentTarget;
            const category = checkbox.getAttribute('category');
            config.set(category, checkbox.checked);
        });
        dlBarRef.filter[filterType] = checkbox;
        const wrap = document.createElement('div');
        wrap.classList.add('pdl-filter');
        wrap.appendChild(checkbox);
        wrap.appendChild(label);
        return wrap;
    }
    function createFilter() {
        const wrapper = document.createElement('div');
        wrapper.classList.add('pdl-filter-wrap');
        wrapper.appendChild(createFilterEl('pdl-filter-exclude_downloaded', 'filterExcludeDownloaded', t('checkbox.filter_exclude_downloaded')));
        wrapper.appendChild(createFilterEl('pdl-filter-illusts', 'filterIllusts', t('checkbox.filter_illusts')));
        wrapper.appendChild(createFilterEl('pdl-filter-manga', 'filterManga', t('checkbox.filter_manga')));
        wrapper.appendChild(createFilterEl('pdl-filter-ugoira', 'filterUgoira', t('checkbox.filter_ugoira')));
        return wrapper;
    }
    function createExcludeDownloadedFilter() {
        const wrapper = document.createElement('div');
        wrapper.classList.add('pdl-filter-wrap');
        wrapper.appendChild(createFilterEl('pdl-filter-exclude_downloaded', 'filterExcludeDownloaded', t('checkbox.filter_exclude_downloaded')));
        return wrapper;
    }
    function createDownloadBar(userId) {
        const nav = document.querySelector('nav');
        if (!nav || nav.previousElementSibling)
            return;
        const dlBtn = nav.querySelector('.pdl-btn-all');
        if (dlBtn) {
            if (dlBtn.getAttribute('pdl-userid') === userId)
                return;
            removeDownloadBar();
        }
        const doesRequestPageLoaded = ["a[href$='illustrations']", "a[href$='manga']", "a[href*='bookmarks']"].some((selector) => !!nav.querySelector(selector));
        if (!doesRequestPageLoaded)
            return;
        const dlBar = document.createElement('div');
        dlBar.classList.add('pdl-dlbar');
        const statusBar = document.createElement('div');
        statusBar.classList.add('pdl-dlbar-status_bar');
        dlBarRef.statusBar = dlBar.appendChild(statusBar);
        const baseClasses = nav.querySelector('a:not([aria-current])').classList;
        dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
            attrs: { 'pdl-userId': userId },
            classList: [...baseClasses, 'pdl-stop', 'pdl-hide'],
            textContent: t('button.download_stop')
        }));
        if (userId !== getSelfId()) {
            const hasWorks = ["a[href$='illustrations']", "a[href$='manga']"].some((selector) => !!nav.querySelector(selector));
            if (hasWorks) {
                const el = createPdlBtn({
                    attrs: { 'pdl-userid': userId },
                    classList: [...baseClasses, 'pdl-btn-all'],
                    textContent: t('button.download_works')
                });
                el.addEventListener('click', downloadWorks);
                dlBar.appendChild(el);
            }
            if (nav.querySelector("a[href*='bookmarks']")) {
                const el = createPdlBtn({
                    attrs: { 'pdl-userid': userId, category: 'bookmarks' },
                    classList: [...baseClasses, 'pdl-btn-all'],
                    textContent: t('button.download_bookmarks')
                });
                el.addEventListener('click', downloadBookmarksOrTags);
                dlBar.appendChild(el);
            }
        }
        else {
            if (nav.querySelector("a[href*='bookmarks']")) {
                dlBar.appendChild(createPdlBtn({
                    attrs: { 'pdl-userid': userId, category: 'bookmarks', rest: 'all' },
                    classList: [...baseClasses, 'pdl-btn-all'],
                    textContent: t('button.download_bookmarks')
                }));
                dlBar.appendChild(createPdlBtn({
                    attrs: {
                        'pdl-userid': userId,
                        category: 'bookmarks',
                        rest: 'show'
                    },
                    classList: [...baseClasses, 'pdl-btn-all'],
                    textContent: t('button.download_bookmarks_public')
                }));
                dlBar.appendChild(createPdlBtn({
                    attrs: {
                        'pdl-userid': userId,
                        category: 'bookmarks',
                        rest: 'hide'
                    },
                    classList: [...baseClasses, 'pdl-btn-all'],
                    textContent: t('button.download_bookmarks_private')
                }));
                dlBar.querySelectorAll('.pdl-btn-all').forEach((node) => {
                    node.addEventListener('click', downloadBookmarksOrTags);
                });
            }
        }
        const filter = createFilter();
        nav.parentElement.insertBefore(filter, nav);
        nav.appendChild(dlBar);
    }
    function removeDownloadBar() {
        const dlBarWrap = document.querySelector('.pdl-dlbar');
        if (dlBarWrap) {
            dlBarWrap.remove();
            document.querySelector('.pdl-filter-wrap')?.remove();
        }
    }
    function updateFollowLatestDownloadBarBtnText(prevDlBtn, prevDlAllBtn) {
        if (location.pathname.includes('r18') && prevDlBtn.textContent !== t('button.download_r18_one_page')) {
            prevDlBtn.textContent = t('button.download_r18_one_page');
            prevDlAllBtn.textContent = t('button.download_r18');
        }
        else if (!location.pathname.includes('r18') && prevDlBtn.textContent !== t('button.download_all_one_page')) {
            prevDlBtn.textContent = t('button.download_all_one_page');
            prevDlAllBtn.textContent = t('button.download_all');
        }
    }
    function createFollowLatestDownloadBar() {
        const prevDlBtn = document.querySelector('.pdl-btn-all');
        if (prevDlBtn) {
            const prevDlAllBtn = document.querySelector('.pdl-dl-all');
            updateFollowLatestDownloadBarBtnText(prevDlBtn, prevDlAllBtn);
            return;
        }
        const nav = document.querySelector('nav');
        if (!nav || nav.parentElement.childElementCount === 1)
            return;
        const navBar = nav.parentElement;
        const modeSwitch = nav.nextElementSibling;
        const filter = createFilter();
        navBar.parentElement.insertBefore(filter, navBar);
        const dlBar = document.createElement('div');
        dlBar.classList.add('pdl-dlbar');
        dlBar.classList.add('pdl-dlbar-follow_latest');
        const statusBar = document.createElement('div');
        statusBar.classList.add('pdl-dlbar-status_bar');
        dlBarRef.statusBar = dlBar.appendChild(statusBar);
        const baseClasses = nav.querySelector('a:not([aria-current])').classList;
        dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
            attrs: { 'pdl-userid': '' },
            classList: [...baseClasses, 'pdl-stop', 'pdl-hide'],
            textContent: t('button.download_stop')
        }));
        const dlBtn = createPdlBtn({
            attrs: { 'pdl-userid': '' },
            classList: [...baseClasses, 'pdl-btn-all'],
            textContent: t('button.download_works')
        });
        dlBtn.addEventListener('click', downloadFollowLatest);
        dlBar.appendChild(dlBtn);
        const dlAllBtn = createPdlBtn({
            attrs: { 'pdl-userid': '' },
            classList: [...baseClasses, 'pdl-btn-all', 'pdl-dl-all'],
            textContent: t('button.download_works')
        });
        dlAllBtn.addEventListener('click', downloadFollowLatest);
        dlBar.appendChild(dlAllBtn);
        navBar.insertBefore(dlBar, modeSwitch);
    }
    function createSearchDownloadbar() {
        if (document.querySelector('.pdl-dlbar'))
            return;
        const sections = document.querySelectorAll('section');
        const worksSection = sections[sections.length - 1];
        const styleRefEle = document.querySelector('nav a:not([aria-current])');
        if (!worksSection || !styleRefEle)
            return;
        const dlBarContainer = worksSection.firstElementChild.firstElementChild;
        const dlBar = document.createElement('div');
        dlBar.classList.add('pdl-dlbar');
        dlBar.classList.add('pdl-dlbar-search');
        const statusBar = document.createElement('div');
        statusBar.classList.add('pdl-dlbar-status_bar');
        dlBarRef.statusBar = dlBar.appendChild(statusBar);
        const baseClasses = styleRefEle.classList;
        dlBarRef.abortBtn = dlBar.appendChild(createPdlBtn({
            attrs: { 'pdl-userid': '' },
            classList: [...baseClasses, 'pdl-stop', 'pdl-hide'],
            textContent: t('button.download_stop')
        }));
        const dlBtn = createPdlBtn({
            attrs: { 'pdl-userid': '' },
            classList: [...baseClasses, 'pdl-btn-all'],
            textContent: t('button.download_all_one_page')
        });
        dlBtn.addEventListener('click', downloadSearchResult);
        const filter = createExcludeDownloadedFilter();
        dlBarContainer.parentElement.insertBefore(filter, dlBarContainer);
        dlBar.appendChild(dlBtn);
        dlBarContainer.appendChild(dlBar);
    }
    function changeDlbarDisplay() {
        document.querySelectorAll('.pdl-dlbar .pdl-btn-all').forEach((ele) => {
            ele.classList.toggle('pdl-hide');
        });
        document.querySelector('.pdl-dlbar .pdl-stop')?.classList.toggle('pdl-hide');
        document.querySelectorAll('.pdl-tag').forEach((ele) => {
            ele.classList.toggle('pdl-tag-hide');
        });
        document.querySelector('.pdl-filter-wrap')?.classList.toggle('pdl-unavailable');
    }

    async function downloadByIds(total, idsGenerators, signal, onProgress) {
        signal.throwIfAborted();
        const failed = [];
        const unavaliable = [];
        const invalid = [];
        const tasks = [];
        let completed = 0;
        let tooManyRequests = false;
        let wakeTooManyRequest;
        let wakeInterval;
        let resolve;
        let reject;
        const done = new Promise((r, j) => {
            resolve = r;
            reject = j;
        });
        signal.addEventListener('abort', () => {
            if (tasks.length) {
                downloader.abort(tasks);
                tasks.length = 0;
            }
            wakeTooManyRequest?.();
            wakeInterval?.();
            reject(signal.aborted ? signal.reason : 'Unexpected generator error');
        }, { once: true });
        const afterEach = (illustId) => {
            const avaliable = total - failed.length - unavaliable.length - invalid.length;
            onProgress({
                illustId,
                avaliable,
                completed
            });
            if (completed === avaliable) {
                resolve({ failed, unavaliable });
            }
        };
        onProgress('Downloading...');
        try {
            for (const idsGenerator of idsGenerators) {
                for await (const ids of idsGenerator) {
                    logger.info('Got ids:', ids);
                    signal.throwIfAborted();
                    if (ids.unavaliable.length) {
                        unavaliable.push(...ids.unavaliable);
                    }
                    if (ids.invalid.length) {
                        invalid.push(...ids.invalid);
                    }
                    if (typeof ids.total === 'number' && !Number.isNaN(ids.total)) {
                        total = ids.total;
                    }
                    if (ids.avaliable.length) {
                        for (const id of ids.avaliable) {
                            signal.throwIfAborted();
                            if (tooManyRequests) {
                                onProgress('Too many requests, wait 30s');
                                const { wake, sleep } = wakeableSleep(30000);
                                wakeTooManyRequest = wake;
                                await sleep;
                                signal.throwIfAborted();
                                tooManyRequests = false;
                                onProgress('Downloading...');
                            }
                            let historyData;
                            pixivParser
                                .parse(id)
                                .then((pixivMeta) => {
                                const { id, tags, artist, userId, title } = pixivMeta;
                                historyData = {
                                    pid: Number(id),
                                    user: artist,
                                    userId: Number(userId),
                                    title,
                                    tags
                                };
                                const downloadConfigs = new PixivDownloadConfig(pixivMeta).getDownloadConfig();
                                tasks.push(downloadConfigs[0].taskId);
                                return downloader.download(downloadConfigs);
                            })
                                .then((taskId) => {
                                historyDb.add(historyData);
                                if (!signal.aborted) {
                                    tasks.splice(tasks.indexOf(taskId[0]), 1);
                                    completed++;
                                    afterEach(id);
                                }
                            }, (reason) => {
                                if (!signal.aborted) {
                                    reason && logger.error(reason);
                                    if (reason instanceof RequestError && reason.response.status === 429) {
                                        tooManyRequests = true;
                                    }
                                    if (reason instanceof JsonDataError) {
                                        unavaliable.push(id);
                                    }
                                    else {
                                        failed.push(id);
                                    }
                                    afterEach(id);
                                }
                            });
                            const { wake, sleep } = wakeableSleep(1000);
                            wakeInterval = wake;
                            await sleep;
                        }
                    }
                    else {
                        afterEach('no avaliable id');
                    }
                }
            }
        }
        catch (error) {
            if (!signal.aborted) {
                done.catch((reason) => {
                    logger.info('catch unexpected abort: ', reason);
                });
                signal.dispatchEvent(new Event('abort'));
                throw error;
            }
        }
        return done;
    }

    function onProgressCB(progressData) {
        if (typeof progressData === 'string') {
            updateStatus(progressData);
        }
        else {
            logger.info('Update progress by', progressData.illustId, ', completed: ', progressData.completed);
            updateStatus(`Downloading: ${progressData.completed} / ${progressData.avaliable}`);
        }
    }
    async function useDownloadBar(chunksGenerators) {
        if (!dlBarRef.abortBtn)
            return;
        let total = 0;
        let failedResult;
        const idsGenerators = [];
        !Array.isArray(chunksGenerators) && (chunksGenerators = [chunksGenerators]);
        isDownloading = true;
        changeDlbarDisplay();
        try {
            await Promise.all(chunksGenerators).then((gens) => {
                gens.forEach((val) => {
                    total += val.total;
                    idsGenerators.push(val.generator);
                });
            });
        }
        catch (error) {
            logger.error(error);
            updateStatus('Network error, see console');
            changeDlbarDisplay();
            isDownloading = false;
            return;
        }
        if (total === 0) {
            updateStatus('No works');
        }
        else {
            try {
                logger.info('Total works:', total);
                const controller = new AbortController();
                const signal = controller.signal;
                !signal.throwIfAborted &&
                    (signal.throwIfAborted = function () {
                        if (this.aborted) {
                            throw this.reason;
                        }
                    });
                if (!('reason' in signal)) {
                    const abort = controller.abort;
                    controller.abort = function (reason) {
                        this.signal.reason = reason ? reason : new DOMException('signal is aborted without reason');
                        abort.apply(this);
                    };
                }
                dlBarRef.abortBtn?.addEventListener('click', () => {
                    controller.abort();
                }, { once: true });
                const { failed, unavaliable } = await downloadByIds(total, idsGenerators, signal, onProgressCB);
                if (failed.length || unavaliable.length) {
                    updateStatus(`Failed: ${failed.length + unavaliable.length}. See console.`);
                    console.log('[Pixiv Downloader] Failed: ', failed.join(', '));
                    console.log('[Pixiv Downloader] Unavaliable: ', unavaliable.join(', '));
                    if (failed.length)
                        failedResult = failed;
                }
                else {
                    console.log('[Pixiv Downloader] Download complete');
                    updateStatus('Complete');
                }
            }
            catch (error) {
                if (error instanceof DOMException) {
                    updateStatus('Stop');
                }
                else {
                    updateStatus('Error, see console');
                    logger.error(error);
                }
            }
        }
        changeDlbarDisplay();
        isDownloading = false;
        return failedResult;
    }
    function getFilterOption() {
        return {
            filterExcludeDownloaded: config.get('filterExcludeDownloaded'),
            filterIllusts: config.get('filterIllusts'),
            filterManga: config.get('filterManga'),
            filterUgoira: config.get('filterUgoira')
        };
    }
    function downloadAndRetry(chunksGenerators) {
        useDownloadBar(chunksGenerators).then((failed) => {
            if (failed instanceof Array && failed.length) {
                const gen = async function* () {
                    yield {
                        avaliable: failed,
                        unavaliable: [],
                        invalid: []
                    };
                };
                console.log('[Pixiv Downloader] Retry...');
                useDownloadBar({ total: failed.length, generator: gen() });
            }
        });
    }
    let isDownloading = false;
    function downloadWorks(evt) {
        evt.preventDefault();
        evt.stopPropagation();
        if (isDownloading)
            return;
        const btn = evt.target;
        const userId = btn.getAttribute('pdl-userid');
        const filterOption = getFilterOption();
        downloader.dirHandleCheck();
        const ids = pixivParser.getAllWorksGenerator(userId, filterOption);
        downloadAndRetry(ids);
    }
    async function downloadBookmarksOrTags(evt) {
        evt.preventDefault();
        evt.stopPropagation();
        if (isDownloading)
            return;
        const btn = evt.target;
        const userId = btn.getAttribute('pdl-userid');
        const category = btn.getAttribute('category');
        const tag = btn.getAttribute('tag') || '';
        const rest = (btn.getAttribute('rest') || 'show');
        downloader.dirHandleCheck();
        const filterOption = getFilterOption();
        let idsGenerators;
        if (rest === 'all') {
            const idsShowPromise = pixivParser.getChunksGenerator(userId, 'bookmarks', '', 'show', filterOption);
            const idsHidePromise = pixivParser.getChunksGenerator(userId, 'bookmarks', '', 'hide', filterOption);
            idsGenerators = [idsShowPromise, idsHidePromise];
        }
        else {
            idsGenerators = pixivParser.getChunksGenerator(userId, category, tag, rest, filterOption);
        }
        downloadAndRetry(idsGenerators);
    }
    function downloadFollowLatest(evt) {
        evt.preventDefault();
        evt.stopPropagation();
        if (isDownloading)
            return;
        const btn = evt.target;
        const mode = location.pathname.includes('r18') ? 'r18' : 'all';
        const filterOption = getFilterOption();
        let idsGenerators;
        if (btn.classList.contains('pdl-dl-all')) {
            idsGenerators = pixivParser.getFollowLatestGenerator(filterOption, mode);
        }
        else {
            const params = new URLSearchParams(location.search);
            const page = Number(params.get('p')) || 1;
            idsGenerators = pixivParser.getFollowLatestGenerator(filterOption, mode, page);
        }
        downloadAndRetry(idsGenerators);
    }
    async function downloadSearchResult(evt) {
        evt.preventDefault();
        evt.stopPropagation();
        if (isDownloading)
            return;
        const pdlNodes = document.querySelectorAll('section ul li button.pdl-btn-sub');
        if (!pdlNodes.length)
            return;
        let ids = Array.prototype.map.call(pdlNodes, (node) => node.getAttribute('pdl-id'));
        if (getFilterOption().filterExcludeDownloaded) {
            const filteredIds = [];
            for (const id of ids) {
                const isDownloaded = await historyDb.has(id);
                !isDownloaded && filteredIds.push(id);
            }
            ids = filteredIds;
        }
        const idsGenerators = {
            total: ids.length,
            generator: (async function* () {
                yield {
                    avaliable: ids,
                    unavaliable: [],
                    invalid: []
                };
            })()
        };
        downloadAndRetry(idsGenerators);
    }

    function createTagsBtn(userId, category) {
        const tagsEles = Array.from(document.querySelectorAll('a[status]')).map((el) => el.parentElement);
        if (!tagsEles.length)
            return;
        let cate;
        if (category === 'illustrations' || category === 'artworks') {
            cate = 'illusts';
        }
        else {
            cate = category;
        }
        let rest = 'show';
        if (userId === getSelfId() && category === 'bookmarks' && location.search.includes('rest=hide'))
            rest = 'hide';
        tagsEles.forEach((ele) => {
            const tagBtn = ele.querySelector('.pdl-btn');
            if (tagBtn) {
                const btnRest = tagBtn.getAttribute('rest');
                if (rest !== btnRest)
                    tagBtn.setAttribute('rest', rest);
                return;
            }
            let tag;
            const tagLink = ele.querySelector('a');
            if (!tagLink)
                return;
            if (tagLink.getAttribute('status') !== 'active') {
                if (rest === 'hide') {
                    tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1, tagLink.href.lastIndexOf('?'));
                }
                else {
                    tag = tagLink.href.slice(tagLink.href.lastIndexOf('/') + 1);
                }
            }
            else {
                const tagTextEles = ele.querySelectorAll('div[title]');
                if (!tagTextEles.length)
                    return logger.info('No Tags Element found.');
                tag = tagTextEles[tagTextEles.length - 1].getAttribute('title').slice(1);
            }
            const attrs = {
                attrs: { 'pdl-userId': userId, category: cate, tag, rest },
                classList: ['pdl-btn', 'pdl-tag']
            };
            if (isDownloading)
                attrs.classList.push('pdl-tag-hide');
            const dlBtn = createPdlBtn(attrs);
            if (!(tagLink.href.includes('bookmarks') && tagLink.getAttribute('status') !== 'active')) {
                dlBtn.style.backgroundColor = tagLink.getAttribute('color') + '80';
            }
            dlBtn.addEventListener('click', downloadBookmarksOrTags);
            ele.appendChild(dlBtn);
        });
        let modalTagsEles;
        let modal;
        if (category === 'bookmarks') {
            modal = document.querySelector('div[role="presentation"]');
            if (!modal)
                return;
            modalTagsEles = modal.querySelectorAll('a');
        }
        else {
            const charcoalTokens = document.querySelectorAll('.charcoal-token');
            modal = charcoalTokens[charcoalTokens.length - 1];
            if (!modal)
                return;
            modalTagsEles = modal.querySelectorAll('a');
        }
        if (!regexp.userPageTags.exec(modalTagsEles[0]?.href))
            return;
        modalTagsEles.forEach((ele) => {
            if (ele.querySelector('.pdl-btn'))
                return;
            let tag;
            if (rest === 'hide') {
                tag = ele.href.slice(ele.href.lastIndexOf('/') + 1, ele.href.lastIndexOf('?'));
            }
            else {
                tag = ele.href.slice(ele.href.lastIndexOf('/') + 1);
            }
            const attrs = {
                attrs: { 'pdl-userId': userId, category: cate, tag, rest },
                classList: ['pdl-btn', 'pdl-modal-tag']
            };
            if (isDownloading)
                attrs.classList.push('pdl-tag-hide');
            const dlBtn = createPdlBtn(attrs);
            dlBtn.addEventListener('click', (evt) => {
                modal.querySelector('svg').parentElement.click();
                downloadBookmarksOrTags(evt);
            });
            ele.appendChild(dlBtn);
        });
    }

    function createToolbarBtn(id) {
        const toolbar = document.querySelector('main section section');
        if (!toolbar || toolbar.querySelector('pdl-button'))
            return;
        const btn = new ThumbnailButton({
            id,
            type: "pixiv-toolbar" ,
            onClick: downloadArtwork
        });
        const pdlBtnWrap = toolbar.lastElementChild.cloneNode();
        pdlBtnWrap.appendChild(btn);
        toolbar.appendChild(pdlBtnWrap);
    }

    function createWorkScrollBtn(id) {
        const works = document.querySelectorAll('figure a.gtm-expand-full-size-illust');
        if (!works.length)
            return;
        const containers = Array.from(works).map((node) => node.parentElement.parentElement);
        if (containers[0].querySelector('pdl-button'))
            return;
        containers.forEach((node, idx) => {
            const wrapper = document.createElement('div');
            wrapper.classList.add('pdl-wrap-artworks');
            wrapper.appendChild(new ThumbnailButton({
                id,
                page: idx,
                type: "gallery" ,
                onClick: downloadArtwork
            }));
            node.appendChild(wrapper);
        });
    }

    const createPresentationBtn = (() => {
        let observer, btn;
        function cb(mutationList) {
            const newImg = mutationList[1]['addedNodes'][0];
            const [pageNum] = regexp.originSrcPageNum.exec(newImg.src) ?? [];
            if (!pageNum)
                throw new Error('[Error]Invalid Image Element.');
            btn?.setAttribute('page', String(pageNum));
        }
        return (id) => {
            const containers = document.querySelector("body > [role='presentation'] > div");
            if (!containers) {
                if (observer) {
                    observer.disconnect();
                    observer = null;
                    btn = null;
                }
                return;
            }
            if (containers.querySelector('.pdl-btn'))
                return;
            const img = containers.querySelector('div > img');
            if (!img)
                return;
            const isOriginImg = regexp.originSrcPageNum.exec(img.src);
            if (!isOriginImg)
                return;
            const [pageNum] = isOriginImg;
            btn = new ThumbnailButton({
                id,
                type: "pixiv-presentation" ,
                page: Number(pageNum),
                onClick: downloadArtwork
            });
            containers.appendChild(btn);
            if (!img.parentElement)
                return;
            observer = new MutationObserver(cb);
            observer.observe(img.parentElement, { childList: true, subtree: true });
        };
    })();

    function createPreviewModalBtn() {
        const illustModalBtn = document.querySelector('.gtm-manga-viewer-preview-modal-open:not(.pdl-listened)');
        const mangaModalBtn = document.querySelector('.gtm-manga-viewer-open-preview:not(.pdl-listened)');
        const mangaViewerModalBtn = document.querySelectorAll('.gtm-manga-viewer-close-icon:not(.pdl-listened)')?.[1];
        if (!illustModalBtn && !mangaModalBtn && !mangaViewerModalBtn)
            return;
        [illustModalBtn, mangaModalBtn, mangaViewerModalBtn].forEach((node) => {
            if (node) {
                node.classList.add('pdl-listened');
                node.addEventListener('click', handleModalClick);
            }
        });
    }
    function handleModalClick() {
        const timer = setInterval(() => {
            logger.info('Start to find modal.');
            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';
                node.appendChild(new ThumbnailButton({
                    id,
                    page: idx,
                    onClick: downloadArtwork
                }));
            });
        }, 300);
    }

    function createMangaViewerBtn(id) {
        const mangaViewerBackBtn = document.querySelector('.gtm-manga-viewer-close-icon');
        if (!mangaViewerBackBtn)
            return;
        const container = mangaViewerBackBtn.parentElement;
        if (container.querySelector('pdl-button'))
            return;
        container.appendChild(new ThumbnailButton({
            id,
            type: "pixiv-manga-viewer" ,
            onClick: downloadArtwork
        }));
    }

    function pageActions() {
        const pathname = location.pathname;
        let param;
        switch (true) {
            case !!(param = regexp.artworksPage.exec(pathname)): {
                const id = param[1];
                createToolbarBtn(id);
                createWorkScrollBtn(id);
                createPresentationBtn(id);
                createPreviewModalBtn();
                createMangaViewerBtn(id);
                break;
            }
            case !!(param = regexp.userPage.exec(pathname)): {
                const id = param[1] || param[2];
                createDownloadBar(id);
                const matchTag = regexp.userPageTags.exec(pathname);
                if (matchTag) {
                    createTagsBtn(id, matchTag[1]);
                }
                break;
            }
            case regexp.followLatest.test(pathname): {
                createFollowLatestDownloadBar();
                break;
            }
            case regexp.searchPage.test(pathname): {
                createSearchDownloadbar();
                break;
            }
            case regexp.historyPage.test(pathname): {
                createThumbnailBtn(document.querySelectorAll('span[style]._history-item'));
                break;
            }
            default:
                removeDownloadBar();
                break;
        }
    }
    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 !== 'PDL-BUTTON' && node.tagName !== 'IMG') {
                    addedNodes.push(node);
                }
            });
        });
        if (!addedNodes.length)
            return;
        if (firstRun) {
            createThumbnailBtn(document.querySelectorAll('a'));
            firstRun = false;
        }
        else {
            fixPixivPreviewer(addedNodes);
            const thumbnails = addedNodes.reduce((prev, current) => {
                return prev.concat(current instanceof HTMLAnchorElement ? [current] : Array.from(current.querySelectorAll('a')));
            }, []);
            createThumbnailBtn(thumbnails);
        }
        pageActions();
    }

    var css = ".pdl-dlbar{display:flex;flex-grow:1}.pdl-dlbar.pdl-dlbar-follow_latest{padding:0 8px}.pdl-dlbar .pdl-dlbar-status_bar{color:#858585;cursor:default;flex-grow:1;font-size:16px;font-weight:700;height:46px;line-height:46px;padding-right:8px;text-align:right;white-space:nowrap}.pdl-dlbar .pdl-btn-all,.pdl-dlbar .pdl-stop{background-color:transparent;border:none;padding:0 8px}.pdl-dlbar .pdl-btn-all:hover,.pdl-dlbar .pdl-stop:hover{color:var(--pdl-text1)}.pdl-dlbar .pdl-btn-all:before,.pdl-dlbar .pdl-stop:before{background:no-repeat 50%/85%;content:\"\";height:24px;transition:background-image .2s ease 0s;width:24px}.pdl-dlbar .pdl-btn-all:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}.pdl-dlbar .pdl-btn-all:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}.pdl-dlbar .pdl-stop:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}.pdl-dlbar .pdl-stop:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-btn-all:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-btn-all:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-stop:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-stop:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root[data-theme=default] .pdl-dlbar .pdl-btn-all:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root[data-theme=default] .pdl-dlbar .pdl-btn-all:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root[data-theme=default] .pdl-dlbar .pdl-stop:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root[data-theme=default] .pdl-dlbar .pdl-stop:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}@media (prefers-color-scheme:light){:root .pdl-dlbar .pdl-btn-all:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-btn-all:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-stop:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-stop:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}}:root[data-theme=dark] .pdl-dlbar .pdl-btn-all:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root[data-theme=dark] .pdl-dlbar .pdl-btn-all:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root[data-theme=dark] .pdl-dlbar .pdl-stop:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root[data-theme=dark] .pdl-dlbar .pdl-stop:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}@media (prefers-color-scheme:dark){:root .pdl-dlbar .pdl-btn-all:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-btn-all:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-stop:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}:root .pdl-dlbar .pdl-stop:hover:before{background-image:url(\"data:image/svg+xml;charset=utf-8,%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/svg%3E\")}}.pdl-filter-wrap{color:#858585;display:flex;font-size:14px;font-weight:700;gap:12px;justify-content:flex-end;line-height:14px;margin:4px 0;transition:color .2s ease 0s}.pdl-filter-wrap .pdl-filter:hover{color:var(--pdl-text1)}.pdl-filter-wrap .pdl-filter label{cursor:pointer;padding-left:8px}";

    class Pixiv extends SiteInject {
        init() {
            super.init();
            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'));
                        }
                    }
                }
            });
        }
        addStyle() {
            super.addStyle();
            GM_addStyle(css$8);
            GM_addStyle(css);
        }
    }

    function getSiteInjector(host) {
        const sitesAdapter = {
            'danbooru.donmai.us': Danbooru,
            'www.pixiv.net': Pixiv,
            'rule34.xxx': Rule34
        };
        if (host in sitesAdapter) {
            return sitesAdapter[host];
        }
    }
    const siteInject = getSiteInjector(location.host);
    siteInject && new siteInject();

})(Dexie, dayjs, JSZip, GIF, workerChunk);