Pixiv Downloader

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.

As of 2024-06-29. See the latest version.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

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

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

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

// ==UserScript==
// @name         Pixiv Downloader
// @namespace    https://greasyfork.org/zh-CN/scripts/432150
// @version      0.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/[email protected]/dist/jszip.min.js
// @require      https://unpkg.com/[email protected]/dist/gif.js
// @require      https://unpkg.com/[email protected]/dayjs.min.js
// @require      https://unpkg.com/[email protected]/dist/dexie.min.js
// @require      https://greasyfork.org/scripts/455256-toanimatedwebp/code/toAnimatedWebp.js?version=1120088
// @resource pako https://unpkg.com/[email protected]/dist/pako.min.js
// @resource upng https://unpkg.com/[email protected]/UPNG.js
// @resource gifWorker https://unpkg.com/[email protected]/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);