您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
一键下载各页面原图。批量下载画师作品,按作品标签下载。转换动图格式:Gif | Apng | Webp | Webm | MP4。自定义图片文件名,保存路径。保留 / 导出下载历史。Pixiv | Danbooru | ATFbooru | Yande.re | Konachan | Sakugabooru | Rule34 | Rule34paheal | Rule34us | Rule34vault | Gelbooru | Safebooru | E621 | E926 | E6ai | Nijie.info
// ==UserScript== // @name Pixiv Downloader // @namespace https://greasyfork.org/zh-CN/scripts/432150 // @version 1.10.0 // @author ruaruarua // @description 一键下载各页面原图。批量下载画师作品,按作品标签下载。转换动图格式:Gif | Apng | Webp | Webm | MP4。自定义图片文件名,保存路径。保留 / 导出下载历史。Pixiv | Danbooru | ATFbooru | Yande.re | Konachan | Sakugabooru | Rule34 | Rule34paheal | Rule34us | Rule34vault | Gelbooru | Safebooru | E621 | E926 | E6ai | Nijie.info // @description:zh-TW 一鍵下載各頁面原圖。批次下載畫師作品,按作品標籤下載。轉換動圖格式:Gif | Apng | Webp | Webm | MP4。自定義圖片檔名,儲存路徑。保留 / 匯出下載歷史。Pixiv | Danbooru | ATFbooru | Yande.re | Konachan | Sakugabooru | Rule34 | Rule34paheal | Rule34us | Rule34vault | Gelbooru | Safebooru | E621 | E926 | E6ai | Nijie.info // @description:en Download artworks with one click. Batch download artworks or download by tags. Convert ugoira formats: Gif | Apng | Webp | Webm | MP4. Customize image file name, save path. Save / export download history. Pixiv | Danbooru | ATFbooru | Yande.re | Konachan | Sakugabooru | Rule34 | Rule34paheal | Rule34us | Rule34vault | Gelbooru | Safebooru | E621 | E926 | E6ai | Nijie.info // @license MIT // @icon https://www.pixiv.net/favicon.ico // @supportURL https://github.com/drunkg00se/Pixiv-Downloader/issues // @match https://www.pixiv.net/* // @match https://rule34.xxx/* // @match https://danbooru.donmai.us/* // @match https://yande.re/* // @match https://booru.allthefallen.moe/* // @match https://konachan.com/* // @match https://konachan.net/* // @match https://www.sakugabooru.com/* // @match https://safebooru.org/* // @match https://gelbooru.com/* // @match https://e621.net/* // @match https://e926.net/* // @match https://e6ai.net/* // @match https://nijie.info/* // @match https://rule34vault.com/* // @match https://rule34.paheal.net/* // @match https://rule34.us/* // @require https://unpkg.com/[email protected]/dist/dexie.min.js // @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://cdn.jsdelivr.net/npm/[email protected]/build/mp4-muxer.min.js // @require https://cdn.jsdelivr.net/npm/[email protected]/build/webm-muxer.min.js // @resource ../wasm/toWebpWorker?raw https://update.greasyfork.org/scripts/500281/1409041/libwebp_wasm.js // @resource gif.js/dist/gif.worker?raw https://unpkg.com/[email protected]/dist/gif.worker.js // @resource pako/dist/pako.js?raw https://unpkg.com/[email protected]/dist/pako.min.js // @resource upng-js?raw https://unpkg.com/[email protected]/UPNG.js // @connect i.pximg.net // @connect source.pixiv.net // @connect rule34.xxx // @connect donmai.us // @connect yande.re // @connect allthefallen.moe // @connect konachan.com // @connect konachan.net // @connect sakugabooru.com // @connect safebooru.org // @connect gelbooru.com // @connect e621.net // @connect e926.net // @connect e6ai.net // @connect nijie.net // @connect rule34vault.com // @connect r34xyz.b-cdn.net // @connect r34i.paheal-cdn.net // @connect rule34.us // @grant GM_download // @grant GM_getResourceText // @grant GM_info // @grant GM_registerMenuCommand // @grant GM_xmlhttpRequest // @grant unsafeWindow // @noframes // ==/UserScript== (t=>{const r=new CSSStyleSheet;r.replaceSync(t),window._pdlShadowStyle=r})(` .anim-indeterminate.svelte-12wvf64{transform-origin:0% 50%;animation:svelte-12wvf64-anim-indeterminate 2s infinite linear}@keyframes svelte-12wvf64-anim-indeterminate{0%{transform:translate(0) scaleX(0)}40%{transform:translate(0) scaleX(.4)}to{transform:translate(100%) scaleX(.5)}}*,:before,:after{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }*,:before,:after{box-sizing:border-box;border-width:0;border-style:solid;border-color:#e5e7eb}:before,:after{--tw-content: ""}:host [data-theme=skeleton],:host{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji",Segoe UI Symbol,"Noto Color Emoji";font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}:host [data-theme=skeleton]{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h3,h4{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,samp{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}dl,dd,h1,h3,h4,hr,figure,p{margin:0}fieldset{margin:0;padding:0}ol,ul,menu{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}button,[role=button]{cursor:pointer}:disabled{cursor:default}img,svg,video,canvas,audio,iframe,embed,object{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]:where(:not([hidden=until-found])){display:none}[type=text],input:where(:not([type])),[type=email],[type=url],[type=password],[type=number],[type=date],[type=datetime-local],[type=month],[type=search],[type=tel],[type=time],[type=week],[multiple],textarea,select{-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:#fff;border-color:#6b7280;border-width:1px;border-radius:0;padding:8px 12px;font-size:16px;line-height:24px;--tw-shadow: 0 0 #0000}[type=text]:focus,input:where(:not([type])):focus,[type=email]:focus,[type=url]:focus,[type=password]:focus,[type=number]:focus,[type=date]:focus,[type=datetime-local]:focus,[type=month]:focus,[type=search]:focus,[type=tel]:focus,[type=time]:focus,[type=week]:focus,[multiple]:focus,textarea:focus,select:focus{outline:2px solid transparent;outline-offset:2px;--tw-ring-inset: var(--tw-empty, );--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: #2563eb;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow);border-color:#2563eb}input::-moz-placeholder,textarea::-moz-placeholder{color:#6b7280;opacity:1}input::placeholder,textarea::placeholder{color:#6b7280;opacity:1}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-date-and-time-value{min-height:1.5em;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit,::-webkit-datetime-edit-year-field,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute-field,::-webkit-datetime-edit-second-field,::-webkit-datetime-edit-millisecond-field,::-webkit-datetime-edit-meridiem-field{padding-top:0;padding-bottom:0}select{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 20 20'%3e%3cpath stroke='%236b7280' stroke-linecap='round' stroke-linejoin='round' stroke-width='1.5' d='M6 8l4 4 4-4'/%3e%3c/svg%3e");background-position:right 8px center;background-repeat:no-repeat;background-size:1.5em 1.5em;padding-right:40px;-webkit-print-color-adjust:exact;print-color-adjust:exact}[multiple],[size]:where(select:not([size="1"])){background-image:initial;background-position:initial;background-repeat:unset;background-size:initial;padding-right:12px;-webkit-print-color-adjust:unset;print-color-adjust:unset}[type=checkbox],[type=radio]{-webkit-appearance:none;-moz-appearance:none;appearance:none;padding:0;-webkit-print-color-adjust:exact;print-color-adjust:exact;display:inline-block;vertical-align:middle;background-origin:border-box;-webkit-user-select:none;-moz-user-select:none;user-select:none;flex-shrink:0;height:16px;width:16px;color:#2563eb;background-color:#fff;border-color:#6b7280;border-width:1px;--tw-shadow: 0 0 #0000}[type=checkbox]{border-radius:0}[type=radio]{border-radius:100%}[type=checkbox]:focus,[type=radio]:focus{outline:2px solid transparent;outline-offset:2px;--tw-ring-inset: var(--tw-empty, );--tw-ring-offset-width: 2px;--tw-ring-offset-color: #fff;--tw-ring-color: #2563eb;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}[type=checkbox]:checked,[type=radio]:checked{border-color:transparent;background-color:currentColor;background-size:100% 100%;background-position:center;background-repeat:no-repeat}[type=checkbox]:checked{background-image:url("data:image/svg+xml,%3csvg viewBox='0 0 16 16' fill='white' xmlns='http://www.w3.org/2000/svg'%3e%3cpath d='M12.207 4.793a1 1 0 010 1.414l-5 5a1 1 0 01-1.414 0l-2-2a1 1 0 011.414-1.414L6.5 9.086l4.293-4.293a1 1 0 011.414 0z'/%3e%3c/svg%3e")}@media (forced-colors: active){[type=checkbox]:checked{-webkit-appearance:auto;-moz-appearance:auto;appearance:auto}}[type=radio]:checked{background-image:url("data:image/svg+xml,%3csvg viewBox='0 0 16 16' fill='white' xmlns='http://www.w3.org/2000/svg'%3e%3ccircle cx='8' cy='8' r='3'/%3e%3c/svg%3e")}@media (forced-colors: active){[type=radio]:checked{-webkit-appearance:auto;-moz-appearance:auto;appearance:auto}}[type=checkbox]:checked:hover,[type=checkbox]:checked:focus,[type=radio]:checked:hover,[type=radio]:checked:focus{border-color:transparent;background-color:currentColor}[type=checkbox]:indeterminate{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 16 16'%3e%3cpath stroke='white' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M4 8h8'/%3e%3c/svg%3e");border-color:transparent;background-color:currentColor;background-size:100% 100%;background-position:center;background-repeat:no-repeat}@media (forced-colors: active){[type=checkbox]:indeterminate{-webkit-appearance:auto;-moz-appearance:auto;appearance:auto}}[type=checkbox]:indeterminate:hover,[type=checkbox]:indeterminate:focus{border-color:transparent;background-color:currentColor}[type=file]{background:unset;border-color:inherit;border-width:0;border-radius:0;padding:0;font-size:unset;line-height:inherit}[type=file]:focus{outline:1px solid ButtonText;outline:1px auto -webkit-focus-ring-color}:host [data-theme=skeleton]{background-color:rgb(var(--color-surface-50));font-size:16px;line-height:24px;font-family:var(--theme-font-family-base);color:rgba(var(--theme-font-color-base))}:host .dark [data-theme=skeleton]{background-color:rgb(var(--color-surface-900));color:rgba(var(--theme-font-color-dark))}::-moz-selection{background-color:rgb(var(--color-primary-500) / .3)}::selection{background-color:rgb(var(--color-primary-500) / .3)}:host [data-theme=skeleton]{-webkit-tap-highlight-color:rgba(128,128,128,.5);scrollbar-color:rgba(0,0,0,.2) rgba(255,255,255,.05)}:host [data-theme=skeleton]{scrollbar-color:rgba(128,128,128,.5) rgba(0,0,0,.1);scrollbar-width:thin}:host.dark{scrollbar-color:rgba(255,255,255,.1) rgba(0,0,0,.05)}hr:not(.divider){display:block;border-top-width:1px;border-style:solid;border-color:rgb(var(--color-surface-300))}.dark hr:not(.divider){border-color:rgb(var(--color-surface-600))}fieldset,label{display:block}::-moz-placeholder{color:rgb(var(--color-surface-500))}::placeholder{color:rgb(var(--color-surface-500))}.dark ::-moz-placeholder{color:rgb(var(--color-surface-400))}.dark ::placeholder{color:rgb(var(--color-surface-400))}:is(.dark input::-webkit-calendar-picker-indicator){--tw-invert: invert(100%);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}input[type=search]::-webkit-search-cancel-button{-webkit-appearance:none;background:url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E%3Cpath d='M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm121.6 313.1c4.7 4.7 4.7 12.3 0 17L338 377.6c-4.7 4.7-12.3 4.7-17 0L256 312l-65.1 65.6c-4.7 4.7-12.3 4.7-17 0L134.4 338c-4.7-4.7-4.7-12.3 0-17l65.6-65-65.6-65.1c-4.7-4.7-4.7-12.3 0-17l39.6-39.6c4.7-4.7 12.3-4.7 17 0l65 65.7 65.1-65.6c4.7-4.7 12.3-4.7 17 0l39.6 39.6c4.7 4.7 4.7 12.3 0 17L312 256l65.6 65.1z'/%3E%3C/svg%3E") no-repeat 50% 50%;pointer-events:none;height:16px;width:16px;border-radius:9999px;background-size:contain;opacity:0}input[type=search]:focus::-webkit-search-cancel-button{pointer-events:auto;opacity:1}:is(.dark input[type=search]::-webkit-search-cancel-button){--tw-invert: invert(100%);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}progress{webkit-appearance:none;-moz-appearance:none;-webkit-appearance:none;appearance:none;height:8px;width:100%;overflow:hidden;border-radius:var(--theme-rounded-base);background-color:rgb(var(--color-surface-400))}.dark progress{background-color:rgb(var(--color-surface-500))}progress::-webkit-progress-bar{background-color:rgb(var(--color-surface-400))}.dark progress::-webkit-progress-bar{background-color:rgb(var(--color-surface-500))}progress::-webkit-progress-value{background-color:rgb(var(--color-surface-900))}.dark progress::-webkit-progress-value{background-color:rgb(var(--color-surface-50))}::-moz-progress-bar{background-color:rgb(var(--color-surface-900))}.dark ::-moz-progress-bar{background-color:rgb(var(--color-surface-50))}:indeterminate::-moz-progress-bar{width:0}input[type=file]:not(.file-dropzone-input)::file-selector-button:disabled{cursor:not-allowed;opacity:.5}input[type=file]:not(.file-dropzone-input)::file-selector-button:disabled:hover{--tw-brightness: brightness(1);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}input[type=file]:not(.file-dropzone-input)::file-selector-button:disabled:active{--tw-scale-x: 1;--tw-scale-y: 1;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}input[type=file]:not(.file-dropzone-input)::file-selector-button{font-size:14px;line-height:20px;padding:6px 12px;white-space:nowrap;text-align:center;display:inline-flex;align-items:center;justify-content:center;transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s;border-radius:var(--theme-rounded-base);background-color:rgb(var(--color-surface-900));color:rgb(var(--color-surface-50));margin-right:8px;border-width:0px}input[type=file]:not(.file-dropzone-input)::file-selector-button>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(8px * var(--tw-space-x-reverse));margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))}input[type=file]:not(.file-dropzone-input)::file-selector-button:hover{--tw-brightness: brightness(1.15);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}input[type=file]:not(.file-dropzone-input)::file-selector-button:active{--tw-scale-x: 95%;--tw-scale-y: 95%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));--tw-brightness: brightness(.9);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.dark input[type=file]:not(.file-dropzone-input)::file-selector-button{background-color:rgb(var(--color-surface-50));color:rgb(var(--color-surface-900))}[type=range]{width:100%;accent-color:rgb(var(--color-surface-900) / 1)}:is(.dark [type=range]){accent-color:rgb(var(--color-surface-50) / 1)}[data-sort]{cursor:pointer}[data-sort]:hover:hover,.dark [data-sort]:hover:hover{background-color:rgb(var(--color-primary-500) / .1)}[data-sort]:after{margin-left:8px!important;opacity:0;--tw-content: "\u2193" !important;content:var(--tw-content)!important}[data-popup]{position:absolute;top:0;left:0;display:none;transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}:host [data-theme=skeleton]{--theme-font-family-base: system-ui;--theme-font-family-heading: system-ui;--theme-font-color-base: 0 0 0;--theme-font-color-dark: 255 255 255;--theme-rounded-base: 9999px;--theme-rounded-container: 8px;--theme-border-base: 1px;--on-primary: 0 0 0;--on-secondary: 255 255 255;--on-tertiary: 0 0 0;--on-success: 0 0 0;--on-warning: 0 0 0;--on-error: 255 255 255;--on-surface: 255 255 255;--color-primary-50: 219 245 236;--color-primary-100: 207 241 230;--color-primary-200: 195 238 224;--color-primary-300: 159 227 205;--color-primary-400: 87 207 167;--color-primary-500: 15 186 129;--color-primary-600: 14 167 116;--color-primary-700: 11 140 97;--color-primary-800: 9 112 77;--color-primary-900: 7 91 63;--color-secondary-50: 229 227 251;--color-secondary-100: 220 218 250;--color-secondary-200: 211 209 249;--color-secondary-300: 185 181 245;--color-secondary-400: 132 126 237;--color-secondary-500: 79 70 229;--color-secondary-600: 71 63 206;--color-secondary-700: 59 53 172;--color-secondary-800: 47 42 137;--color-secondary-900: 39 34 112;--color-tertiary-50: 219 242 252;--color-tertiary-100: 207 237 251;--color-tertiary-200: 195 233 250;--color-tertiary-300: 159 219 246;--color-tertiary-400: 86 192 240;--color-tertiary-500: 14 165 233;--color-tertiary-600: 13 149 210;--color-tertiary-700: 11 124 175;--color-tertiary-800: 8 99 140;--color-tertiary-900: 7 81 114;--color-success-50: 237 247 220;--color-success-100: 230 245 208;--color-success-200: 224 242 197;--color-success-300: 206 235 162;--color-success-400: 169 219 92;--color-success-500: 132 204 22;--color-success-600: 119 184 20;--color-success-700: 99 153 17;--color-success-800: 79 122 13;--color-success-900: 65 100 11;--color-warning-50: 252 244 218;--color-warning-100: 251 240 206;--color-warning-200: 250 236 193;--color-warning-300: 247 225 156;--color-warning-400: 240 202 82;--color-warning-500: 234 179 8;--color-warning-600: 211 161 7;--color-warning-700: 176 134 6;--color-warning-800: 140 107 5;--color-warning-900: 115 88 4;--color-error-50: 249 221 234;--color-error-100: 246 209 228;--color-error-200: 244 198 221;--color-error-300: 238 163 200;--color-error-400: 225 94 159;--color-error-500: 212 25 118;--color-error-600: 191 23 106;--color-error-700: 159 19 89;--color-error-800: 127 15 71;--color-error-900: 104 12 58;--color-surface-50: 228 230 238;--color-surface-100: 219 222 233;--color-surface-200: 210 214 227;--color-surface-300: 182 189 210;--color-surface-400: 128 140 177;--color-surface-500: 73 90 143;--color-surface-600: 66 81 129;--color-surface-700: 55 68 107;--color-surface-800: 44 54 86;--color-surface-900: 36 44 70}[data-theme=skeleton] h1,[data-theme=skeleton] h3,[data-theme=skeleton] h4{font-weight:700}[data-theme=skeleton]{background-image:radial-gradient(at 0% 0%,rgba(var(--color-secondary-500) / .33) 0px,transparent 50%),radial-gradient(at 98% 1%,rgba(var(--color-error-500) / .33) 0px,transparent 50%);background-attachment:fixed;background-position:center;background-repeat:no-repeat;background-size:cover}*{scrollbar-color:initial;scrollbar-width:initial}.\\!container{width:100%!important}.container{width:100%}@media (min-width: 640px){.\\!container{max-width:640px!important}.container{max-width:640px}}@media (min-width: 768px){.\\!container{max-width:768px!important}.container{max-width:768px}}@media (min-width: 1024px){.\\!container{max-width:1024px!important}.container{max-width:1024px}}@media (min-width: 1280px){.\\!container{max-width:1280px!important}.container{max-width:1280px}}@media (min-width: 1536px){.\\!container{max-width:1536px!important}.container{max-width:1536px}}.hide-scrollbar::-webkit-scrollbar{display:none}.hide-scrollbar{-ms-overflow-style:none;scrollbar-width:none}.h3{font-size:20px;line-height:28px;font-family:var(--theme-font-family-heading)}.h4{font-size:18px;line-height:28px;font-family:var(--theme-font-family-heading)}.anchor{--tw-text-opacity: 1;color:rgb(var(--color-primary-700) / var(--tw-text-opacity));text-decoration-line:underline}.anchor:hover{--tw-brightness: brightness(1.1);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}:is(.dark .anchor){--tw-text-opacity: 1;color:rgb(var(--color-primary-500) / var(--tw-text-opacity))}.time{font-size:14px;line-height:20px;--tw-text-opacity: 1;color:rgb(var(--color-surface-500) / var(--tw-text-opacity))}:is(.dark .time){--tw-text-opacity: 1;color:rgb(var(--color-surface-400) / var(--tw-text-opacity))}.code{white-space:nowrap;font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-size:12px;line-height:16px;--tw-text-opacity: 1;color:rgb(var(--color-primary-700) / var(--tw-text-opacity));background-color:rgb(var(--color-primary-500) / .3);border-radius:4px;padding:2px 4px}:is(.dark .code){--tw-text-opacity: 1;color:rgb(var(--color-primary-400) / var(--tw-text-opacity));background-color:rgb(var(--color-primary-500) / .2)}.alert{display:flex;flex-direction:column;align-items:flex-start;padding:16px;color:rgb(var(--color-surface-900));border-radius:var(--theme-rounded-container)}.alert>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(16px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(16px * var(--tw-space-y-reverse))}.dark .alert{color:rgb(var(--color-surface-50))}.badge{display:inline-flex;align-items:center;justify-content:center;white-space:nowrap;font-size:12px;line-height:16px;font-weight:600;padding:4px 8px;border-radius:var(--theme-rounded-base)}.badge>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(8px * var(--tw-space-x-reverse));margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))}.\\!btn:disabled{cursor:not-allowed!important;opacity:.5!important}.btn:disabled,.btn-icon:disabled,.btn-group>*:disabled{cursor:not-allowed!important;opacity:.5!important}.\\!btn:disabled:hover{--tw-brightness: brightness(1) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.btn:disabled:hover,.btn-icon:disabled:hover,.btn-group>*:disabled:hover{--tw-brightness: brightness(1);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.\\!btn:disabled:active{--tw-scale-x: 1 !important;--tw-scale-y: 1 !important;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))!important}.btn:disabled:active,.btn-icon:disabled:active,.btn-group>*:disabled:active{--tw-scale-x: 1;--tw-scale-y: 1;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.\\!btn{font-size:16px!important;line-height:24px!important;padding:9px 20px!important;white-space:nowrap!important;text-align:center!important;display:inline-flex!important;align-items:center!important;justify-content:center!important;transition-property:all!important;transition-timing-function:cubic-bezier(.4,0,.2,1)!important;transition-duration:.15s!important;border-radius:var(--theme-rounded-base)!important}.btn{font-size:16px;line-height:24px;padding:9px 20px;white-space:nowrap;text-align:center;display:inline-flex;align-items:center;justify-content:center;transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s;border-radius:var(--theme-rounded-base)}.\\!btn>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0 !important;margin-right:calc(8px * var(--tw-space-x-reverse))!important;margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))!important}.btn>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(8px * var(--tw-space-x-reverse));margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))}.\\!btn:hover{--tw-brightness: brightness(1.15) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.btn:hover{--tw-brightness: brightness(1.15);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.\\!btn:active{--tw-scale-x: 95% !important;--tw-scale-y: 95% !important;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))!important;--tw-brightness: brightness(.9) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.btn:active{--tw-scale-x: 95%;--tw-scale-y: 95%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));--tw-brightness: brightness(.9);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.btn-sm{padding:6px 12px;font-size:14px;line-height:20px}.btn-icon{font-size:16px;line-height:24px;white-space:nowrap;text-align:center;display:inline-flex;align-items:center;justify-content:center;transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s;padding:0;aspect-ratio:1 / 1;width:43px;border-radius:9999px}.btn-icon>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(8px * var(--tw-space-x-reverse));margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))}.btn-icon:hover{--tw-brightness: brightness(1.15);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.btn-icon:active{--tw-scale-x: 95%;--tw-scale-y: 95%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y));--tw-brightness: brightness(.9);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.btn-icon-sm{aspect-ratio:1 / 1;width:33px;font-size:14px;line-height:20px}.btn-group{display:inline-flex;flex-direction:row;overflow:hidden;border-radius:var(--theme-rounded-base);isolation:isolate}.btn-group>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(0px * var(--tw-space-x-reverse));margin-left:calc(0px * calc(1 - var(--tw-space-x-reverse)))}.btn-group button,.btn-group a{font-size:16px;line-height:24px;padding:9px 20px;white-space:nowrap;text-align:center;display:inline-flex;align-items:center;justify-content:center;transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s;color:inherit!important;text-decoration-line:none!important}.btn-group button>:not([hidden])~:not([hidden]),.btn-group a>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(8px * var(--tw-space-x-reverse));margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))}.btn-group button:hover,.btn-group a:hover{--tw-brightness: brightness(1.15);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow);background-color:rgb(var(--color-surface-50) / 3%)}.btn-group button:active,.btn-group a:active{background-color:rgb(var(--color-surface-900) / 3%)}.btn-group>*+*{border-top-width:0px;border-left-width:1px;border-color:rgb(var(--color-surface-500) / .2)}.card{background-color:rgb(var(--color-surface-100));--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-inset: inset;--tw-ring-color: rgb(23 23 23 / .05);border-radius:var(--theme-rounded-container)}.dark .card{background-color:rgb(var(--color-surface-800));--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-inset: inset;--tw-ring-color: rgb(250 250 250 / .05)}a.card{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}a.card:hover{--tw-brightness: brightness(1.05);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.\\!chip{cursor:pointer!important;white-space:nowrap!important;padding:6px 12px!important;text-align:center!important;font-size:12px!important;line-height:16px!important;border-radius:4px!important;display:inline-flex!important;align-items:center!important;justify-content:center!important;transition-property:all!important;transition-timing-function:cubic-bezier(.4,0,.2,1)!important;transition-duration:.15s!important}.chip{cursor:pointer;white-space:nowrap;padding:6px 12px;text-align:center;font-size:12px;line-height:16px;border-radius:4px;display:inline-flex;align-items:center;justify-content:center;transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.\\!chip>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0 !important;margin-right:calc(8px * var(--tw-space-x-reverse))!important;margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))!important}.chip>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(8px * var(--tw-space-x-reverse));margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))}a.chip:hover,button.chip:hover{--tw-brightness: brightness(1.15);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}a.\\!chip:hover,button.\\!chip:hover{--tw-brightness: brightness(1.15) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.\\!chip:disabled{cursor:not-allowed!important;opacity:.5!important}.chip:disabled{cursor:not-allowed!important;opacity:.5!important}.\\!chip:disabled:active{--tw-scale-x: 1 !important;--tw-scale-y: 1 !important;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))!important}.chip:disabled:active{--tw-scale-x: 1;--tw-scale-y: 1;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.label>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(4px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(4px * var(--tw-space-y-reverse))}.\\!input{width:100%!important;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter!important;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter!important;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter!important;transition-timing-function:cubic-bezier(.4,0,.2,1)!important;transition-duration:.2s!important;background-color:rgb(var(--color-surface-200))!important;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;border-width:var(--theme-border-base)!important;border-color:rgb(var(--color-surface-400))!important}.input,.textarea,.select,.input-group{width:100%;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.2s;background-color:rgb(var(--color-surface-200));--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;border-width:var(--theme-border-base);border-color:rgb(var(--color-surface-400))}.dark .input,.dark .textarea,.dark .select,.dark .input-group{background-color:rgb(var(--color-surface-700));border-color:rgb(var(--color-surface-500))}.dark .\\!input{background-color:rgb(var(--color-surface-700))!important;border-color:rgb(var(--color-surface-500))!important}.\\!input:hover{--tw-brightness: brightness(1.05) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.input:hover,.textarea:hover,.select:hover,.input-group:hover{--tw-brightness: brightness(1.05);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.\\!input:focus{--tw-brightness: brightness(1.05) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.input:focus,.textarea:focus,.select:focus,.input-group:focus{--tw-brightness: brightness(1.05);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.\\!input:focus-within{--tw-border-opacity: 1 !important;border-color:rgb(var(--color-primary-500) / var(--tw-border-opacity))!important}.input:focus-within,.textarea:focus-within,.select:focus-within,.input-group:focus-within{--tw-border-opacity: 1;border-color:rgb(var(--color-primary-500) / var(--tw-border-opacity))}.\\!input{border-radius:var(--theme-rounded-base)!important}.input,.input-group{border-radius:var(--theme-rounded-base)}.textarea,.select{border-radius:var(--theme-rounded-container)}.select>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(4px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(4px * var(--tw-space-y-reverse))}.select{padding:8px 32px 8px 8px}.select[size]{background-image:none}.select optgroup>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(4px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(4px * var(--tw-space-y-reverse))}.select optgroup{font-weight:700}.select optgroup option{margin-left:0;padding-left:0}.select optgroup option:first-of-type{margin-top:12px}.select optgroup option:last-child{margin-bottom:12px!important}.select option{cursor:pointer;padding:8px 16px;background-color:rgb(var(--color-surface-200));border-radius:var(--theme-rounded-base)}.dark .select option{background-color:rgb(var(--color-surface-700))}.select option:checked{background:rgb(var(--color-primary-500)) linear-gradient(0deg,rgb(var(--color-primary-500)),rgb(var(--color-primary-500)));color:rgb(var(--on-primary))}.checkbox,.radio{height:20px;width:20px;cursor:pointer;border-radius:4px;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;background-color:rgb(var(--color-surface-200));border-width:var(--theme-border-base);border-color:rgb(var(--color-surface-400))}.dark .checkbox,.dark .radio{background-color:rgb(var(--color-surface-700));border-color:rgb(var(--color-surface-500))}.checkbox:hover,.radio:hover{--tw-brightness: brightness(1.05);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.checkbox:focus,.radio:focus{--tw-brightness: brightness(1.05);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow);--tw-border-opacity: 1;border-color:rgb(var(--color-primary-500) / var(--tw-border-opacity))}.checkbox:checked,.checkbox:indeterminate,.radio:checked{--tw-bg-opacity: 1;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity))}.checkbox:checked:hover,.checkbox:indeterminate:hover,.radio:checked:hover{--tw-bg-opacity: 1;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity))}.checkbox:checked:focus,.checkbox:indeterminate:focus,.radio:checked:focus{--tw-bg-opacity: 1;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity));--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)}.radio{border-radius:var(--theme-rounded-base)}.\\!input[type=file]{padding:4px!important}.input[type=file]{padding:4px}.\\!input[type=color]{height:40px!important;width:40px!important;cursor:pointer!important;overflow:hidden!important;border-style:none!important;border-radius:var(--theme-rounded-base)!important;-webkit-appearance:none!important}.input[type=color]{height:40px;width:40px;cursor:pointer;overflow:hidden;border-style:none;border-radius:var(--theme-rounded-base);-webkit-appearance:none}.\\!input[type=color]::-webkit-color-swatch-wrapper{padding:0!important}.input[type=color]::-webkit-color-swatch-wrapper{padding:0}.\\!input[type=color]::-webkit-color-swatch{border-style:none!important}.input[type=color]::-webkit-color-swatch{border-style:none}.\\!input[type=color]::-webkit-color-swatch:hover{--tw-brightness: brightness(1.1) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.input[type=color]::-webkit-color-swatch:hover{--tw-brightness: brightness(1.1);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.\\!input[type=color]::-moz-color-swatch{border-style:none!important}.input[type=color]::-moz-color-swatch{border-style:none}.\\!input:disabled{cursor:not-allowed!important;opacity:.5!important}.input:disabled,.textarea:disabled,.select:disabled,.input-group>input:disabled,.input-group>textarea:disabled,.input-group>select:disabled{cursor:not-allowed!important;opacity:.5!important}.\\!input:disabled:hover{--tw-brightness: brightness(1) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.input:disabled:hover,.textarea:disabled:hover,.select:disabled:hover,.input-group>input:disabled:hover,.input-group>textarea:disabled:hover,.input-group>select:disabled:hover{--tw-brightness: brightness(1) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.\\!input[readonly],.input[readonly],.textarea[readonly],.select[readonly]{cursor:not-allowed!important;border-color:transparent!important}.\\!input[readonly]:hover,.input[readonly]:hover,.textarea[readonly]:hover,.select[readonly]:hover{--tw-brightness: brightness(1) !important;filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)!important}.input-group{display:grid;overflow:hidden}.input-group input,.input-group select{border-width:0px;background-color:transparent;--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important}.input-group select option{background-color:rgb(var(--color-surface-200))}.dark .input-group select option{background-color:rgb(var(--color-surface-700))}.input-group div,.input-group a,.input-group button{display:flex;align-items:center;justify-content:space-between;padding-left:16px;padding-right:16px}.input-group-divider input,.input-group-divider select,.input-group-divider div,.input-group-divider a{border-left-width:1px;border-color:rgb(var(--color-surface-400));--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;min-width:-moz-fit-content!important;min-width:fit-content!important}.dark .input-group-divider input,.dark .input-group-divider select,.dark .input-group-divider div,.dark .input-group-divider a{border-color:rgb(var(--color-surface-500))}.input-group-divider input:focus,.input-group-divider select:focus,.input-group-divider div:focus,.input-group-divider a:focus{border-color:rgb(var(--color-surface-400))}.dark .input-group-divider input:focus,.dark .input-group-divider select:focus,.dark .input-group-divider div:focus,.dark .input-group-divider a:focus{border-color:rgb(var(--color-surface-500))}.input-group-divider *:first-child{border-left-width:0px!important}.input-group-shim{background-color:rgb(var(--color-surface-400) / .1);color:rgb(var(--color-surface-600))}.dark .input-group-shim{color:rgb(var(--color-surface-300))}.input-error{--tw-border-opacity: 1;border-color:rgb(var(--color-error-500) / var(--tw-border-opacity));--tw-bg-opacity: 1;background-color:rgb(var(--color-error-200) / var(--tw-bg-opacity));--tw-text-opacity: 1;color:rgb(var(--color-error-500) / var(--tw-text-opacity))}:is(.dark .input-error){--tw-border-opacity: 1;border-color:rgb(var(--color-error-500) / var(--tw-border-opacity));--tw-bg-opacity: 1;background-color:rgb(var(--color-error-200) / var(--tw-bg-opacity));--tw-text-opacity: 1;color:rgb(var(--color-error-500) / var(--tw-text-opacity))}.input-error::-moz-placeholder{--tw-text-opacity: 1;color:rgb(var(--color-error-500) / var(--tw-text-opacity))}.input-error::placeholder{--tw-text-opacity: 1;color:rgb(var(--color-error-500) / var(--tw-text-opacity))}.variant-form-material{border-radius:4px 4px 0 0/4px 4px 0px 0px!important;background-color:rgb(var(--color-surface-500) / .1);border-width:0px;border-bottom-width:2px}:is(.dark .variant-form-material){background-color:rgb(var(--color-surface-500) / .2)}.variant-form-material[type=file]{padding-top:6px!important;padding-bottom:6px!important}.list{list-style-type:none}.list>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(4px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(4px * var(--tw-space-y-reverse))}.list li{display:flex;align-items:center;border-radius:var(--theme-rounded-base)}.list li>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(16px * var(--tw-space-x-reverse));margin-left:calc(16px * calc(1 - var(--tw-space-x-reverse)))}.placeholder{height:20px;background-color:rgb(var(--color-surface-300));border-radius:var(--theme-rounded-base)}.dark .placeholder{background-color:rgb(var(--color-surface-600))}.w-modal{width:100%;max-width:640px}.modal *:focus:not([tabindex="-1"]):not(.input):not(.textarea):not(.select):not(.input-group):not(.input-group input){outline-style:auto;outline-color:-webkit-focus-ring-color}.variant-filled{background-color:rgb(var(--color-surface-900));color:rgb(var(--color-surface-50))}.dark .variant-filled{background-color:rgb(var(--color-surface-50));color:rgb(var(--color-surface-900))}.\\!variant-filled-primary{--tw-bg-opacity: 1 !important;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity))!important;color:rgb(var(--on-primary))!important}.variant-filled-primary{--tw-bg-opacity: 1;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity));color:rgb(var(--on-primary))}:is(.dark .variant-filled-primary){--tw-bg-opacity: 1;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity));color:rgb(var(--on-primary))}:is(.dark .\\!variant-filled-primary){--tw-bg-opacity: 1 !important;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity))!important;color:rgb(var(--on-primary))!important}.variant-filled-secondary{--tw-bg-opacity: 1;background-color:rgb(var(--color-secondary-500) / var(--tw-bg-opacity));color:rgb(var(--on-secondary))}:is(.dark .variant-filled-secondary){--tw-bg-opacity: 1;background-color:rgb(var(--color-secondary-500) / var(--tw-bg-opacity));color:rgb(var(--on-secondary))}.variant-filled-warning{--tw-bg-opacity: 1;background-color:rgb(var(--color-warning-500) / var(--tw-bg-opacity));color:rgb(var(--on-warning))}:is(.dark .variant-filled-warning){--tw-bg-opacity: 1;background-color:rgb(var(--color-warning-500) / var(--tw-bg-opacity));color:rgb(var(--on-warning))}.variant-filled-error{--tw-bg-opacity: 1;background-color:rgb(var(--color-error-500) / var(--tw-bg-opacity));color:rgb(var(--on-error))}:is(.dark .variant-filled-error){--tw-bg-opacity: 1;background-color:rgb(var(--color-error-500) / var(--tw-bg-opacity));color:rgb(var(--on-error))}.variant-ghost-surface{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000);--tw-ring-inset: inset;--tw-ring-opacity: 1;--tw-ring-color: rgb(var(--color-surface-500) / var(--tw-ring-opacity));background-color:rgb(var(--color-surface-500) / .2)}:is(.dark .variant-ghost-surface){--tw-ring-opacity: 1;--tw-ring-color: rgb(var(--color-surface-500) / var(--tw-ring-opacity));background-color:rgb(var(--color-surface-500) / .2)}.variant-soft-primary{background-color:rgb(var(--color-primary-400) / .2);--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;color:rgb(var(--color-primary-700))}.dark .variant-soft-primary{color:rgb(var(--color-primary-200))}:is(.dark .variant-soft-primary){background-color:rgb(var(--color-primary-500) / .2)}.variant-soft-error{background-color:rgb(var(--color-error-400) / .2);--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;color:rgb(var(--color-error-700))}.dark .variant-soft-error{color:rgb(var(--color-error-200))}:is(.dark .variant-soft-error){background-color:rgb(var(--color-error-500) / .2)}.variant-soft,.variant-soft-surface{background-color:rgb(var(--color-surface-400) / .2);--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;color:rgb(var(--color-surface-700))}.dark .variant-soft,.dark .variant-soft-surface{color:rgb(var(--color-surface-200))}:is(.dark .variant-soft),:is(.dark .variant-soft-surface){background-color:rgb(var(--color-surface-500) / .2)}@media (min-width: 768px){.h3{font-size:24px;line-height:32px}.h4{font-size:20px;line-height:28px}}@media (min-width: 1024px){.alert{flex-direction:row;align-items:center}.alert>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(0px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(0px * var(--tw-space-y-reverse));--tw-space-x-reverse: 0;margin-right:calc(16px * var(--tw-space-x-reverse));margin-left:calc(16px * calc(1 - var(--tw-space-x-reverse)))}}.modal *:focus:not([tabindex="-1"]):not(.input):not(.textarea):not(.select):not(.input-group):not(.input-group input){outline-width:0px!important}.pointer-events-none{pointer-events:none}.pointer-events-auto{pointer-events:auto}.visible{visibility:visible}.static{position:static}.fixed{position:fixed}.\\!absolute{position:absolute!important}.absolute{position:absolute}.relative{position:relative}.sticky{position:sticky}.bottom-0{bottom:0}.bottom-24{bottom:96px}.left-0{left:0}.left-1\\/2{left:50%}.right-0{right:0}.right-2{right:8px}.right-20{right:80px}.right-4{right:16px}.top-0{top:0}.top-1\\/2{top:50%}.top-2{top:8px}.top-36{top:144px}.-z-10{z-index:-10}.z-\\[888\\]{z-index:888}.z-\\[999\\]{z-index:999}.row-span-2{grid-row:span 2 / span 2}.row-start-1{grid-row-start:1}.\\!m-0{margin:0!important}.m-auto{margin:auto}.mx-2{margin-left:8px;margin-right:8px}.my-4{margin-top:16px;margin-bottom:16px}.my-\\[1px\\]{margin-top:1px;margin-bottom:1px}.ml-1{margin-left:4px}.ml-3{margin-left:12px}.ml-4{margin-left:16px}.mr-2{margin-right:8px}.mr-6{margin-right:24px}.mt-2{margin-top:8px}.mt-4{margin-top:16px}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.inline-flex{display:inline-flex}.grid{display:grid}.contents{display:contents}.hidden{display:none}.size-14{width:56px;height:56px}.size-full{width:100%;height:100%}.h-0{height:0px}.h-10{height:40px}.h-2{height:8px}.h-4{height:16px}.h-48{height:192px}.h-6{height:24px}.h-8{height:32px}.h-\\[38px\\]{height:38px}.h-auto{height:auto}.h-fit{height:-moz-fit-content;height:fit-content}.h-full{height:100%}.h-screen{height:100vh}.max-h-80{max-height:320px}.max-h-\\[200px\\]{max-height:200px}.min-h-0{min-height:0px}.min-h-full{min-height:100%}.w-0{width:0px}.w-12{width:48px}.w-16{width:64px}.w-20{width:80px}.w-3{width:12px}.w-32{width:128px}.w-36{width:144px}.w-48{width:192px}.w-5{width:20px}.w-6{width:24px}.w-8{width:32px}.w-\\[140px\\]{width:140px}.w-\\[38px\\]{width:38px}.w-\\[50\\%\\]{width:50%}.w-\\[600px\\]{width:600px}.w-full{width:100%}.w-screen{width:100vw}.max-w-\\[640px\\]{max-width:640px}.max-w-full{max-width:100%}.flex-1{flex:1 1 0%}.flex-\\[0_0_20\\%\\]{flex:0 0 20%}.flex-\\[1\\]{flex:1}.flex-\\[3\\]{flex:3}.flex-auto{flex:1 1 auto}.flex-none{flex:none}.shrink-0{flex-shrink:0}.flex-grow{flex-grow:1}.basis-0{flex-basis:0px}.origin-\\[50\\%_50\\%\\]{transform-origin:50% 50%}.-translate-x-1\\/2{--tw-translate-x: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-translate-x-full{--tw-translate-x: -100%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-translate-y-1\\/2{--tw-translate-y: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-x-0{--tw-translate-x: 0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-x-\\[calc\\(100\\%-44px\\)\\]{--tw-translate-x: calc(100% - 44px) ;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.translate-x-full{--tw-translate-x: 100%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.-rotate-90{--tw-rotate: -90deg;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.rotate-180{--tw-rotate: 180deg;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-\\[0\\.8\\]{--tw-scale-x: .8;--tw-scale-y: .8;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.scale-\\[105\\%\\]{--tw-scale-x: 105%;--tw-scale-y: 105%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.transform{transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.\\!transform-none{transform:none!important}@keyframes spin{to{transform:rotate(360deg)}}.animate-spin{animation:spin 1s linear infinite}.\\!cursor-default{cursor:default!important}.cursor-not-allowed{cursor:not-allowed}.cursor-pointer{cursor:pointer}.select-none{-webkit-user-select:none;-moz-user-select:none;user-select:none}.list-inside{list-style-position:inside}.list-disc{list-style-type:disc}.grid-cols-\\[0px_1fr\\]{grid-template-columns:0px 1fr}.grid-cols-\\[140px_1fr\\]{grid-template-columns:140px 1fr}.grid-cols-\\[auto_1fr_auto\\]{grid-template-columns:auto 1fr auto}.grid-cols-\\[auto_1fr_auto_auto\\]{grid-template-columns:auto 1fr auto auto}.grid-rows-\\[0fr\\]{grid-template-rows:0fr}.grid-rows-\\[1fr\\]{grid-template-rows:1fr}.grid-rows-\\[auto_1fr\\]{grid-template-rows:auto 1fr}.flex-row{flex-direction:row}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-start{align-items:flex-start}.items-end{align-items:flex-end}.items-center{align-items:center}.\\!items-stretch{align-items:stretch!important}.justify-start{justify-content:flex-start}.justify-end{justify-content:flex-end}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.justify-evenly{justify-content:space-evenly}.gap-1{gap:4px}.gap-14{gap:56px}.gap-2{gap:8px}.gap-3{gap:12px}.gap-4{gap:16px}.gap-6{gap:24px}.gap-x-2{-moz-column-gap:8px;column-gap:8px}.gap-y-1{row-gap:4px}.gap-y-2{row-gap:8px}.space-x-2>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(8px * var(--tw-space-x-reverse));margin-left:calc(8px * calc(1 - var(--tw-space-x-reverse)))}.space-x-4>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(16px * var(--tw-space-x-reverse));margin-left:calc(16px * calc(1 - var(--tw-space-x-reverse)))}.space-y-1>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(4px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(4px * var(--tw-space-y-reverse))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(8px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(8px * var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(16px * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(16px * var(--tw-space-y-reverse))}.divide-y-\\[1px\\]>:not([hidden])~:not([hidden]){--tw-divide-y-reverse: 0;border-top-width:calc(1px * calc(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(1px * var(--tw-divide-y-reverse))}.self-start{align-self:flex-start}.self-center{align-self:center}.self-stretch{align-self:stretch}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.overflow-y-hidden{overflow-y:hidden}.truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.break-words{overflow-wrap:break-word}.rounded{border-radius:4px}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:8px}.rounded-md{border-radius:6px}.rounded-none{border-radius:0}.rounded-e-\\[4px\\]{border-start-end-radius:4px;border-end-end-radius:4px}.rounded-l-none{border-top-left-radius:0;border-bottom-left-radius:0}.rounded-r-none{border-top-right-radius:0;border-bottom-right-radius:0}.rounded-s-full{border-start-start-radius:9999px;border-end-start-radius:9999px}.border{border-width:1px}.border-0{border-width:0px}.\\!border-t-0{border-top-width:0px!important}.border-b{border-bottom-width:1px}.border-b-2{border-bottom-width:2px}.border-l{border-left-width:1px}.\\!border-surface-700{--tw-border-opacity: 1 !important;border-color:rgb(var(--color-surface-700) / var(--tw-border-opacity, 1))!important}.border-surface-400\\/20{border-color:rgb(var(--color-surface-400) / .2)}.bg-primary-500{--tw-bg-opacity: 1;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity, 1))}.bg-primary-500\\/30{background-color:rgb(var(--color-primary-500) / .3)}.bg-surface-400{--tw-bg-opacity: 1;background-color:rgb(var(--color-surface-400) / var(--tw-bg-opacity, 1))}.bg-surface-400\\/20{background-color:rgb(var(--color-surface-400) / .2)}.bg-surface-900{--tw-bg-opacity: 1;background-color:rgb(var(--color-surface-900) / var(--tw-bg-opacity, 1))}.bg-transparent{background-color:transparent}.bg-white{--tw-bg-opacity: 1;background-color:rgb(255 255 255 / var(--tw-bg-opacity, 1))}.bg-white\\/30{background-color:#ffffff4d}.bg-white\\/75{background-color:#ffffffbf}.bg-scroll{background-attachment:scroll}.fill-current{fill:currentColor}.fill-primary-500{fill:rgb(var(--color-primary-500) / 1)}.fill-slate-700{fill:#334155}.fill-transparent{fill:transparent}.stroke-primary-500{stroke:rgb(var(--color-primary-500) / 1)}.stroke-primary-500\\/30{stroke:rgb(var(--color-primary-500) / .3)}.stroke-surface-500\\/30{stroke:rgb(var(--color-surface-500) / .3)}.stroke-surface-900{stroke:rgb(var(--color-surface-900) / 1)}.object-cover{-o-object-fit:cover;object-fit:cover}.object-center{-o-object-position:center;object-position:center}.\\!p-0{padding:0!important}.p-0{padding:0}.p-1{padding:4px}.p-2{padding:8px}.p-4{padding:16px}.\\!px-1{padding-left:4px!important;padding-right:4px!important}.\\!py-2{padding-top:8px!important;padding-bottom:8px!important}.\\!py-\\[7px\\]{padding-top:7px!important;padding-bottom:7px!important}.px-3{padding-left:12px;padding-right:12px}.px-4{padding-left:16px;padding-right:16px}.px-8{padding-left:32px;padding-right:32px}.py-1{padding-top:4px;padding-bottom:4px}.py-2{padding-top:8px;padding-bottom:8px}.py-6{padding-top:24px;padding-bottom:24px}.pb-4{padding-bottom:16px}.pb-6{padding-bottom:24px}.pl-6{padding-left:24px}.pr-0{padding-right:0}.pr-2{padding-right:8px}.pr-4{padding-right:16px}.pr-6{padding-right:24px}.pt-4{padding-top:16px}.text-center{text-align:center}.text-start{text-align:start}.text-2xl{font-size:24px;line-height:32px}.text-\\[12px\\]{font-size:12px}.text-base{font-size:16px;line-height:24px}.text-sm{font-size:14px;line-height:20px}.text-xl{font-size:20px;line-height:28px}.text-xs{font-size:12px;line-height:16px}.font-bold{font-weight:700}.italic{font-style:italic}.leading-\\[14px\\]{line-height:14px}.leading-loose{line-height:2}.\\!text-error-500{--tw-text-opacity: 1 !important;color:rgb(var(--color-error-500) / var(--tw-text-opacity, 1))!important}.text-error-400{--tw-text-opacity: 1;color:rgb(var(--color-error-400) / var(--tw-text-opacity, 1))}.text-error-500{--tw-text-opacity: 1;color:rgb(var(--color-error-500) / var(--tw-text-opacity, 1))}.text-surface-400{--tw-text-opacity: 1;color:rgb(var(--color-surface-400) / var(--tw-text-opacity, 1))}.underline-offset-2{text-underline-offset:2px}.accent-surface-900{accent-color:rgb(var(--color-surface-900) / 1)}.opacity-40{opacity:.4}.opacity-50{opacity:.5}.opacity-70{opacity:.7}.mix-blend-hard-light{mix-blend-mode:hard-light}.shadow{--tw-shadow: 0 1px 3px 0 rgb(0 0 0 / .1), 0 1px 2px -1px rgb(0 0 0 / .1);--tw-shadow-colored: 0 1px 3px 0 var(--tw-shadow-color), 0 1px 2px -1px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-lg{--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / .1), 0 4px 6px -4px rgb(0 0 0 / .1);--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.shadow-xl{--tw-shadow: 0 20px 25px -5px rgb(0 0 0 / .1), 0 8px 10px -6px rgb(0 0 0 / .1);--tw-shadow-colored: 0 20px 25px -5px var(--tw-shadow-color), 0 8px 10px -6px var(--tw-shadow-color);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow)}.-outline-offset-\\[3px\\]{outline-offset:-3px}.\\!ring-0{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important}.blur{--tw-blur: blur(8px);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.blur-\\[1px\\]{--tw-blur: blur(1px);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.drop-shadow-xl{--tw-drop-shadow: drop-shadow(0 20px 13px rgb(0 0 0 / .03)) drop-shadow(0 8px 5px rgb(0 0 0 / .08));filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur-sm{--tw-backdrop-blur: blur(4px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.transition{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,-webkit-backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter,-webkit-backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-\\[grid-template-columns\\]{transition-property:grid-template-columns;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-\\[grid-template-rows\\]{transition-property:grid-template-rows;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-\\[stroke-dashoffset\\]{transition-property:stroke-dashoffset;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-\\[transform\\]{transition-property:transform;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-\\[width\\]{transition-property:width;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-opacity{transition-property:opacity;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.transition-transform{transition-property:transform;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.delay-100{transition-delay:.1s}.duration-100{transition-duration:.1s}.duration-\\[200ms\\]{transition-duration:.2s}.duration-\\[250ms\\]{transition-duration:.25s}.duration-\\[400ms\\]{transition-duration:.4s}.bg-surface-backdrop-token{background-color:rgb(var(--color-surface-400) / .7)}.dark .bg-surface-backdrop-token{background-color:rgb(var(--color-surface-900) / .7)}.bg-surface-100-800-token{background-color:rgb(var(--color-surface-100))}.dark .bg-surface-100-800-token{background-color:rgb(var(--color-surface-800))}.bg-surface-200-700-token{background-color:rgb(var(--color-surface-200))}.dark .bg-surface-200-700-token{background-color:rgb(var(--color-surface-700))}.bg-surface-900-50-token{background-color:rgb(var(--color-surface-900))}.dark .bg-surface-900-50-token{background-color:rgb(var(--color-surface-50))}.border-token{border-width:var(--theme-border-base)}.border-surface-400-500-token{border-color:rgb(var(--color-surface-400))}.dark .border-surface-400-500-token{border-color:rgb(var(--color-surface-500))}.border-surface-900-50-token{border-color:rgb(var(--color-surface-900))}.dark .border-surface-900-50-token{border-color:rgb(var(--color-surface-50))}.border-surface-800-100-token{border-color:rgb(var(--color-surface-800))}.dark .border-surface-800-100-token{border-color:rgb(var(--color-surface-100))}.rounded-token{border-radius:var(--theme-rounded-base)}.rounded-container-token{border-radius:var(--theme-rounded-container)}.rounded-tl-container-token{border-top-left-radius:var(--theme-rounded-container)}.rounded-tr-container-token{border-top-right-radius:var(--theme-rounded-container)}.fill-token{fill:rgba(var(--theme-font-color-base))}.dark .fill-token{fill:rgba(var(--theme-font-color-dark))}.text-surface-700-200-token{color:rgb(var(--color-surface-700))}.dark .text-surface-700-200-token{color:rgb(var(--color-surface-200))}.scrollbar-thin::-webkit-scrollbar-track{background-color:var(--scrollbar-track);border-radius:var(--scrollbar-track-radius)}.scrollbar-thin::-webkit-scrollbar-track:hover{background-color:var(--scrollbar-track-hover, var(--scrollbar-track))}.scrollbar-thin::-webkit-scrollbar-track:active{background-color:var(--scrollbar-track-active, var(--scrollbar-track-hover, var(--scrollbar-track)))}.scrollbar-thin::-webkit-scrollbar-thumb{background-color:var(--scrollbar-thumb);border-radius:var(--scrollbar-thumb-radius)}.scrollbar-thin::-webkit-scrollbar-thumb:hover{background-color:var(--scrollbar-thumb-hover, var(--scrollbar-thumb))}.scrollbar-thin::-webkit-scrollbar-thumb:active{background-color:var(--scrollbar-thumb-active, var(--scrollbar-thumb-hover, var(--scrollbar-thumb)))}.scrollbar-thin::-webkit-scrollbar-corner{background-color:var(--scrollbar-corner);border-radius:var(--scrollbar-corner-radius)}.scrollbar-thin::-webkit-scrollbar-corner:hover{background-color:var(--scrollbar-corner-hover, var(--scrollbar-corner))}.scrollbar-thin::-webkit-scrollbar-corner:active{background-color:var(--scrollbar-corner-active, var(--scrollbar-corner-hover, var(--scrollbar-corner)))}.scrollbar-thin{scrollbar-width:thin;scrollbar-color:var(--scrollbar-thumb, initial) var(--scrollbar-track, initial)}.scrollbar-thin::-webkit-scrollbar{display:block;width:8px;height:8px}.scrollbar-track-transparent{--scrollbar-track: transparent !important}.scrollbar-thumb-slate-400\\/50{--scrollbar-thumb: rgb(148 163 184 / .5) !important}.scrollbar-corner-transparent{--scrollbar-corner: transparent !important}.dark .hover\\:bg-primary-hover-token:hover:hover{background-color:rgb(var(--color-primary-500) / .1)}.has-\\[\\:checked\\]\\:\\!variant-filled-primary:has(:checked){--tw-bg-opacity: 1 !important;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity))!important;color:rgb(var(--on-primary))!important}:is(.dark .has-\\[\\:checked\\]\\:\\!variant-filled-primary:has(:checked)){--tw-bg-opacity: 1 !important;background-color:rgb(var(--color-primary-500) / var(--tw-bg-opacity))!important;color:rgb(var(--on-primary))!important}.hover\\:variant-filled:hover{background-color:rgb(var(--color-surface-900));color:rgb(var(--color-surface-50))}.dark .hover\\:variant-filled:hover{background-color:rgb(var(--color-surface-50));color:rgb(var(--color-surface-900))}.hover\\:variant-soft:hover,.hover\\:variant-soft-surface:hover{background-color:rgb(var(--color-surface-400) / .2);--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;color:rgb(var(--color-surface-700))}.dark .hover\\:variant-soft:hover,.dark .hover\\:variant-soft-surface:hover{color:rgb(var(--color-surface-200))}:is(.dark .hover\\:variant-soft:hover){background-color:rgb(var(--color-surface-500) / .2)}:is(.dark .hover\\:variant-soft-surface:hover){background-color:rgb(var(--color-surface-500) / .2)}.\\[\\&\\:not\\(\\[disabled\\]\\)\\]\\:variant-soft-primary:not([disabled]){background-color:rgb(var(--color-primary-400) / .2);--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color) !important;--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color) !important;box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)!important;color:rgb(var(--color-primary-700))}.dark .\\[\\&\\:not\\(\\[disabled\\]\\)\\]\\:variant-soft-primary:not([disabled]){color:rgb(var(--color-primary-200))}:is(.dark .\\[\\&\\:not\\(\\[disabled\\]\\)\\]\\:variant-soft-primary:not([disabled])){background-color:rgb(var(--color-primary-500) / .2)}.\\*\\:\\!m-0>*{margin:0!important}.\\*\\:items-center>*{align-items:center}.\\*\\:\\!rounded-none>*{border-radius:0!important}.\\*\\:py-2>*{padding-top:8px;padding-bottom:8px}.\\*\\:py-4>*{padding-top:16px;padding-bottom:16px}.\\*\\:text-sm>*{font-size:14px;line-height:20px}.\\*\\:border-surface-300-600-token>*{border-color:rgb(var(--color-surface-300))}.dark .\\*\\:border-surface-300-600-token>*{border-color:rgb(var(--color-surface-600))}.hover\\:translate-x-0:hover{--tw-translate-x: 0px;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}.hover\\:text-xl:hover{font-size:20px;line-height:28px}.hover\\:opacity-100:hover{opacity:1}.hover\\:brightness-110:hover{--tw-brightness: brightness(1.1);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.hover\\:brightness-\\[105\\%\\]:hover{--tw-brightness: brightness(105%);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.hover\\:bg-primary-hover-token:hover:hover,.dark .hover\\:bg-primary-hover-token:hover:hover{background-color:rgb(var(--color-primary-500) / .1)}.focus\\:decoration-wavy:focus{text-decoration-style:wavy}.focus\\:\\!outline-none:focus{outline:2px solid transparent!important;outline-offset:2px!important}.disabled\\:cursor-wait:disabled{cursor:wait}.disabled\\:opacity-70:disabled{opacity:.7}.dark\\:\\!border-surface-200:is(.dark *){--tw-border-opacity: 1 !important;border-color:rgb(var(--color-surface-200) / var(--tw-border-opacity, 1))!important}.dark\\:border-surface-500\\/20:is(.dark *){border-color:rgb(var(--color-surface-500) / .2)}.dark\\:bg-black\\/15:is(.dark *){background-color:#00000026}.dark\\:bg-surface-300:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(var(--color-surface-300) / var(--tw-bg-opacity, 1))}.dark\\:bg-surface-500\\/20:is(.dark *){background-color:rgb(var(--color-surface-500) / .2)}.dark\\:bg-surface-700:is(.dark *){--tw-bg-opacity: 1;background-color:rgb(var(--color-surface-700) / var(--tw-bg-opacity, 1))}.dark\\:fill-slate-200:is(.dark *){fill:#e2e8f0}.dark\\:stroke-surface-50:is(.dark *){stroke:rgb(var(--color-surface-50) / 1)}.dark\\:accent-surface-50:is(.dark *){accent-color:rgb(var(--color-surface-50) / 1)}.dark\\:hover\\:brightness-110:hover:is(.dark *){--tw-brightness: brightness(1.1);filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}@media (min-width: 768px){.md\\:h-\\[600px\\]{height:600px}.md\\:max-w-screen-md{max-width:768px}.md\\:max-w-screen-sm{max-width:640px}.md\\:flex-row{flex-direction:row}.md\\:\\!items-center{align-items:center!important}.md\\:\\!items-baseline{align-items:baseline!important}}@media (min-width: 1024px){.lg\\:max-w-screen-md{max-width:768px}}@media (min-width: 1280px){.xl\\:max-w-screen-lg{max-width:1024px}}.\\[\\&\\:last-child\\]\\:\\*\\:pt-4>*:last-child{padding-top:16px}.\\[\\&\\:not\\(\\:last-child\\)\\]\\:\\*\\:py-4>*:not(:last-child){padding-top:16px;padding-bottom:16px}.\\[\\&\\:not\\(\\[disabled\\]\\)\\]\\:hover\\:bg-slate-400\\/30:hover:not([disabled]){background-color:#94a3b84d}.\\[\\&\\>input\\]\\:\\!min-w-0>input{min-width:0px!important}.\\[\\&\\>input\\]\\:\\!border-transparent>input{border-color:transparent!important} `); (function (Dexie, GIF, webmMuxer, mp4Muxer, dayjs, JSZip) { 'use strict'; var __defProp = Object.defineProperty; var __typeError = (msg) => { throw TypeError(msg); }; var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value); var __accessCheck = (obj, member, msg) => member.has(obj) || __typeError("Cannot " + msg); var __privateGet = (obj, member, getter) => (__accessCheck(obj, member, "read from private field"), getter ? getter.call(obj) : member.get(obj)); var __privateAdd = (obj, member, value) => member.has(obj) ? __typeError("Cannot add the same private member more than once") : member instanceof WeakSet ? member.add(obj) : member.set(obj, value); var __privateSet = (obj, member, value, setter) => (__accessCheck(obj, member, "write to private field"), member.set(obj, value), value); var __privateMethod = (obj, member, method) => (__accessCheck(obj, member, "access private method"), method); var __privateWrapper = (obj, member, setter, getter) => ({ set _(value) { __privateSet(obj, member, value); }, get _() { return __privateGet(obj, member, getter); } }); var _events, _instance, _channel, _event, _DIRECTORY_HANDLE_NAME, _downloadingId, _downloadingPage, _ThumbnailButton_instances, resetStatus_fn, _stickyContainer, _thumbnailButton, _unsubscribers, _fn, _subscribe, _a, _fn2, _subscribe2, _queue, _carryoverConcurrencyCount, _isIntervalIgnored, _intervalCount, _intervalCap, _interval, _intervalEnd, _intervalId, _timeoutId, _queue2, _queueClass, _pending, _concurrency, _isPaused, _throwOnTimeout, _PQueue_instances, doesIntervalAllowAnother_get, doesConcurrentAllowAnother_get, next_fn, onResumeInterval_fn, isIntervalPaused_get, tryToStartAnother_fn, initializeIntervalIfNeeded_fn, onInterval_fn, processQueue_fn, throwOnAbort_fn, onEvent_fn, _DOWNLOAD_RETRY, _downloadQueue, _Downloader_instances, xhr_fn, dispatchDownload_fn, _b, _c, _d, _MediaDownloadConfig_instances, replaceTemplate_fn, _GelbooruV020_instances, validityCheckFactory_fn, addBookmark_fn, _DanbooruParser_instances, parseBlacklistItem_fn, _AbstractDanbooru_instances, validityCheckFactory_fn2, _ugoiraFramesData, _queue3, _Converter_instances, processConvert_fn, _MoebooruParser_instances, parsePostListData_fn, parseTagListData_fn, parseBlacklist_fn, _Moebooru_instances, validityCallbackFactory_fn, buildMetaByGeneratorData_fn, getPopularDataFactory_fn, addBookmark_fn2, downloadArtwork_fn, _Konachan_instances, fixPoolImageStyle_fn, _authParams, _E621ng_instances, isPoolGallery_fn, isPoolView_fn, isPostView_fn, isFavoritesPage_fn, isPostsPage_fn, isAuthorized_fn, throwIfNotAuthorized_fn, validityCallbackFactory_fn2, addFavorites_fn, _NijieParser_instances, parseIdByAnchors_fn, _searchParams, _Nijie_instances, isViewPage_fn, isViewPopupPage_fn, isOkazuPage_fn, isSupportedUserPage_fn, isSupportedHistoryPage_fn, getSearchId_fn, addBookmark_fn3, _Rule34VaultApi_instances, requestSearch_fn, _tagTypeMap, _currentPage, _notifyPageChange, _Rule34Vault_instances, getCurrentSearchSetting_fn, addBookmark_fn4, downloadArtwork_fn2, createOrUpdatePostActionButton_fn, createThumbnailButton_fn, _Rule34PahealParser_instances, buildMetaByThumbnailEl_fn, _api, _parser, _Rule34Paheal_instances, isPostList_fn, isPostView_fn2, downloadArtwork_fn3, createArtworkButton_fn, createThumbnailButtons_fn, _Rule34UsParser_instances, getTagsByEL_fn, _api2, _parser2, _searchParam, _Rule34Us_instances, isPostList_fn2, isPostView_fn3, isFavorites_fn, getQueryTag_fn, getQueryId_fn, addBookmark_fn5, downloadArtwork_fn4, createThumbnailButtons_fn2, createArtworkButton_fn2; (() => { if (/\[native code\]/.test(Array.from.toString())) return; const iframe = document.createElement("iframe"); document.body.append(iframe); Array.from = iframe.contentWindow.Array.from; iframe.remove(); })(); const DEV = false; var is_array = Array.isArray; var index_of = Array.prototype.indexOf; var array_from = Array.from; var object_keys = Object.keys; var define_property = Object.defineProperty; var get_descriptor = Object.getOwnPropertyDescriptor; var get_descriptors = Object.getOwnPropertyDescriptors; var object_prototype = Object.prototype; var array_prototype = Array.prototype; var get_prototype_of = Object.getPrototypeOf; function is_function(thing) { return typeof thing === "function"; } const noop = () => { }; function run(fn) { return fn(); } function run_all(arr) { for (var i = 0; i < arr.length; i++) { arr[i](); } } const DERIVED = 1 << 1; const EFFECT = 1 << 2; const RENDER_EFFECT = 1 << 3; const BLOCK_EFFECT = 1 << 4; const BRANCH_EFFECT = 1 << 5; const ROOT_EFFECT = 1 << 6; const BOUNDARY_EFFECT = 1 << 7; const UNOWNED = 1 << 8; const DISCONNECTED = 1 << 9; const CLEAN = 1 << 10; const DIRTY = 1 << 11; const MAYBE_DIRTY = 1 << 12; const INERT = 1 << 13; const DESTROYED = 1 << 14; const EFFECT_RAN = 1 << 15; const EFFECT_TRANSPARENT = 1 << 16; const LEGACY_DERIVED_PROP = 1 << 17; const HEAD_EFFECT = 1 << 19; const EFFECT_HAS_DERIVED = 1 << 20; const STATE_SYMBOL = Symbol("$state"); const LEGACY_PROPS = Symbol("legacy props"); const LOADING_ATTR_SYMBOL = Symbol(""); const request_idle_callback = typeof requestIdleCallback === "undefined" ? (cb) => setTimeout(cb, 1) : requestIdleCallback; let is_micro_task_queued$1 = false; let is_idle_task_queued = false; let current_queued_micro_tasks = []; let current_queued_idle_tasks = []; function process_micro_tasks() { is_micro_task_queued$1 = false; const tasks = current_queued_micro_tasks.slice(); current_queued_micro_tasks = []; run_all(tasks); } function process_idle_tasks() { is_idle_task_queued = false; const tasks = current_queued_idle_tasks.slice(); current_queued_idle_tasks = []; run_all(tasks); } function queue_micro_task(fn) { if (!is_micro_task_queued$1) { is_micro_task_queued$1 = true; queueMicrotask(process_micro_tasks); } current_queued_micro_tasks.push(fn); } function queue_idle_task(fn) { if (!is_idle_task_queued) { is_idle_task_queued = true; request_idle_callback(process_idle_tasks); } current_queued_idle_tasks.push(fn); } function flush_tasks() { if (is_micro_task_queued$1) { process_micro_tasks(); } if (is_idle_task_queued) { process_idle_tasks(); } } function hydration_mismatch(location2) { { console.warn(`https://svelte.dev/e/hydration_mismatch`); } } const EACH_ITEM_REACTIVE = 1; const EACH_INDEX_REACTIVE = 1 << 1; const EACH_IS_CONTROLLED = 1 << 2; const EACH_IS_ANIMATED = 1 << 3; const EACH_ITEM_IMMUTABLE = 1 << 4; const PROPS_IS_IMMUTABLE = 1; const PROPS_IS_RUNES = 1 << 1; const PROPS_IS_UPDATED = 1 << 2; const PROPS_IS_BINDABLE = 1 << 3; const PROPS_IS_LAZY_INITIAL = 1 << 4; const TRANSITION_IN = 1; const TRANSITION_OUT = 1 << 1; const TRANSITION_GLOBAL = 1 << 2; const TEMPLATE_FRAGMENT = 1; const TEMPLATE_USE_IMPORT_NODE = 1 << 1; const HYDRATION_START = "["; const HYDRATION_START_ELSE = "[!"; const HYDRATION_END = "]"; const HYDRATION_ERROR = {}; const UNINITIALIZED = Symbol(); function equals(value) { return value === this.v; } function safe_not_equal(a, b) { return a != a ? b == b : a !== b || a !== null && typeof a === "object" || typeof a === "function"; } function not_equal(a, b) { return a !== b; } function safe_equals(value) { return !safe_not_equal(value, this.v); } function effect_in_teardown(rune) { { throw new Error(`https://svelte.dev/e/effect_in_teardown`); } } function effect_in_unowned_derived() { { throw new Error(`https://svelte.dev/e/effect_in_unowned_derived`); } } function effect_orphan(rune) { { throw new Error(`https://svelte.dev/e/effect_orphan`); } } function effect_update_depth_exceeded() { { throw new Error(`https://svelte.dev/e/effect_update_depth_exceeded`); } } function hydration_failed() { { throw new Error(`https://svelte.dev/e/hydration_failed`); } } function lifecycle_legacy_only(name) { { throw new Error(`https://svelte.dev/e/lifecycle_legacy_only`); } } function props_invalid_value(key) { { throw new Error(`https://svelte.dev/e/props_invalid_value`); } } function state_descriptors_fixed() { { throw new Error(`https://svelte.dev/e/state_descriptors_fixed`); } } function state_prototype_fixed() { { throw new Error(`https://svelte.dev/e/state_prototype_fixed`); } } function state_unsafe_local_read() { { throw new Error(`https://svelte.dev/e/state_unsafe_local_read`); } } function state_unsafe_mutation() { { throw new Error(`https://svelte.dev/e/state_unsafe_mutation`); } } let legacy_mode_flag = false; let tracing_mode_flag = false; function enable_legacy_mode_flag() { legacy_mode_flag = true; } const empty = []; function snapshot(value, skip_warning = false) { return clone(value, /* @__PURE__ */ new Map(), "", empty); } function clone(value, cloned, path, paths, original = null) { if (typeof value === "object" && value !== null) { var unwrapped = cloned.get(value); if (unwrapped !== undefined) return unwrapped; if (value instanceof Map) return ( /** @type {Snapshot<T>} */ new Map(value) ); if (value instanceof Set) return ( /** @type {Snapshot<T>} */ new Set(value) ); if (is_array(value)) { var copy = ( /** @type {Snapshot<any>} */ Array(value.length) ); cloned.set(value, copy); if (original !== null) { cloned.set(original, copy); } for (var i = 0; i < value.length; i += 1) { var element = value[i]; if (i in value) { copy[i] = clone(element, cloned, path, paths); } } return copy; } if (get_prototype_of(value) === object_prototype) { copy = {}; cloned.set(value, copy); if (original !== null) { cloned.set(original, copy); } for (var key in value) { copy[key] = clone(value[key], cloned, path, paths); } return copy; } if (value instanceof Date) { return ( /** @type {Snapshot<T>} */ structuredClone(value) ); } if (typeof /** @type {T & { toJSON?: any } } */ value.toJSON === "function") { return clone( /** @type {T & { toJSON(): any } } */ value.toJSON(), cloned, path, paths, // Associate the instance with the toJSON clone value ); } } if (value instanceof EventTarget) { return ( /** @type {Snapshot<T>} */ value ); } try { return ( /** @type {Snapshot<T>} */ structuredClone(value) ); } catch (e) { return ( /** @type {Snapshot<T>} */ value ); } } function source(v, stack) { var signal = { f: 0, // TODO ideally we could skip this altogether, but it causes type errors v, reactions: null, equals, rv: 0, wv: 0 }; return signal; } function state(v) { return /* @__PURE__ */ push_derived_source(source(v)); } // @__NO_SIDE_EFFECTS__ function mutable_source(initial_value, immutable = false) { var _a2; const s = source(initial_value); if (!immutable) { s.equals = safe_equals; } if (legacy_mode_flag && component_context !== null && component_context.l !== null) { ((_a2 = component_context.l).s ?? (_a2.s = [])).push(s); } return s; } function mutable_state(v, immutable = false) { return /* @__PURE__ */ push_derived_source(/* @__PURE__ */ mutable_source(v, immutable)); } // @__NO_SIDE_EFFECTS__ function push_derived_source(source2) { if (active_reaction !== null && (active_reaction.f & DERIVED) !== 0) { if (derived_sources === null) { set_derived_sources([source2]); } else { derived_sources.push(source2); } } return source2; } function set(source2, value) { if (active_reaction !== null && is_runes() && (active_reaction.f & (DERIVED | BLOCK_EFFECT)) !== 0 && // If the source was created locally within the current derived, then // we allow the mutation. (derived_sources === null || !derived_sources.includes(source2))) { state_unsafe_mutation(); } return internal_set(source2, value); } function internal_set(source2, value) { if (!source2.equals(value)) { source2.v; source2.v = value; source2.wv = increment_write_version(); mark_reactions(source2, DIRTY); if (is_runes() && active_effect !== null && (active_effect.f & CLEAN) !== 0 && (active_effect.f & (BRANCH_EFFECT | ROOT_EFFECT)) === 0) { if (untracked_writes === null) { set_untracked_writes([source2]); } else { untracked_writes.push(source2); } } } return value; } function mark_reactions(signal, status) { var reactions = signal.reactions; if (reactions === null) return; var runes = is_runes(); var length = reactions.length; for (var i = 0; i < length; i++) { var reaction = reactions[i]; var flags = reaction.f; if ((flags & DIRTY) !== 0) continue; if (!runes && reaction === active_effect) continue; set_signal_status(reaction, status); if ((flags & (CLEAN | UNOWNED)) !== 0) { if ((flags & DERIVED) !== 0) { mark_reactions( /** @type {Derived} */ reaction, MAYBE_DIRTY ); } else { schedule_effect( /** @type {Effect} */ reaction ); } } } } // @__NO_SIDE_EFFECTS__ function derived(fn) { var flags = DERIVED | DIRTY; if (active_effect === null) { flags |= UNOWNED; } else { active_effect.f |= EFFECT_HAS_DERIVED; } var parent_derived = active_reaction !== null && (active_reaction.f & DERIVED) !== 0 ? ( /** @type {Derived} */ active_reaction ) : null; const signal = { children: null, ctx: component_context, deps: null, equals, f: flags, fn, reactions: null, rv: 0, v: ( /** @type {V} */ null ), wv: 0, parent: parent_derived ?? active_effect }; if (parent_derived !== null) { (parent_derived.children ?? (parent_derived.children = [])).push(signal); } return signal; } // @__NO_SIDE_EFFECTS__ function derived_safe_equal(fn) { const signal = /* @__PURE__ */ derived(fn); signal.equals = safe_equals; return signal; } function destroy_derived_children(derived2) { var children = derived2.children; if (children !== null) { derived2.children = null; for (var i = 0; i < children.length; i += 1) { var child2 = children[i]; if ((child2.f & DERIVED) !== 0) { destroy_derived( /** @type {Derived} */ child2 ); } else { destroy_effect( /** @type {Effect} */ child2 ); } } } } function get_derived_parent_effect(derived2) { var parent = derived2.parent; while (parent !== null) { if ((parent.f & DERIVED) === 0) { return ( /** @type {Effect} */ parent ); } parent = parent.parent; } return null; } function execute_derived(derived2) { var value; var prev_active_effect = active_effect; set_active_effect(get_derived_parent_effect(derived2)); { try { destroy_derived_children(derived2); value = update_reaction(derived2); } finally { set_active_effect(prev_active_effect); } } return value; } function update_derived(derived2) { var value = execute_derived(derived2); var status = (skip_reaction || (derived2.f & UNOWNED) !== 0) && derived2.deps !== null ? MAYBE_DIRTY : CLEAN; set_signal_status(derived2, status); if (!derived2.equals(value)) { derived2.v = value; derived2.wv = increment_write_version(); } } function destroy_derived(derived2) { destroy_derived_children(derived2); remove_reactions(derived2, 0); set_signal_status(derived2, DESTROYED); derived2.v = derived2.children = derived2.deps = derived2.ctx = derived2.reactions = null; } function lifecycle_outside_component(name) { { throw new Error(`https://svelte.dev/e/lifecycle_outside_component`); } } const FLUSH_MICROTASK = 0; const FLUSH_SYNC = 1; let is_throwing_error = false; let scheduler_mode = FLUSH_MICROTASK; let is_micro_task_queued = false; let last_scheduled_effect = null; let is_flushing_effect = false; let is_destroying_effect = false; function set_is_flushing_effect(value) { is_flushing_effect = value; } function set_is_destroying_effect(value) { is_destroying_effect = value; } let queued_root_effects = []; let flush_count = 0; let active_reaction = null; function set_active_reaction(reaction) { active_reaction = reaction; } let active_effect = null; function set_active_effect(effect2) { active_effect = effect2; } let derived_sources = null; function set_derived_sources(sources) { derived_sources = sources; } let new_deps = null; let skipped_deps = 0; let untracked_writes = null; function set_untracked_writes(value) { untracked_writes = value; } let write_version = 1; let read_version = 0; let skip_reaction = false; let captured_signals = null; let component_context = null; function increment_write_version() { return ++write_version; } function is_runes() { return !legacy_mode_flag || component_context !== null && component_context.l === null; } function check_dirtiness(reaction) { var _a2; var flags = reaction.f; if ((flags & DIRTY) !== 0) { return true; } if ((flags & MAYBE_DIRTY) !== 0) { var dependencies = reaction.deps; var is_unowned = (flags & UNOWNED) !== 0; if (dependencies !== null) { var i; var dependency; var is_disconnected = (flags & DISCONNECTED) !== 0; var is_unowned_connected = is_unowned && active_effect !== null && !skip_reaction; var length = dependencies.length; if (is_disconnected || is_unowned_connected) { for (i = 0; i < length; i++) { dependency = dependencies[i]; if (is_disconnected || !((_a2 = dependency == null ? undefined : dependency.reactions) == null ? undefined : _a2.includes(reaction))) { (dependency.reactions ?? (dependency.reactions = [])).push(reaction); } } if (is_disconnected) { reaction.f ^= DISCONNECTED; } } for (i = 0; i < length; i++) { dependency = dependencies[i]; if (check_dirtiness( /** @type {Derived} */ dependency )) { update_derived( /** @type {Derived} */ dependency ); } if (dependency.wv > reaction.wv) { return true; } } } if (!is_unowned || active_effect !== null && !skip_reaction) { set_signal_status(reaction, CLEAN); } } return false; } function propagate_error(error, effect2) { var current = effect2; while (current !== null) { if ((current.f & BOUNDARY_EFFECT) !== 0) { try { current.fn(error); return; } catch { current.f ^= BOUNDARY_EFFECT; } } current = current.parent; } is_throwing_error = false; throw error; } function should_rethrow_error(effect2) { return (effect2.f & DESTROYED) === 0 && (effect2.parent === null || (effect2.parent.f & BOUNDARY_EFFECT) === 0); } function handle_error(error, effect2, previous_effect, component_context2) { if (is_throwing_error) { if (previous_effect === null) { is_throwing_error = false; } if (should_rethrow_error(effect2)) { throw error; } return; } if (previous_effect !== null) { is_throwing_error = true; } { propagate_error(error, effect2); return; } } function schedule_possible_effect_self_invalidation(signal, effect2, depth = 0) { var reactions = signal.reactions; if (reactions === null) return; for (var i = 0; i < reactions.length; i++) { var reaction = reactions[i]; if ((reaction.f & DERIVED) !== 0) { schedule_possible_effect_self_invalidation( /** @type {Derived} */ reaction, effect2, depth + 1 ); } else if (effect2 === reaction) { if (depth === 0) { set_signal_status(reaction, DIRTY); } else if ((reaction.f & CLEAN) !== 0) { set_signal_status(reaction, MAYBE_DIRTY); } schedule_effect( /** @type {Effect} */ reaction ); } } } function update_reaction(reaction) { var _a2; var previous_deps = new_deps; var previous_skipped_deps = skipped_deps; var previous_untracked_writes = untracked_writes; var previous_reaction = active_reaction; var previous_skip_reaction = skip_reaction; var prev_derived_sources = derived_sources; var previous_component_context = component_context; var flags = reaction.f; new_deps = /** @type {null | Value[]} */ null; skipped_deps = 0; untracked_writes = null; active_reaction = (flags & (BRANCH_EFFECT | ROOT_EFFECT)) === 0 ? reaction : null; skip_reaction = !is_flushing_effect && (flags & UNOWNED) !== 0; derived_sources = null; component_context = reaction.ctx; read_version++; try { var result = ( /** @type {Function} */ (0, reaction.fn)() ); var deps = reaction.deps; if (new_deps !== null) { var i; remove_reactions(reaction, skipped_deps); if (deps !== null && skipped_deps > 0) { deps.length = skipped_deps + new_deps.length; for (i = 0; i < new_deps.length; i++) { deps[skipped_deps + i] = new_deps[i]; } } else { reaction.deps = deps = new_deps; } if (!skip_reaction) { for (i = skipped_deps; i < deps.length; i++) { ((_a2 = deps[i]).reactions ?? (_a2.reactions = [])).push(reaction); } } } else if (deps !== null && skipped_deps < deps.length) { remove_reactions(reaction, skipped_deps); deps.length = skipped_deps; } if (is_runes() && untracked_writes !== null && (reaction.f & (DERIVED | MAYBE_DIRTY | DIRTY)) === 0) { for (i = 0; i < /** @type {Source[]} */ untracked_writes.length; i++) { schedule_possible_effect_self_invalidation( untracked_writes[i], /** @type {Effect} */ reaction ); } } if (previous_reaction !== null) { read_version++; } return result; } finally { new_deps = previous_deps; skipped_deps = previous_skipped_deps; untracked_writes = previous_untracked_writes; active_reaction = previous_reaction; skip_reaction = previous_skip_reaction; derived_sources = prev_derived_sources; component_context = previous_component_context; } } function remove_reaction(signal, dependency) { let reactions = dependency.reactions; if (reactions !== null) { var index2 = index_of.call(reactions, signal); if (index2 !== -1) { var new_length = reactions.length - 1; if (new_length === 0) { reactions = dependency.reactions = null; } else { reactions[index2] = reactions[new_length]; reactions.pop(); } } } if (reactions === null && (dependency.f & DERIVED) !== 0 && // Destroying a child effect while updating a parent effect can cause a dependency to appear // to be unused, when in fact it is used by the currently-updating parent. Checking `new_deps` // allows us to skip the expensive work of disconnecting and immediately reconnecting it (new_deps === null || !new_deps.includes(dependency))) { set_signal_status(dependency, MAYBE_DIRTY); if ((dependency.f & (UNOWNED | DISCONNECTED)) === 0) { dependency.f ^= DISCONNECTED; } remove_reactions( /** @type {Derived} **/ dependency, 0 ); } } function remove_reactions(signal, start_index) { var dependencies = signal.deps; if (dependencies === null) return; for (var i = start_index; i < dependencies.length; i++) { remove_reaction(signal, dependencies[i]); } } function update_effect(effect2) { var flags = effect2.f; if ((flags & DESTROYED) !== 0) { return; } set_signal_status(effect2, CLEAN); var previous_effect = active_effect; var previous_component_context = component_context; active_effect = effect2; try { if ((flags & BLOCK_EFFECT) !== 0) { destroy_block_effect_children(effect2); } else { destroy_effect_children(effect2); } destroy_effect_deriveds(effect2); execute_effect_teardown(effect2); var teardown2 = update_reaction(effect2); effect2.teardown = typeof teardown2 === "function" ? teardown2 : null; effect2.wv = write_version; var deps = effect2.deps; var dep; if (DEV && tracing_mode_flag && (effect2.f & DIRTY) !== 0 && deps !== null) ; if (DEV) ; } catch (error) { handle_error(error, effect2, previous_effect, previous_component_context || effect2.ctx); } finally { active_effect = previous_effect; } } function infinite_loop_guard() { if (flush_count > 1e3) { flush_count = 0; try { effect_update_depth_exceeded(); } catch (error) { if (last_scheduled_effect !== null) { { handle_error(error, last_scheduled_effect, null); } } else { throw error; } } } flush_count++; } function flush_queued_root_effects(root_effects) { var length = root_effects.length; if (length === 0) { return; } infinite_loop_guard(); var previously_flushing_effect = is_flushing_effect; is_flushing_effect = true; try { for (var i = 0; i < length; i++) { var effect2 = root_effects[i]; if ((effect2.f & CLEAN) === 0) { effect2.f ^= CLEAN; } var collected_effects = []; process_effects(effect2, collected_effects); flush_queued_effects(collected_effects); } } finally { is_flushing_effect = previously_flushing_effect; } } function flush_queued_effects(effects) { var length = effects.length; if (length === 0) return; for (var i = 0; i < length; i++) { var effect2 = effects[i]; if ((effect2.f & (DESTROYED | INERT)) === 0) { try { if (check_dirtiness(effect2)) { update_effect(effect2); if (effect2.deps === null && effect2.first === null && effect2.nodes_start === null) { if (effect2.teardown === null) { unlink_effect(effect2); } else { effect2.fn = null; } } } } catch (error) { handle_error(error, effect2, null, effect2.ctx); } } } } function process_deferred() { is_micro_task_queued = false; if (flush_count > 1001) { return; } const previous_queued_root_effects = queued_root_effects; queued_root_effects = []; flush_queued_root_effects(previous_queued_root_effects); if (!is_micro_task_queued) { flush_count = 0; last_scheduled_effect = null; } } function schedule_effect(signal) { if (scheduler_mode === FLUSH_MICROTASK) { if (!is_micro_task_queued) { is_micro_task_queued = true; queueMicrotask(process_deferred); } } last_scheduled_effect = signal; var effect2 = signal; while (effect2.parent !== null) { effect2 = effect2.parent; var flags = effect2.f; if ((flags & (ROOT_EFFECT | BRANCH_EFFECT)) !== 0) { if ((flags & CLEAN) === 0) return; effect2.f ^= CLEAN; } } queued_root_effects.push(effect2); } function process_effects(effect2, collected_effects) { var current_effect = effect2.first; var effects = []; main_loop: while (current_effect !== null) { var flags = current_effect.f; var is_branch = (flags & BRANCH_EFFECT) !== 0; var is_skippable_branch = is_branch && (flags & CLEAN) !== 0; var sibling2 = current_effect.next; if (!is_skippable_branch && (flags & INERT) === 0) { if ((flags & RENDER_EFFECT) !== 0) { if (is_branch) { current_effect.f ^= CLEAN; } else { try { if (check_dirtiness(current_effect)) { update_effect(current_effect); } } catch (error) { handle_error(error, current_effect, null, current_effect.ctx); } } var child2 = current_effect.first; if (child2 !== null) { current_effect = child2; continue; } } else if ((flags & EFFECT) !== 0) { effects.push(current_effect); } } if (sibling2 === null) { let parent = current_effect.parent; while (parent !== null) { if (effect2 === parent) { break main_loop; } var parent_sibling = parent.next; if (parent_sibling !== null) { current_effect = parent_sibling; continue main_loop; } parent = parent.parent; } } current_effect = sibling2; } for (var i = 0; i < effects.length; i++) { child2 = effects[i]; collected_effects.push(child2); process_effects(child2, collected_effects); } } function flush_sync(fn) { var previous_scheduler_mode = scheduler_mode; var previous_queued_root_effects = queued_root_effects; try { infinite_loop_guard(); const root_effects = []; scheduler_mode = FLUSH_SYNC; queued_root_effects = root_effects; is_micro_task_queued = false; flush_queued_root_effects(previous_queued_root_effects); var result = fn == null ? void 0 : fn(); flush_tasks(); if (queued_root_effects.length > 0 || root_effects.length > 0) { flush_sync(); } flush_count = 0; last_scheduled_effect = null; if (DEV) ; return result; } finally { scheduler_mode = previous_scheduler_mode; queued_root_effects = previous_queued_root_effects; } } async function tick() { await Promise.resolve(); flush_sync(); } function get$1(signal) { var _a2; var flags = signal.f; var is_derived = (flags & DERIVED) !== 0; if (is_derived && (flags & DESTROYED) !== 0) { var value = execute_derived( /** @type {Derived} */ signal ); destroy_derived( /** @type {Derived} */ signal ); return value; } if (captured_signals !== null) { captured_signals.add(signal); } if (active_reaction !== null) { if (derived_sources !== null && derived_sources.includes(signal)) { state_unsafe_local_read(); } var deps = active_reaction.deps; if (signal.rv < read_version) { signal.rv = read_version; if (new_deps === null && deps !== null && deps[skipped_deps] === signal) { skipped_deps++; } else if (new_deps === null) { new_deps = [signal]; } else { new_deps.push(signal); } } } else if (is_derived && /** @type {Derived} */ signal.deps === null) { var derived2 = ( /** @type {Derived} */ signal ); var parent = derived2.parent; var target = derived2; while (parent !== null) { if ((parent.f & DERIVED) !== 0) { var parent_derived = ( /** @type {Derived} */ parent ); target = parent_derived; parent = parent_derived.parent; } else { var parent_effect = ( /** @type {Effect} */ parent ); if (!((_a2 = parent_effect.deriveds) == null ? undefined : _a2.includes(target))) { (parent_effect.deriveds ?? (parent_effect.deriveds = [])).push(target); } break; } } } if (is_derived) { derived2 = /** @type {Derived} */ signal; if (check_dirtiness(derived2)) { update_derived(derived2); } } return signal.v; } function capture_signals(fn) { var previous_captured_signals = captured_signals; captured_signals = /* @__PURE__ */ new Set(); var captured = captured_signals; var signal; try { untrack(fn); if (previous_captured_signals !== null) { for (signal of captured_signals) { previous_captured_signals.add(signal); } } } finally { captured_signals = previous_captured_signals; } return captured; } function invalidate_inner_signals(fn) { var captured = capture_signals(() => untrack(fn)); for (var signal of captured) { if ((signal.f & LEGACY_DERIVED_PROP) !== 0) { for ( const dep of /** @type {Derived} */ signal.deps || [] ) { if ((dep.f & DERIVED) === 0) { internal_set(dep, dep.v); } } } else { internal_set(signal, signal.v); } } } function untrack(fn) { const previous_reaction = active_reaction; try { active_reaction = null; return fn(); } finally { active_reaction = previous_reaction; } } const STATUS_MASK = -7169; function set_signal_status(signal, status) { signal.f = signal.f & STATUS_MASK | status; } function getContext(key) { const context_map = get_or_init_context_map(); const result = ( /** @type {T} */ context_map.get(key) ); return result; } function setContext(key, context) { const context_map = get_or_init_context_map(); context_map.set(key, context); return context; } function get_or_init_context_map(name) { if (component_context === null) { lifecycle_outside_component(); } return component_context.c ?? (component_context.c = new Map(get_parent_context(component_context) || undefined)); } function get_parent_context(component_context2) { let parent = component_context2.p; while (parent !== null) { const context_map = parent.c; if (context_map !== null) { return context_map; } parent = parent.p; } return null; } function update(signal, d = 1) { var value = get$1(signal); var result = d === 1 ? value++ : value--; set(signal, value); return result; } function push(props, runes = false, fn) { component_context = { p: component_context, c: null, e: null, m: false, s: props, x: null, l: null }; if (legacy_mode_flag && !runes) { component_context.l = { s: null, u: null, r1: [], r2: source(false) }; } } function pop(component2) { const context_stack_item = component_context; if (context_stack_item !== null) { if (component2 !== undefined) { context_stack_item.x = component2; } const component_effects = context_stack_item.e; if (component_effects !== null) { var previous_effect = active_effect; var previous_reaction = active_reaction; context_stack_item.e = null; try { for (var i = 0; i < component_effects.length; i++) { var component_effect = component_effects[i]; set_active_effect(component_effect.effect); set_active_reaction(component_effect.reaction); effect(component_effect.fn); } } finally { set_active_effect(previous_effect); set_active_reaction(previous_reaction); } } component_context = context_stack_item.p; context_stack_item.m = true; } return component2 || /** @type {T} */ {}; } function deep_read_state(value) { if (typeof value !== "object" || !value || value instanceof EventTarget) { return; } if (STATE_SYMBOL in value) { deep_read(value); } else if (!Array.isArray(value)) { for (let key in value) { const prop2 = value[key]; if (typeof prop2 === "object" && prop2 && STATE_SYMBOL in prop2) { deep_read(prop2); } } } } function deep_read(value, visited = /* @__PURE__ */ new Set()) { if (typeof value === "object" && value !== null && // We don't want to traverse DOM elements !(value instanceof EventTarget) && !visited.has(value)) { visited.add(value); if (value instanceof Date) { value.getTime(); } for (let key in value) { try { deep_read(value[key], visited); } catch (e) { } } const proto = get_prototype_of(value); if (proto !== Object.prototype && proto !== Array.prototype && proto !== Map.prototype && proto !== Set.prototype && proto !== Date.prototype) { const descriptors = get_descriptors(proto); for (let key in descriptors) { const get2 = descriptors[key].get; if (get2) { try { get2.call(value); } catch (e) { } } } } } } let hydrating = false; function set_hydrating(value) { hydrating = value; } let hydrate_node; function set_hydrate_node(node) { if (node === null) { hydration_mismatch(); throw HYDRATION_ERROR; } return hydrate_node = node; } function hydrate_next() { return set_hydrate_node( /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(hydrate_node) ); } function reset(node) { if (!hydrating) return; if (/* @__PURE__ */ get_next_sibling(hydrate_node) !== null) { hydration_mismatch(); throw HYDRATION_ERROR; } hydrate_node = node; } function next(count = 1) { if (hydrating) { var i = count; var node = hydrate_node; while (i--) { node = /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(node); } hydrate_node = node; } } function remove_nodes() { var depth = 0; var node = hydrate_node; while (true) { if (node.nodeType === 8) { var data = ( /** @type {Comment} */ node.data ); if (data === HYDRATION_END) { if (depth === 0) return node; depth -= 1; } else if (data === HYDRATION_START || data === HYDRATION_START_ELSE) { depth += 1; } } var next2 = ( /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(node) ); node.remove(); node = next2; } } function proxy(value, parent = null, prev) { if (typeof value !== "object" || value === null || STATE_SYMBOL in value) { return value; } const prototype = get_prototype_of(value); if (prototype !== object_prototype && prototype !== array_prototype) { return value; } var sources = /* @__PURE__ */ new Map(); var is_proxied_array = is_array(value); var version = source(0); if (is_proxied_array) { sources.set("length", source( /** @type {any[]} */ value.length )); } var metadata; return new Proxy( /** @type {any} */ value, { defineProperty(_, prop2, descriptor) { if (!("value" in descriptor) || descriptor.configurable === false || descriptor.enumerable === false || descriptor.writable === false) { state_descriptors_fixed(); } var s = sources.get(prop2); if (s === undefined) { s = source(descriptor.value); sources.set(prop2, s); } else { set(s, proxy(descriptor.value, metadata)); } return true; }, deleteProperty(target, prop2) { var s = sources.get(prop2); if (s === undefined) { if (prop2 in target) { sources.set(prop2, source(UNINITIALIZED)); } } else { if (is_proxied_array && typeof prop2 === "string") { var ls = ( /** @type {Source<number>} */ sources.get("length") ); var n = Number(prop2); if (Number.isInteger(n) && n < ls.v) { set(ls, n); } } set(s, UNINITIALIZED); update_version(version); } return true; }, get(target, prop2, receiver) { var _a2; if (prop2 === STATE_SYMBOL) { return value; } var s = sources.get(prop2); var exists = prop2 in target; if (s === undefined && (!exists || ((_a2 = get_descriptor(target, prop2)) == null ? undefined : _a2.writable))) { s = source(proxy(exists ? target[prop2] : UNINITIALIZED, metadata)); sources.set(prop2, s); } if (s !== undefined) { var v = get$1(s); return v === UNINITIALIZED ? undefined : v; } return Reflect.get(target, prop2, receiver); }, getOwnPropertyDescriptor(target, prop2) { var descriptor = Reflect.getOwnPropertyDescriptor(target, prop2); if (descriptor && "value" in descriptor) { var s = sources.get(prop2); if (s) descriptor.value = get$1(s); } else if (descriptor === undefined) { var source2 = sources.get(prop2); var value2 = source2 == null ? undefined : source2.v; if (source2 !== undefined && value2 !== UNINITIALIZED) { return { enumerable: true, configurable: true, value: value2, writable: true }; } } return descriptor; }, has(target, prop2) { var _a2; if (prop2 === STATE_SYMBOL) { return true; } var s = sources.get(prop2); var has = s !== undefined && s.v !== UNINITIALIZED || Reflect.has(target, prop2); if (s !== undefined || active_effect !== null && (!has || ((_a2 = get_descriptor(target, prop2)) == null ? undefined : _a2.writable))) { if (s === undefined) { s = source(has ? proxy(target[prop2], metadata) : UNINITIALIZED); sources.set(prop2, s); } var value2 = get$1(s); if (value2 === UNINITIALIZED) { return false; } } return has; }, set(target, prop2, value2, receiver) { var _a2; var s = sources.get(prop2); var has = prop2 in target; if (is_proxied_array && prop2 === "length") { for (var i = value2; i < /** @type {Source<number>} */ s.v; i += 1) { var other_s = sources.get(i + ""); if (other_s !== undefined) { set(other_s, UNINITIALIZED); } else if (i in target) { other_s = source(UNINITIALIZED); sources.set(i + "", other_s); } } } if (s === undefined) { if (!has || ((_a2 = get_descriptor(target, prop2)) == null ? undefined : _a2.writable)) { s = source(undefined); set(s, proxy(value2, metadata)); sources.set(prop2, s); } } else { has = s.v !== UNINITIALIZED; set(s, proxy(value2, metadata)); } var descriptor = Reflect.getOwnPropertyDescriptor(target, prop2); if (descriptor == null ? undefined : descriptor.set) { descriptor.set.call(receiver, value2); } if (!has) { if (is_proxied_array && typeof prop2 === "string") { var ls = ( /** @type {Source<number>} */ sources.get("length") ); var n = Number(prop2); if (Number.isInteger(n) && n >= ls.v) { set(ls, n + 1); } } update_version(version); } return true; }, ownKeys(target) { get$1(version); var own_keys = Reflect.ownKeys(target).filter((key2) => { var source3 = sources.get(key2); return source3 === undefined || source3.v !== UNINITIALIZED; }); for (var [key, source2] of sources) { if (source2.v !== UNINITIALIZED && !(key in target)) { own_keys.push(key); } } return own_keys; }, setPrototypeOf() { state_prototype_fixed(); } } ); } function update_version(signal, d = 1) { set(signal, signal.v + d); } function get_proxied_value(value) { if (value !== null && typeof value === "object" && STATE_SYMBOL in value) { return value[STATE_SYMBOL]; } return value; } function is(a, b) { return Object.is(get_proxied_value(a), get_proxied_value(b)); } var $window; var first_child_getter; var next_sibling_getter; function init_operations() { if ($window !== undefined) { return; } $window = window; var element_prototype = Element.prototype; var node_prototype = Node.prototype; first_child_getter = get_descriptor(node_prototype, "firstChild").get; next_sibling_getter = get_descriptor(node_prototype, "nextSibling").get; element_prototype.__click = undefined; element_prototype.__className = ""; element_prototype.__attributes = null; element_prototype.__styles = null; element_prototype.__e = undefined; Text.prototype.__t = undefined; } function create_text(value = "") { return document.createTextNode(value); } // @__NO_SIDE_EFFECTS__ function get_first_child(node) { return first_child_getter.call(node); } // @__NO_SIDE_EFFECTS__ function get_next_sibling(node) { return next_sibling_getter.call(node); } function child(node, is_text) { if (!hydrating) { return /* @__PURE__ */ get_first_child(node); } var child2 = ( /** @type {TemplateNode} */ /* @__PURE__ */ get_first_child(hydrate_node) ); if (child2 === null) { child2 = hydrate_node.appendChild(create_text()); } else if (is_text && child2.nodeType !== 3) { var text2 = create_text(); child2 == null ? undefined : child2.before(text2); set_hydrate_node(text2); return text2; } set_hydrate_node(child2); return child2; } function first_child(fragment, is_text) { if (!hydrating) { var first = ( /** @type {DocumentFragment} */ /* @__PURE__ */ get_first_child( /** @type {Node} */ fragment ) ); if (first instanceof Comment && first.data === "") return /* @__PURE__ */ get_next_sibling(first); return first; } return hydrate_node; } function sibling(node, count = 1, is_text = false) { let next_sibling = hydrating ? hydrate_node : node; var last_sibling; while (count--) { last_sibling = next_sibling; next_sibling = /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(next_sibling); } if (!hydrating) { return next_sibling; } var type = next_sibling == null ? undefined : next_sibling.nodeType; if (is_text && type !== 3) { var text2 = create_text(); if (next_sibling === null) { last_sibling == null ? undefined : last_sibling.after(text2); } else { next_sibling.before(text2); } set_hydrate_node(text2); return text2; } set_hydrate_node(next_sibling); return ( /** @type {TemplateNode} */ next_sibling ); } function clear_text_content(node) { node.textContent = ""; } function validate_effect(rune) { if (active_effect === null && active_reaction === null) { effect_orphan(); } if (active_reaction !== null && (active_reaction.f & UNOWNED) !== 0) { effect_in_unowned_derived(); } if (is_destroying_effect) { effect_in_teardown(); } } function push_effect(effect2, parent_effect) { var parent_last = parent_effect.last; if (parent_last === null) { parent_effect.last = parent_effect.first = effect2; } else { parent_last.next = effect2; effect2.prev = parent_last; parent_effect.last = effect2; } } function create_effect(type, fn, sync, push2 = true) { var is_root = (type & ROOT_EFFECT) !== 0; var parent_effect = active_effect; var effect2 = { ctx: component_context, deps: null, deriveds: null, nodes_start: null, nodes_end: null, f: type | DIRTY, first: null, fn, last: null, next: null, parent: is_root ? null : parent_effect, prev: null, teardown: null, transitions: null, wv: 0 }; if (sync) { var previously_flushing_effect = is_flushing_effect; try { set_is_flushing_effect(true); update_effect(effect2); effect2.f |= EFFECT_RAN; } catch (e) { destroy_effect(effect2); throw e; } finally { set_is_flushing_effect(previously_flushing_effect); } } else if (fn !== null) { schedule_effect(effect2); } var inert = sync && effect2.deps === null && effect2.first === null && effect2.nodes_start === null && effect2.teardown === null && (effect2.f & (EFFECT_HAS_DERIVED | BOUNDARY_EFFECT)) === 0; if (!inert && !is_root && push2) { if (parent_effect !== null) { push_effect(effect2, parent_effect); } if (active_reaction !== null && (active_reaction.f & DERIVED) !== 0) { var derived2 = ( /** @type {Derived} */ active_reaction ); (derived2.children ?? (derived2.children = [])).push(effect2); } } return effect2; } function effect_tracking() { if (active_reaction === null) { return false; } return !skip_reaction; } function teardown(fn) { const effect2 = create_effect(RENDER_EFFECT, null, false); set_signal_status(effect2, CLEAN); effect2.teardown = fn; return effect2; } function user_effect(fn) { validate_effect(); var defer = active_effect !== null && (active_effect.f & BRANCH_EFFECT) !== 0 && component_context !== null && !component_context.m; if (defer) { var context = ( /** @type {ComponentContext} */ component_context ); (context.e ?? (context.e = [])).push({ fn, effect: active_effect, reaction: active_reaction }); } else { var signal = effect(fn); return signal; } } function user_pre_effect(fn) { validate_effect(); return render_effect(fn); } function effect_root(fn) { const effect2 = create_effect(ROOT_EFFECT, fn, true); return () => { destroy_effect(effect2); }; } function component_root(fn) { const effect2 = create_effect(ROOT_EFFECT, fn, true); return (options = {}) => { return new Promise((fulfil) => { if (options.outro) { pause_effect(effect2, () => { destroy_effect(effect2); fulfil(undefined); }); } else { destroy_effect(effect2); fulfil(undefined); } }); }; } function effect(fn) { return create_effect(EFFECT, fn, false); } function legacy_pre_effect(deps, fn) { var context = ( /** @type {ComponentContextLegacy} */ component_context ); var token = { effect: null, ran: false }; context.l.r1.push(token); token.effect = render_effect(() => { deps(); if (token.ran) return; token.ran = true; set(context.l.r2, true); untrack(fn); }); } function legacy_pre_effect_reset() { var context = ( /** @type {ComponentContextLegacy} */ component_context ); render_effect(() => { if (!get$1(context.l.r2)) return; for (var token of context.l.r1) { var effect2 = token.effect; if ((effect2.f & CLEAN) !== 0) { set_signal_status(effect2, MAYBE_DIRTY); } if (check_dirtiness(effect2)) { update_effect(effect2); } token.ran = false; } context.l.r2.v = false; }); } function render_effect(fn) { return create_effect(RENDER_EFFECT, fn, true); } function template_effect(fn) { return block(fn); } function block(fn, flags = 0) { return create_effect(RENDER_EFFECT | BLOCK_EFFECT | flags, fn, true); } function branch(fn, push2 = true) { return create_effect(RENDER_EFFECT | BRANCH_EFFECT, fn, true, push2); } function execute_effect_teardown(effect2) { var teardown2 = effect2.teardown; if (teardown2 !== null) { const previously_destroying_effect = is_destroying_effect; const previous_reaction = active_reaction; set_is_destroying_effect(true); set_active_reaction(null); try { teardown2.call(null); } finally { set_is_destroying_effect(previously_destroying_effect); set_active_reaction(previous_reaction); } } } function destroy_effect_deriveds(signal) { var deriveds = signal.deriveds; if (deriveds !== null) { signal.deriveds = null; for (var i = 0; i < deriveds.length; i += 1) { destroy_derived(deriveds[i]); } } } function destroy_effect_children(signal, remove_dom = false) { var effect2 = signal.first; signal.first = signal.last = null; while (effect2 !== null) { var next2 = effect2.next; destroy_effect(effect2, remove_dom); effect2 = next2; } } function destroy_block_effect_children(signal) { var effect2 = signal.first; while (effect2 !== null) { var next2 = effect2.next; if ((effect2.f & BRANCH_EFFECT) === 0) { destroy_effect(effect2); } effect2 = next2; } } function destroy_effect(effect2, remove_dom = true) { var removed = false; if ((remove_dom || (effect2.f & HEAD_EFFECT) !== 0) && effect2.nodes_start !== null) { var node = effect2.nodes_start; var end = effect2.nodes_end; while (node !== null) { var next2 = node === end ? null : ( /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(node) ); node.remove(); node = next2; } removed = true; } destroy_effect_children(effect2, remove_dom && !removed); destroy_effect_deriveds(effect2); remove_reactions(effect2, 0); set_signal_status(effect2, DESTROYED); var transitions = effect2.transitions; if (transitions !== null) { for (const transition2 of transitions) { transition2.stop(); } } execute_effect_teardown(effect2); var parent = effect2.parent; if (parent !== null && parent.first !== null) { unlink_effect(effect2); } effect2.next = effect2.prev = effect2.teardown = effect2.ctx = effect2.deps = effect2.fn = effect2.nodes_start = effect2.nodes_end = null; } function unlink_effect(effect2) { var parent = effect2.parent; var prev = effect2.prev; var next2 = effect2.next; if (prev !== null) prev.next = next2; if (next2 !== null) next2.prev = prev; if (parent !== null) { if (parent.first === effect2) parent.first = next2; if (parent.last === effect2) parent.last = prev; } } function pause_effect(effect2, callback) { var transitions = []; pause_children(effect2, transitions, true); run_out_transitions(transitions, () => { destroy_effect(effect2); if (callback) callback(); }); } function run_out_transitions(transitions, fn) { var remaining = transitions.length; if (remaining > 0) { var check2 = () => --remaining || fn(); for (var transition2 of transitions) { transition2.out(check2); } } else { fn(); } } function pause_children(effect2, transitions, local) { if ((effect2.f & INERT) !== 0) return; effect2.f ^= INERT; if (effect2.transitions !== null) { for (const transition2 of effect2.transitions) { if (transition2.is_global || local) { transitions.push(transition2); } } } var child2 = effect2.first; while (child2 !== null) { var sibling2 = child2.next; var transparent = (child2.f & EFFECT_TRANSPARENT) !== 0 || (child2.f & BRANCH_EFFECT) !== 0; pause_children(child2, transitions, transparent ? local : false); child2 = sibling2; } } function resume_effect(effect2) { resume_children(effect2, true); } function resume_children(effect2, local) { if ((effect2.f & INERT) === 0) return; effect2.f ^= INERT; if ((effect2.f & CLEAN) === 0) { effect2.f ^= CLEAN; } if (check_dirtiness(effect2)) { set_signal_status(effect2, DIRTY); schedule_effect(effect2); } var child2 = effect2.first; while (child2 !== null) { var sibling2 = child2.next; var transparent = (child2.f & EFFECT_TRANSPARENT) !== 0 || (child2.f & BRANCH_EFFECT) !== 0; resume_children(child2, transparent ? local : false); child2 = sibling2; } if (effect2.transitions !== null) { for (const transition2 of effect2.transitions) { if (transition2.is_global || local) { transition2.in(); } } } } function is_capture_event(name) { return name.endsWith("capture") && name !== "gotpointercapture" && name !== "lostpointercapture"; } const DELEGATED_EVENTS = [ "beforeinput", "click", "change", "dblclick", "contextmenu", "focusin", "focusout", "input", "keydown", "keyup", "mousedown", "mousemove", "mouseout", "mouseover", "mouseup", "pointerdown", "pointermove", "pointerout", "pointerover", "pointerup", "touchend", "touchmove", "touchstart" ]; function is_delegated(event_name) { return DELEGATED_EVENTS.includes(event_name); } const ATTRIBUTE_ALIASES = { // no `class: 'className'` because we handle that separately formnovalidate: "formNoValidate", ismap: "isMap", nomodule: "noModule", playsinline: "playsInline", readonly: "readOnly", defaultvalue: "defaultValue", defaultchecked: "defaultChecked", srcobject: "srcObject" }; function normalize_attribute(name) { name = name.toLowerCase(); return ATTRIBUTE_ALIASES[name] ?? name; } const PASSIVE_EVENTS = ["touchstart", "touchmove"]; function is_passive_event(name) { return PASSIVE_EVENTS.includes(name); } function autofocus(dom, value) { if (value) { const body = document.body; dom.autofocus = true; queue_micro_task(() => { if (document.activeElement === body) { dom.focus(); } }); } } let listening_to_form_reset = false; function add_form_reset_listener() { if (!listening_to_form_reset) { listening_to_form_reset = true; document.addEventListener( "reset", (evt) => { Promise.resolve().then(() => { var _a2; if (!evt.defaultPrevented) { for ( const e of /**@type {HTMLFormElement} */ evt.target.elements ) { (_a2 = e.__on_r) == null ? undefined : _a2.call(e); } } }); }, // In the capture phase to guarantee we get noticed of it (no possiblity of stopPropagation) { capture: true } ); } } function listen(target, events, handler, call_handler_immediately = true) { if (call_handler_immediately) { handler(); } for (var name of events) { target.addEventListener(name, handler); } teardown(() => { for (var name2 of events) { target.removeEventListener(name2, handler); } }); } function without_reactive_context(fn) { var previous_reaction = active_reaction; var previous_effect = active_effect; set_active_reaction(null); set_active_effect(null); try { return fn(); } finally { set_active_reaction(previous_reaction); set_active_effect(previous_effect); } } function listen_to_event_and_reset_event(element, event2, handler, on_reset = handler) { element.addEventListener(event2, () => without_reactive_context(handler)); const prev = element.__on_r; if (prev) { element.__on_r = () => { prev(); on_reset(true); }; } else { element.__on_r = () => on_reset(true); } add_form_reset_listener(); } const all_registered_events = /* @__PURE__ */ new Set(); const root_event_handles = /* @__PURE__ */ new Set(); function create_event(event_name, dom, handler, options) { function target_handler(event2) { if (!options.capture) { handle_event_propagation.call(dom, event2); } if (!event2.cancelBubble) { return without_reactive_context(() => { return handler.call(this, event2); }); } } if (event_name.startsWith("pointer") || event_name.startsWith("touch") || event_name === "wheel") { queue_micro_task(() => { dom.addEventListener(event_name, target_handler, options); }); } else { dom.addEventListener(event_name, target_handler, options); } return target_handler; } function on(element, type, handler, options = {}) { var target_handler = create_event(type, element, handler, options); return () => { element.removeEventListener(type, target_handler, options); }; } function event(event_name, dom, handler, capture, passive) { var options = { capture, passive }; var target_handler = create_event(event_name, dom, handler, options); if (dom === document.body || dom === window || dom === document) { teardown(() => { dom.removeEventListener(event_name, target_handler, options); }); } } function delegate(events) { for (var i = 0; i < events.length; i++) { all_registered_events.add(events[i]); } for (var fn of root_event_handles) { fn(events); } } function handle_event_propagation(event2) { var _a2; var handler_element = this; var owner_document = ( /** @type {Node} */ handler_element.ownerDocument ); var event_name = event2.type; var path = ((_a2 = event2.composedPath) == null ? undefined : _a2.call(event2)) || []; var current_target = ( /** @type {null | Element} */ path[0] || event2.target ); var path_idx = 0; var handled_at = event2.__root; if (handled_at) { var at_idx = path.indexOf(handled_at); if (at_idx !== -1 && (handler_element === document || handler_element === /** @type {any} */ window)) { event2.__root = handler_element; return; } var handler_idx = path.indexOf(handler_element); if (handler_idx === -1) { return; } if (at_idx <= handler_idx) { path_idx = at_idx; } } current_target = /** @type {Element} */ path[path_idx] || event2.target; if (current_target === handler_element) return; define_property(event2, "currentTarget", { configurable: true, get() { return current_target || owner_document; } }); var previous_reaction = active_reaction; var previous_effect = active_effect; set_active_reaction(null); set_active_effect(null); try { var throw_error; var other_errors = []; while (current_target !== null) { var parent_element = current_target.assignedSlot || current_target.parentNode || /** @type {any} */ current_target.host || null; try { var delegated = current_target["__" + event_name]; if (delegated !== void 0 && !/** @type {any} */ current_target.disabled) { if (is_array(delegated)) { var [fn, ...data] = delegated; fn.apply(current_target, [event2, ...data]); } else { delegated.call(current_target, event2); } } } catch (error) { if (throw_error) { other_errors.push(error); } else { throw_error = error; } } if (event2.cancelBubble || parent_element === handler_element || parent_element === null) { break; } current_target = parent_element; } if (throw_error) { for (let error of other_errors) { queueMicrotask(() => { throw error; }); } throw throw_error; } } finally { event2.__root = handler_element; delete event2.currentTarget; set_active_reaction(previous_reaction); set_active_effect(previous_effect); } } function create_fragment_from_html(html2) { var elem = document.createElement("template"); elem.innerHTML = html2; return elem.content; } function assign_nodes(start, end) { var effect2 = ( /** @type {Effect} */ active_effect ); if (effect2.nodes_start === null) { effect2.nodes_start = start; effect2.nodes_end = end; } } // @__NO_SIDE_EFFECTS__ function template(content, flags) { var is_fragment = (flags & TEMPLATE_FRAGMENT) !== 0; var use_import_node = (flags & TEMPLATE_USE_IMPORT_NODE) !== 0; var node; var has_start = !content.startsWith("<!>"); return () => { if (hydrating) { assign_nodes(hydrate_node, null); return hydrate_node; } if (node === undefined) { node = create_fragment_from_html(has_start ? content : "<!>" + content); if (!is_fragment) node = /** @type {Node} */ /* @__PURE__ */ get_first_child(node); } var clone2 = ( /** @type {TemplateNode} */ use_import_node ? document.importNode(node, true) : node.cloneNode(true) ); if (is_fragment) { var start = ( /** @type {TemplateNode} */ /* @__PURE__ */ get_first_child(clone2) ); var end = ( /** @type {TemplateNode} */ clone2.lastChild ); assign_nodes(start, end); } else { assign_nodes(clone2, clone2); } return clone2; }; } // @__NO_SIDE_EFFECTS__ function ns_template(content, flags, ns = "svg") { var has_start = !content.startsWith("<!>"); var wrapped = `<${ns}>${has_start ? content : "<!>" + content}</${ns}>`; var node; return () => { if (hydrating) { assign_nodes(hydrate_node, null); return hydrate_node; } if (!node) { var fragment = ( /** @type {DocumentFragment} */ create_fragment_from_html(wrapped) ); var root2 = ( /** @type {Element} */ /* @__PURE__ */ get_first_child(fragment) ); { node = /** @type {Element} */ /* @__PURE__ */ get_first_child(root2); } } var clone2 = ( /** @type {TemplateNode} */ node.cloneNode(true) ); { assign_nodes(clone2, clone2); } return clone2; }; } function text(value = "") { if (!hydrating) { var t2 = create_text(value + ""); assign_nodes(t2, t2); return t2; } var node = hydrate_node; if (node.nodeType !== 3) { node.before(node = create_text()); set_hydrate_node(node); } assign_nodes(node, node); return node; } function comment() { if (hydrating) { assign_nodes(hydrate_node, null); return hydrate_node; } var frag = document.createDocumentFragment(); var start = document.createComment(""); var anchor = create_text(); frag.append(start, anchor); assign_nodes(start, anchor); return frag; } function append(anchor, dom) { if (hydrating) { active_effect.nodes_end = hydrate_node; hydrate_next(); return; } if (anchor === null) { return; } anchor.before( /** @type {Node} */ dom ); } let should_intro = true; function set_text(text2, value) { var str = value == null ? "" : typeof value === "object" ? value + "" : value; if (str !== (text2.__t ?? (text2.__t = text2.nodeValue))) { text2.__t = str; text2.nodeValue = str == null ? "" : str + ""; } } function mount(component2, options) { return _mount(component2, options); } function hydrate(component2, options) { init_operations(); options.intro = options.intro ?? false; const target = options.target; const was_hydrating = hydrating; const previous_hydrate_node = hydrate_node; try { var anchor = ( /** @type {TemplateNode} */ /* @__PURE__ */ get_first_child(target) ); while (anchor && (anchor.nodeType !== 8 || /** @type {Comment} */ anchor.data !== HYDRATION_START)) { anchor = /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(anchor); } if (!anchor) { throw HYDRATION_ERROR; } set_hydrating(true); set_hydrate_node( /** @type {Comment} */ anchor ); hydrate_next(); const instance = _mount(component2, { ...options, anchor }); if (hydrate_node === null || hydrate_node.nodeType !== 8 || /** @type {Comment} */ hydrate_node.data !== HYDRATION_END) { hydration_mismatch(); throw HYDRATION_ERROR; } set_hydrating(false); return ( /** @type {Exports} */ instance ); } catch (error) { if (error === HYDRATION_ERROR) { if (options.recover === false) { hydration_failed(); } init_operations(); clear_text_content(target); set_hydrating(false); return mount(component2, options); } throw error; } finally { set_hydrating(was_hydrating); set_hydrate_node(previous_hydrate_node); } } const document_listeners = /* @__PURE__ */ new Map(); function _mount(Component, { target, anchor, props = {}, events, context, intro = true }) { init_operations(); var registered_events = /* @__PURE__ */ new Set(); var event_handle = (events2) => { for (var i = 0; i < events2.length; i++) { var event_name = events2[i]; if (registered_events.has(event_name)) continue; registered_events.add(event_name); var passive = is_passive_event(event_name); target.addEventListener(event_name, handle_event_propagation, { passive }); var n = document_listeners.get(event_name); if (n === undefined) { document.addEventListener(event_name, handle_event_propagation, { passive }); document_listeners.set(event_name, 1); } else { document_listeners.set(event_name, n + 1); } } }; event_handle(array_from(all_registered_events)); root_event_handles.add(event_handle); var component2 = undefined; var unmount2 = component_root(() => { var anchor_node = anchor ?? target.appendChild(create_text()); branch(() => { if (context) { push({}); var ctx = ( /** @type {ComponentContext} */ component_context ); ctx.c = context; } if (events) { props.$$events = events; } if (hydrating) { assign_nodes( /** @type {TemplateNode} */ anchor_node, null ); } should_intro = intro; component2 = Component(anchor_node, props) || {}; should_intro = true; if (hydrating) { active_effect.nodes_end = hydrate_node; } if (context) { pop(); } }); return () => { var _a2; for (var event_name of registered_events) { target.removeEventListener(event_name, handle_event_propagation); var n = ( /** @type {number} */ document_listeners.get(event_name) ); if (--n === 0) { document.removeEventListener(event_name, handle_event_propagation); document_listeners.delete(event_name); } else { document_listeners.set(event_name, n); } } root_event_handles.delete(event_handle); if (anchor_node !== anchor) { (_a2 = anchor_node.parentNode) == null ? undefined : _a2.removeChild(anchor_node); } }; }); mounted_components.set(component2, unmount2); return component2; } let mounted_components = /* @__PURE__ */ new WeakMap(); function unmount(component2, options) { const fn = mounted_components.get(component2); if (fn) { mounted_components.delete(component2); return fn(options); } return Promise.resolve(); } function if_block(node, fn, elseif = false) { if (hydrating) { hydrate_next(); } var anchor = node; var consequent_effect = null; var alternate_effect = null; var condition = UNINITIALIZED; var flags = elseif ? EFFECT_TRANSPARENT : 0; var has_branch = false; const set_branch = (fn2, flag = true) => { has_branch = true; update_branch(flag, fn2); }; const update_branch = (new_condition, fn2) => { if (condition === (condition = new_condition)) return; let mismatch = false; if (hydrating) { const is_else = ( /** @type {Comment} */ anchor.data === HYDRATION_START_ELSE ); if (!!condition === is_else) { anchor = remove_nodes(); set_hydrate_node(anchor); set_hydrating(false); mismatch = true; } } if (condition) { if (consequent_effect) { resume_effect(consequent_effect); } else if (fn2) { consequent_effect = branch(() => fn2(anchor)); } if (alternate_effect) { pause_effect(alternate_effect, () => { alternate_effect = null; }); } } else { if (alternate_effect) { resume_effect(alternate_effect); } else if (fn2) { alternate_effect = branch(() => fn2(anchor)); } if (consequent_effect) { pause_effect(consequent_effect, () => { consequent_effect = null; }); } } if (mismatch) { set_hydrating(true); } }; block(() => { has_branch = false; fn(set_branch); if (!has_branch) { update_branch(null, null); } }, flags); if (hydrating) { anchor = hydrate_node; } } function key_block(node, get_key, render_fn) { if (hydrating) { hydrate_next(); } var anchor = node; var key = UNINITIALIZED; var effect2; var changed = is_runes() ? not_equal : safe_not_equal; block(() => { if (changed(key, key = get_key())) { if (effect2) { pause_effect(effect2); } effect2 = branch(() => render_fn(anchor)); } }); if (hydrating) { anchor = hydrate_node; } } let current_each_item = null; function index(_, i) { return i; } function pause_effects(state2, items, controlled_anchor, items_map) { var transitions = []; var length = items.length; for (var i = 0; i < length; i++) { pause_children(items[i].e, transitions, true); } var is_controlled = length > 0 && transitions.length === 0 && controlled_anchor !== null; if (is_controlled) { var parent_node = ( /** @type {Element} */ /** @type {Element} */ controlled_anchor.parentNode ); clear_text_content(parent_node); parent_node.append( /** @type {Element} */ controlled_anchor ); items_map.clear(); link(state2, items[0].prev, items[length - 1].next); } run_out_transitions(transitions, () => { for (var i2 = 0; i2 < length; i2++) { var item = items[i2]; if (!is_controlled) { items_map.delete(item.k); link(state2, item.prev, item.next); } destroy_effect(item.e, !is_controlled); } }); } function each(node, flags, get_collection, get_key, render_fn, fallback_fn = null) { var anchor = node; var state2 = { flags, items: /* @__PURE__ */ new Map(), first: null }; var is_controlled = (flags & EACH_IS_CONTROLLED) !== 0; if (is_controlled) { var parent_node = ( /** @type {Element} */ node ); anchor = hydrating ? set_hydrate_node( /** @type {Comment | Text} */ /* @__PURE__ */ get_first_child(parent_node) ) : parent_node.appendChild(create_text()); } if (hydrating) { hydrate_next(); } var fallback = null; var was_empty = false; var each_array = /* @__PURE__ */ derived_safe_equal(() => { var collection = get_collection(); return is_array(collection) ? collection : collection == null ? [] : array_from(collection); }); block(() => { var array = get$1(each_array); var length = array.length; if (was_empty && length === 0) { return; } was_empty = length === 0; let mismatch = false; if (hydrating) { var is_else = ( /** @type {Comment} */ anchor.data === HYDRATION_START_ELSE ); if (is_else !== (length === 0)) { anchor = remove_nodes(); set_hydrate_node(anchor); set_hydrating(false); mismatch = true; } } if (hydrating) { var prev = null; var item; for (var i = 0; i < length; i++) { if (hydrate_node.nodeType === 8 && /** @type {Comment} */ hydrate_node.data === HYDRATION_END) { anchor = /** @type {Comment} */ hydrate_node; mismatch = true; set_hydrating(false); break; } var value = array[i]; var key = get_key(value, i); item = create_item( hydrate_node, state2, prev, null, value, key, i, render_fn, flags ); state2.items.set(key, item); prev = item; } if (length > 0) { set_hydrate_node(remove_nodes()); } } if (!hydrating) { var effect2 = ( /** @type {Effect} */ active_reaction ); reconcile( array, state2, anchor, render_fn, flags, (effect2.f & INERT) !== 0, get_key ); } if (fallback_fn !== null) { if (length === 0) { if (fallback) { resume_effect(fallback); } else { fallback = branch(() => fallback_fn(anchor)); } } else if (fallback !== null) { pause_effect(fallback, () => { fallback = null; }); } } if (mismatch) { set_hydrating(true); } get$1(each_array); }); if (hydrating) { anchor = hydrate_node; } } function reconcile(array, state2, anchor, render_fn, flags, is_inert, get_key, get_collection) { var _a2, _b2, _c2, _d2; var is_animated = (flags & EACH_IS_ANIMATED) !== 0; var should_update = (flags & (EACH_ITEM_REACTIVE | EACH_INDEX_REACTIVE)) !== 0; var length = array.length; var items = state2.items; var first = state2.first; var current = first; var seen; var prev = null; var to_animate; var matched = []; var stashed = []; var value; var key; var item; var i; if (is_animated) { for (i = 0; i < length; i += 1) { value = array[i]; key = get_key(value, i); item = items.get(key); if (item !== undefined) { (_a2 = item.a) == null ? undefined : _a2.measure(); (to_animate ?? (to_animate = /* @__PURE__ */ new Set())).add(item); } } } for (i = 0; i < length; i += 1) { value = array[i]; key = get_key(value, i); item = items.get(key); if (item === undefined) { var child_anchor = current ? ( /** @type {TemplateNode} */ current.e.nodes_start ) : anchor; prev = create_item( child_anchor, state2, prev, prev === null ? state2.first : prev.next, value, key, i, render_fn, flags ); items.set(key, prev); matched = []; stashed = []; current = prev.next; continue; } if (should_update) { update_item(item, value, i, flags); } if ((item.e.f & INERT) !== 0) { resume_effect(item.e); if (is_animated) { (_b2 = item.a) == null ? undefined : _b2.unfix(); (to_animate ?? (to_animate = /* @__PURE__ */ new Set())).delete(item); } } if (item !== current) { if (seen !== undefined && seen.has(item)) { if (matched.length < stashed.length) { var start = stashed[0]; var j; prev = start.prev; var a = matched[0]; var b = matched[matched.length - 1]; for (j = 0; j < matched.length; j += 1) { move(matched[j], start, anchor); } for (j = 0; j < stashed.length; j += 1) { seen.delete(stashed[j]); } link(state2, a.prev, b.next); link(state2, prev, a); link(state2, b, start); current = start; prev = b; i -= 1; matched = []; stashed = []; } else { seen.delete(item); move(item, current, anchor); link(state2, item.prev, item.next); link(state2, item, prev === null ? state2.first : prev.next); link(state2, prev, item); prev = item; } continue; } matched = []; stashed = []; while (current !== null && current.k !== key) { if (is_inert || (current.e.f & INERT) === 0) { (seen ?? (seen = /* @__PURE__ */ new Set())).add(current); } stashed.push(current); current = current.next; } if (current === null) { continue; } item = current; } matched.push(item); prev = item; current = item.next; } if (current !== null || seen !== undefined) { var to_destroy = seen === undefined ? [] : array_from(seen); while (current !== null) { if (is_inert || (current.e.f & INERT) === 0) { to_destroy.push(current); } current = current.next; } var destroy_length = to_destroy.length; if (destroy_length > 0) { var controlled_anchor = (flags & EACH_IS_CONTROLLED) !== 0 && length === 0 ? anchor : null; if (is_animated) { for (i = 0; i < destroy_length; i += 1) { (_c2 = to_destroy[i].a) == null ? undefined : _c2.measure(); } for (i = 0; i < destroy_length; i += 1) { (_d2 = to_destroy[i].a) == null ? undefined : _d2.fix(); } } pause_effects(state2, to_destroy, controlled_anchor, items); } } if (is_animated) { queue_micro_task(() => { var _a3; if (to_animate === undefined) return; for (item of to_animate) { (_a3 = item.a) == null ? undefined : _a3.apply(); } }); } active_effect.first = state2.first && state2.first.e; active_effect.last = prev && prev.e; } function update_item(item, value, index2, type) { if ((type & EACH_ITEM_REACTIVE) !== 0) { internal_set(item.v, value); } if ((type & EACH_INDEX_REACTIVE) !== 0) { internal_set( /** @type {Value<number>} */ item.i, index2 ); } else { item.i = index2; } } function create_item(anchor, state2, prev, next2, value, key, index2, render_fn, flags, get_collection) { var previous_each_item = current_each_item; var reactive = (flags & EACH_ITEM_REACTIVE) !== 0; var mutable = (flags & EACH_ITEM_IMMUTABLE) === 0; var v = reactive ? mutable ? /* @__PURE__ */ mutable_source(value) : source(value) : value; var i = (flags & EACH_INDEX_REACTIVE) === 0 ? index2 : source(index2); var item = { i, v, k: key, a: null, // @ts-expect-error e: null, prev, next: next2 }; current_each_item = item; try { item.e = branch(() => render_fn(anchor, v, i), hydrating); item.e.prev = prev && prev.e; item.e.next = next2 && next2.e; if (prev === null) { state2.first = item; } else { prev.next = item; prev.e.next = item.e; } if (next2 !== null) { next2.prev = item; next2.e.prev = item.e; } return item; } finally { current_each_item = previous_each_item; } } function move(item, next2, anchor) { var end = item.next ? ( /** @type {TemplateNode} */ item.next.e.nodes_start ) : anchor; var dest = next2 ? ( /** @type {TemplateNode} */ next2.e.nodes_start ) : anchor; var node = ( /** @type {TemplateNode} */ item.e.nodes_start ); while (node !== end) { var next_node = ( /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(node) ); dest.before(node); node = next_node; } } function link(state2, prev, next2) { if (prev === null) { state2.first = next2; } else { prev.next = next2; prev.e.next = next2 && next2.e; } if (next2 !== null) { next2.prev = prev; next2.e.prev = prev && prev.e; } } function html(node, get_value, svg, mathml, skip_warning) { var anchor = node; var value = ""; var effect2; block(() => { if (value === (value = get_value() ?? "")) { if (hydrating) { hydrate_next(); } return; } if (effect2 !== undefined) { destroy_effect(effect2); effect2 = undefined; } if (value === "") return; effect2 = branch(() => { if (hydrating) { hydrate_node.data; var next2 = hydrate_next(); var last = next2; while (next2 !== null && (next2.nodeType !== 8 || /** @type {Comment} */ next2.data !== "")) { last = next2; next2 = /** @type {TemplateNode} */ /* @__PURE__ */ get_next_sibling(next2); } if (next2 === null) { hydration_mismatch(); throw HYDRATION_ERROR; } assign_nodes(hydrate_node, last); anchor = set_hydrate_node(next2); return; } var html2 = value + ""; var node2 = create_fragment_from_html(html2); assign_nodes( /** @type {TemplateNode} */ /* @__PURE__ */ get_first_child(node2), /** @type {TemplateNode} */ node2.lastChild ); { anchor.before(node2); } }); }); } function slot(anchor, $$props, name, slot_props, fallback_fn) { var _a2; if (hydrating) { hydrate_next(); } var slot_fn = (_a2 = $$props.$$slots) == null ? undefined : _a2[name]; var is_interop = false; if (slot_fn === true) { slot_fn = $$props[name === "default" ? "children" : name]; is_interop = true; } if (slot_fn === undefined) { if (fallback_fn !== null) { fallback_fn(anchor); } } else { slot_fn(anchor, is_interop ? () => slot_props : slot_props); } } function sanitize_slots(props) { const sanitized = {}; if (props.children) sanitized.default = true; for (const key in props.$$slots) { sanitized[key] = true; } return sanitized; } function snippet(node, get_snippet, ...args) { var anchor = node; var snippet2 = noop; var snippet_effect; block(() => { if (snippet2 === (snippet2 = get_snippet())) return; if (snippet_effect) { destroy_effect(snippet_effect); snippet_effect = null; } snippet_effect = branch(() => ( /** @type {SnippetFn} */ snippet2(anchor, ...args) )); }, EFFECT_TRANSPARENT); if (hydrating) { anchor = hydrate_node; } } function component(node, get_component, render_fn) { if (hydrating) { hydrate_next(); } var anchor = node; var component2; var effect2; block(() => { if (component2 === (component2 = get_component())) return; if (effect2) { pause_effect(effect2); effect2 = null; } if (component2) { effect2 = branch(() => render_fn(anchor, component2)); } }, EFFECT_TRANSPARENT); if (hydrating) { anchor = hydrate_node; } } function action(dom, action2, get_value) { effect(() => { var payload = untrack(() => action2(dom, get_value == null ? void 0 : get_value()) || {}); if (get_value && (payload == null ? undefined : payload.update)) { var inited = false; var prev = ( /** @type {any} */ {} ); render_effect(() => { var value = get_value(); deep_read_state(value); if (inited && safe_not_equal(prev, value)) { prev = value; payload.update(value); } }); inited = true; } if (payload == null ? undefined : payload.destroy) { return () => ( /** @type {Function} */ payload.destroy() ); } }); } function r(e) { var t2, f, n = ""; if ("string" == typeof e || "number" == typeof e) n += e; else if ("object" == typeof e) if (Array.isArray(e)) { var o = e.length; for (t2 = 0; t2 < o; t2++) e[t2] && (f = r(e[t2])) && (n && (n += " "), n += f); } else for (f in e) e[f] && (n && (n += " "), n += f); return n; } function clsx$1() { for (var e, t2, f = 0, n = "", o = arguments.length; f < o; f++) (e = arguments[f]) && (t2 = r(e)) && (n && (n += " "), n += t2); return n; } function clsx(value) { if (typeof value === "object") { return clsx$1(value); } else { return value ?? ""; } } function remove_input_defaults(input) { if (!hydrating) return; var already_removed = false; var remove_defaults = () => { if (already_removed) return; already_removed = true; if (input.hasAttribute("value")) { var value = input.value; set_attribute(input, "value", null); input.value = value; } if (input.hasAttribute("checked")) { var checked = input.checked; set_attribute(input, "checked", null); input.checked = checked; } }; input.__on_r = remove_defaults; queue_idle_task(remove_defaults); add_form_reset_listener(); } function set_selected(element, selected) { if (selected) { if (!element.hasAttribute("selected")) { element.setAttribute("selected", ""); } } else { element.removeAttribute("selected"); } } function set_attribute(element, attribute, value, skip_warning) { var attributes = element.__attributes ?? (element.__attributes = {}); if (hydrating) { attributes[attribute] = element.getAttribute(attribute); if (attribute === "src" || attribute === "srcset" || attribute === "href" && element.nodeName === "LINK") { return; } } if (attributes[attribute] === (attributes[attribute] = value)) return; if (attribute === "style" && "__styles" in element) { element.__styles = {}; } if (attribute === "loading") { element[LOADING_ATTR_SYMBOL] = value; } if (value == null) { element.removeAttribute(attribute); } else if (typeof value !== "string" && get_setters(element).includes(attribute)) { element[attribute] = value; } else { element.setAttribute(attribute, value); } } function set_attributes(element, prev, next2, css_hash, preserve_attribute_case = false, is_custom_element = false, skip_warning = false) { var current = prev || {}; var is_option_element = element.tagName === "OPTION"; for (var key in prev) { if (!(key in next2)) { next2[key] = null; } } if (next2.class) { next2.class = clsx(next2.class); } var setters = get_setters(element); var attributes = ( /** @type {Record<string, unknown>} **/ element.__attributes ?? (element.__attributes = {}) ); for (const key2 in next2) { let value = next2[key2]; if (is_option_element && key2 === "value" && value == null) { element.value = element.__value = ""; current[key2] = value; continue; } var prev_value = current[key2]; if (value === prev_value) continue; current[key2] = value; var prefix = key2[0] + key2[1]; if (prefix === "$$") continue; if (prefix === "on") { const opts = {}; const event_handle_key = "$$" + key2; let event_name = key2.slice(2); var delegated = is_delegated(event_name); if (is_capture_event(event_name)) { event_name = event_name.slice(0, -7); opts.capture = true; } if (!delegated && prev_value) { if (value != null) continue; element.removeEventListener(event_name, current[event_handle_key], opts); current[event_handle_key] = null; } if (value != null) { if (!delegated) { let handle = function(evt) { current[key2].call(this, evt); }; current[event_handle_key] = create_event(event_name, element, handle, opts); } else { element[`__${event_name}`] = value; delegate([event_name]); } } else if (delegated) { element[`__${event_name}`] = undefined; } } else if (key2 === "style" && value != null) { element.style.cssText = value + ""; } else if (key2 === "autofocus") { autofocus( /** @type {HTMLElement} */ element, Boolean(value) ); } else if (key2 === "__value" || key2 === "value" && value != null) { element.value = element[key2] = element.__value = value; } else if (key2 === "selected" && is_option_element) { set_selected( /** @type {HTMLOptionElement} */ element, value ); } else { var name = key2; if (!preserve_attribute_case) { name = normalize_attribute(name); } var is_default = name === "defaultValue" || name === "defaultChecked"; if (value == null && !is_custom_element && !is_default) { attributes[key2] = null; if (name === "value" || name === "checked") { let input = ( /** @type {HTMLInputElement} */ element ); if (name === "value") { let prev2 = input.defaultValue; input.removeAttribute(name); input.defaultValue = prev2; } else { let prev2 = input.defaultChecked; input.removeAttribute(name); input.defaultChecked = prev2; } } else { element.removeAttribute(key2); } } else if (is_default || setters.includes(name) && (is_custom_element || typeof value !== "string")) { element[name] = value; } else if (typeof value !== "function") { if (hydrating && (name === "src" || name === "href" || name === "srcset")) ; else { set_attribute(element, name, value); } } } if (key2 === "style" && "__styles" in element) { element.__styles = {}; } } return current; } var setters_cache = /* @__PURE__ */ new Map(); function get_setters(element) { var setters = setters_cache.get(element.nodeName); if (setters) return setters; setters_cache.set(element.nodeName, setters = []); var descriptors; var proto = element; var element_proto = Element.prototype; while (element_proto !== proto) { descriptors = get_descriptors(proto); for (var key in descriptors) { if (descriptors[key].set) { setters.push(key); } } proto = get_prototype_of(proto); } return setters; } function set_svg_class(dom, value, hash) { var prev_class_name = dom.__className; var next_class_name = to_class(value); if (hydrating && dom.getAttribute("class") === next_class_name) { dom.__className = next_class_name; } else if (prev_class_name !== next_class_name || hydrating && dom.getAttribute("class") !== next_class_name) { if (next_class_name === "") { dom.removeAttribute("class"); } else { dom.setAttribute("class", next_class_name); } dom.__className = next_class_name; } } function set_class(dom, value, hash) { var prev_class_name = dom.__className; var next_class_name = to_class(value); if (hydrating && dom.className === next_class_name) { dom.__className = next_class_name; } else if (prev_class_name !== next_class_name || hydrating && dom.className !== next_class_name) { if (value == null && true) { dom.removeAttribute("class"); } else { dom.className = next_class_name; } dom.__className = next_class_name; } } function to_class(value, hash) { return (value == null ? "" : value) + ""; } function toggle_class(dom, class_name, value) { if (value) { if (dom.classList.contains(class_name)) return; dom.classList.add(class_name); } else { if (!dom.classList.contains(class_name)) return; dom.classList.remove(class_name); } } function set_style(dom, key, value, important) { var styles = dom.__styles ?? (dom.__styles = {}); if (styles[key] === value) { return; } styles[key] = value; if (value == null) { dom.style.removeProperty(key); } else { dom.style.setProperty(key, value, ""); } } const now = () => performance.now(); const raf = { // don't access requestAnimationFrame eagerly outside method // this allows basic testing of user code without JSDOM // bunder will eval and remove ternary when the user's app is built tick: ( /** @param {any} _ */ (_) => requestAnimationFrame(_) ), now: () => now(), tasks: /* @__PURE__ */ new Set() }; function run_tasks() { const now2 = raf.now(); raf.tasks.forEach((task) => { if (!task.c(now2)) { raf.tasks.delete(task); task.f(); } }); if (raf.tasks.size !== 0) { raf.tick(run_tasks); } } function loop(callback) { let task; if (raf.tasks.size === 0) { raf.tick(run_tasks); } return { promise: new Promise((fulfill) => { raf.tasks.add(task = { c: callback, f: fulfill }); }), abort() { raf.tasks.delete(task); } }; } function dispatch_event(element, type) { element.dispatchEvent(new CustomEvent(type)); } function css_property_to_camelcase(style) { if (style === "float") return "cssFloat"; if (style === "offset") return "cssOffset"; if (style.startsWith("--")) return style; const parts = style.split("-"); if (parts.length === 1) return parts[0]; return parts[0] + parts.slice(1).map( /** @param {any} word */ (word) => word[0].toUpperCase() + word.slice(1) ).join(""); } function css_to_keyframe(css) { const keyframe = {}; const parts = css.split(";"); for (const part of parts) { const [property, value] = part.split(":"); if (!property || value === undefined) break; const formatted_property = css_property_to_camelcase(property.trim()); keyframe[formatted_property] = value.trim(); } return keyframe; } const linear$1 = (t2) => t2; function animation(element, get_fn, get_params) { var item = ( /** @type {EachItem} */ current_each_item ); var from; var to; var animation2; var original_styles = null; item.a ?? (item.a = { element, measure() { from = this.element.getBoundingClientRect(); }, apply() { animation2 == null ? undefined : animation2.abort(); to = this.element.getBoundingClientRect(); if (from.left !== to.left || from.right !== to.right || from.top !== to.top || from.bottom !== to.bottom) { const options = get_fn()(this.element, { from, to }, get_params == null ? undefined : get_params()); animation2 = animate(this.element, options, undefined, 1, () => { animation2 == null ? undefined : animation2.abort(); animation2 = undefined; }); } }, fix() { if (element.getAnimations().length) return; var { position, width, height } = getComputedStyle(element); if (position !== "absolute" && position !== "fixed") { var style = ( /** @type {HTMLElement | SVGElement} */ element.style ); original_styles = { position: style.position, width: style.width, height: style.height, transform: style.transform }; style.position = "absolute"; style.width = width; style.height = height; var to2 = element.getBoundingClientRect(); if (from.left !== to2.left || from.top !== to2.top) { var transform = `translate(${from.left - to2.left}px, ${from.top - to2.top}px)`; style.transform = style.transform ? `${style.transform} ${transform}` : transform; } } }, unfix() { if (original_styles) { var style = ( /** @type {HTMLElement | SVGElement} */ element.style ); style.position = original_styles.position; style.width = original_styles.width; style.height = original_styles.height; style.transform = original_styles.transform; } } }); item.a.element = element; } function transition(flags, element, get_fn, get_params) { var is_intro = (flags & TRANSITION_IN) !== 0; var is_outro = (flags & TRANSITION_OUT) !== 0; var is_both = is_intro && is_outro; var is_global = (flags & TRANSITION_GLOBAL) !== 0; var direction = is_both ? "both" : is_intro ? "in" : "out"; var current_options; var inert = element.inert; var overflow = element.style.overflow; var intro; var outro; function get_options() { var previous_reaction = active_reaction; var previous_effect = active_effect; set_active_reaction(null); set_active_effect(null); try { return current_options ?? (current_options = get_fn()(element, (get_params == null ? void 0 : get_params()) ?? /** @type {P} */ {}, { direction })); } finally { set_active_reaction(previous_reaction); set_active_effect(previous_effect); } } var transition2 = { is_global, in() { var _a2; element.inert = inert; if (!is_intro) { outro == null ? undefined : outro.abort(); (_a2 = outro == null ? undefined : outro.reset) == null ? undefined : _a2.call(outro); return; } if (!is_outro) { intro == null ? undefined : intro.abort(); } dispatch_event(element, "introstart"); intro = animate(element, get_options(), outro, 1, () => { dispatch_event(element, "introend"); intro == null ? undefined : intro.abort(); intro = current_options = undefined; element.style.overflow = overflow; }); }, out(fn) { if (!is_outro) { fn == null ? undefined : fn(); current_options = undefined; return; } element.inert = true; dispatch_event(element, "outrostart"); outro = animate(element, get_options(), intro, 0, () => { dispatch_event(element, "outroend"); fn == null ? undefined : fn(); }); }, stop: () => { intro == null ? undefined : intro.abort(); outro == null ? undefined : outro.abort(); } }; var e = ( /** @type {Effect} */ active_effect ); (e.transitions ?? (e.transitions = [])).push(transition2); if (is_intro && should_intro) { var run2 = is_global; if (!run2) { var block2 = ( /** @type {Effect | null} */ e.parent ); while (block2 && (block2.f & EFFECT_TRANSPARENT) !== 0) { while (block2 = block2.parent) { if ((block2.f & BLOCK_EFFECT) !== 0) break; } } run2 = !block2 || (block2.f & EFFECT_RAN) !== 0; } if (run2) { effect(() => { untrack(() => transition2.in()); }); } } } function animate(element, options, counterpart, t2, on_finish) { var is_intro = t2 === 1; if (is_function(options)) { var a; var aborted = false; queue_micro_task(() => { if (aborted) return; var o = options({ direction: is_intro ? "in" : "out" }); a = animate(element, o, counterpart, t2, on_finish); }); return { abort: () => { aborted = true; a == null ? undefined : a.abort(); }, deactivate: () => a.deactivate(), reset: () => a.reset(), t: () => a.t() }; } counterpart == null ? undefined : counterpart.deactivate(); if (!(options == null ? undefined : options.duration)) { on_finish(); return { abort: noop, deactivate: noop, reset: noop, t: () => t2 }; } const { delay = 0, css, tick: tick2, easing = linear$1 } = options; var keyframes = []; if (is_intro && counterpart === undefined) { if (tick2) { tick2(0, 1); } if (css) { var styles = css_to_keyframe(css(0, 1)); keyframes.push(styles, styles); } } var get_t = () => 1 - t2; var animation2 = element.animate(keyframes, { duration: delay }); animation2.onfinish = () => { var t1 = (counterpart == null ? undefined : counterpart.t()) ?? 1 - t2; counterpart == null ? undefined : counterpart.abort(); var delta = t2 - t1; var duration = ( /** @type {number} */ options.duration * Math.abs(delta) ); var keyframes2 = []; if (duration > 0) { var needs_overflow_hidden = false; if (css) { var n = Math.ceil(duration / (1e3 / 60)); for (var i = 0; i <= n; i += 1) { var t3 = t1 + delta * easing(i / n); var styles2 = css_to_keyframe(css(t3, 1 - t3)); keyframes2.push(styles2); needs_overflow_hidden || (needs_overflow_hidden = styles2.overflow === "hidden"); } } if (needs_overflow_hidden) { element.style.overflow = "hidden"; } get_t = () => { var time = ( /** @type {number} */ /** @type {globalThis.Animation} */ animation2.currentTime ); return t1 + delta * easing(time / duration); }; if (tick2) { loop(() => { if (animation2.playState !== "running") return false; var t4 = get_t(); tick2(t4, 1 - t4); return true; }); } } animation2 = element.animate(keyframes2, { duration, fill: "forwards" }); animation2.onfinish = () => { get_t = () => t2; tick2 == null ? undefined : tick2(t2, 1 - t2); on_finish(); }; }; return { abort: () => { if (animation2) { animation2.cancel(); animation2.effect = null; animation2.onfinish = noop; } }, deactivate: () => { on_finish = noop; }, reset: () => { if (t2 === 0) { tick2 == null ? undefined : tick2(1, 0); } }, t: () => get_t() }; } function bind_value(input, get2, set2 = get2) { var runes = is_runes(); listen_to_event_and_reset_event(input, "input", (is_reset) => { var value = is_reset ? input.defaultValue : input.value; value = is_numberlike_input(input) ? to_number(value) : value; set2(value); if (runes && value !== (value = get2())) { var start = input.selectionStart; var end = input.selectionEnd; input.value = value ?? ""; if (end !== null) { input.selectionStart = start; input.selectionEnd = Math.min(end, input.value.length); } } }); if ( // If we are hydrating and the value has since changed, // then use the updated value from the input instead. hydrating && input.defaultValue !== input.value || // If defaultValue is set, then value == defaultValue // TODO Svelte 6: remove input.value check and set to empty string? untrack(get2) == null && input.value ) { set2(is_numberlike_input(input) ? to_number(input.value) : input.value); } render_effect(() => { var value = get2(); if (is_numberlike_input(input) && value === to_number(input.value)) { return; } if (input.type === "date" && !value && !input.value) { return; } if (value !== input.value) { input.value = value ?? ""; } }); } const pending = /* @__PURE__ */ new Set(); function bind_group(inputs, group_index, input, get2, set2 = get2) { var is_checkbox = input.getAttribute("type") === "checkbox"; var binding_group = inputs; let hydration_mismatch2 = false; if (group_index !== null) { for (var index2 of group_index) { binding_group = binding_group[index2] ?? (binding_group[index2] = []); } } binding_group.push(input); listen_to_event_and_reset_event( input, "change", () => { var value = input.__value; if (is_checkbox) { value = get_binding_group_value(binding_group, value, input.checked); } set2(value); }, // TODO better default value handling () => set2(is_checkbox ? [] : null) ); render_effect(() => { var value = get2(); if (hydrating && input.defaultChecked !== input.checked) { hydration_mismatch2 = true; return; } if (is_checkbox) { value = value || []; input.checked = value.includes(input.__value); } else { input.checked = is(input.__value, value); } }); teardown(() => { var index3 = binding_group.indexOf(input); if (index3 !== -1) { binding_group.splice(index3, 1); } }); if (!pending.has(binding_group)) { pending.add(binding_group); queue_micro_task(() => { binding_group.sort((a, b) => a.compareDocumentPosition(b) === 4 ? -1 : 1); pending.delete(binding_group); }); } queue_micro_task(() => { if (hydration_mismatch2) { var value; if (is_checkbox) { value = get_binding_group_value(binding_group, value, input.checked); } else { var hydration_input = binding_group.find((input2) => input2.checked); value = hydration_input == null ? undefined : hydration_input.__value; } set2(value); } }); } function bind_checked(input, get2, set2 = get2) { listen_to_event_and_reset_event(input, "change", (is_reset) => { var value = is_reset ? input.defaultChecked : input.checked; set2(value); }); if ( // If we are hydrating and the value has since changed, // then use the update value from the input instead. hydrating && input.defaultChecked !== input.checked || // If defaultChecked is set, then checked == defaultChecked untrack(get2) == null ) { set2(input.checked); } render_effect(() => { var value = get2(); input.checked = Boolean(value); }); } function get_binding_group_value(group, __value, checked) { var value = /* @__PURE__ */ new Set(); for (var i = 0; i < group.length; i += 1) { if (group[i].checked) { value.add(group[i].__value); } } if (!checked) { value.delete(__value); } return Array.from(value); } function is_numberlike_input(input) { var type = input.type; return type === "number" || type === "range"; } function to_number(value) { return value === "" ? null : +value; } function bind_files(input, get2, set2 = get2) { listen_to_event_and_reset_event(input, "change", () => { set2(input.files); }); render_effect(() => { input.files = get2(); }); } function bind_prop(props, prop2, value) { var desc = get_descriptor(props, prop2); if (desc && desc.set) { props[prop2] = value; teardown(() => { props[prop2] = null; }); } } function select_option(select, value, mounting) { if (select.multiple) { return select_options(select, value); } for (var option of select.options) { var option_value = get_option_value(option); if (is(option_value, value)) { option.selected = true; return; } } if (!mounting || value !== undefined) { select.selectedIndex = -1; } } function init_select(select, get_value) { effect(() => { var observer2 = new MutationObserver(() => { var value = select.__value; select_option(select, value); }); observer2.observe(select, { // Listen to option element changes childList: true, subtree: true, // because of <optgroup> // Listen to option element value attribute changes // (doesn't get notified of select value changes, // because that property is not reflected as an attribute) attributes: true, attributeFilter: ["value"] }); return () => { observer2.disconnect(); }; }); } function bind_select_value(select, get2, set2 = get2) { var mounting = true; listen_to_event_and_reset_event(select, "change", (is_reset) => { var query = is_reset ? "[selected]" : ":checked"; var value; if (select.multiple) { value = [].map.call(select.querySelectorAll(query), get_option_value); } else { var selected_option = select.querySelector(query) ?? // will fall back to first non-disabled option if no option is selected select.querySelector("option:not([disabled])"); value = selected_option && get_option_value(selected_option); } set2(value); }); effect(() => { var value = get2(); select_option(select, value, mounting); if (mounting && value === undefined) { var selected_option = select.querySelector(":checked"); if (selected_option !== null) { value = get_option_value(selected_option); set2(value); } } select.__value = value; mounting = false; }); init_select(select); } function select_options(select, value) { for (var option of select.options) { option.selected = ~value.indexOf(get_option_value(option)); } } function get_option_value(option) { if ("__value" in option) { return option.__value; } else { return option.value; } } function is_bound_this(bound_value, element_or_component) { return bound_value === element_or_component || (bound_value == null ? undefined : bound_value[STATE_SYMBOL]) === element_or_component; } function bind_this(element_or_component = {}, update2, get_value, get_parts) { effect(() => { var old_parts; var parts; render_effect(() => { old_parts = parts; parts = []; untrack(() => { if (element_or_component !== get_value(...parts)) { update2(element_or_component, ...parts); if (old_parts && is_bound_this(get_value(...old_parts), element_or_component)) { update2(null, ...old_parts); } } }); }); return () => { queue_micro_task(() => { if (parts && is_bound_this(get_value(...parts), element_or_component)) { update2(null, ...parts); } }); }; }); return element_or_component; } function bind_window_size(type, set2) { listen(window, ["resize"], () => without_reactive_context(() => set2(window[type]))); } function init(immutable = false) { const context = ( /** @type {ComponentContextLegacy} */ component_context ); const callbacks = context.l.u; if (!callbacks) return; let props = () => deep_read_state(context.s); if (immutable) { let version = 0; let prev = ( /** @type {Record<string, any>} */ {} ); const d = /* @__PURE__ */ derived(() => { let changed = false; const props2 = context.s; for (const key in props2) { if (props2[key] !== prev[key]) { prev[key] = props2[key]; changed = true; } } if (changed) version++; return version; }); props = () => get$1(d); } if (callbacks.b.length) { user_pre_effect(() => { observe_all(context, props); run_all(callbacks.b); }); } user_effect(() => { const fns = untrack(() => callbacks.m.map(run)); return () => { for (const fn of fns) { if (typeof fn === "function") { fn(); } } }; }); if (callbacks.a.length) { user_effect(() => { observe_all(context, props); run_all(callbacks.a); }); } } function observe_all(context, props) { if (context.l.s) { for (const signal of context.l.s) get$1(signal); } props(); } function bubble_event($$props, event2) { var _a2; var events = ( /** @type {Record<string, Function[] | Function>} */ (_a2 = $$props.$$events) == null ? undefined : _a2[event2.type] ); var callbacks = is_array(events) ? events.slice() : events == null ? [] : [events]; for (var fn of callbacks) { fn.call(this, event2); } } let is_store_binding = false; let IS_UNMOUNTED = Symbol(); function store_get(store, store_name, stores2) { const entry = stores2[store_name] ?? (stores2[store_name] = { store: null, source: /* @__PURE__ */ mutable_source(undefined), unsubscribe: noop }); if (entry.store !== store && !(IS_UNMOUNTED in stores2)) { entry.unsubscribe(); entry.store = store ?? null; if (store == null) { entry.source.v = undefined; entry.unsubscribe = noop; } else { var is_synchronous_callback = true; entry.unsubscribe = subscribe_to_store(store, (v) => { if (is_synchronous_callback) { entry.source.v = v; } else { set(entry.source, v); } }); is_synchronous_callback = false; } } if (store && IS_UNMOUNTED in stores2) { return get(store); } return get$1(entry.source); } function store_set(store, value) { store.set(value); return value; } function setup_stores() { const stores2 = {}; function cleanup() { teardown(() => { for (var store_name in stores2) { const ref = stores2[store_name]; ref.unsubscribe(); } define_property(stores2, IS_UNMOUNTED, { enumerable: false, value: true }); }); } return [stores2, cleanup]; } function mark_store_binding() { is_store_binding = true; } function capture_store_binding(fn) { var previous_is_store_binding = is_store_binding; try { is_store_binding = false; return [fn(), is_store_binding]; } finally { is_store_binding = previous_is_store_binding; } } const legacy_rest_props_handler = { get(target, key) { if (target.exclude.includes(key)) return; get$1(target.version); return key in target.special ? target.special[key]() : target.props[key]; }, set(target, key, value) { if (!(key in target.special)) { target.special[key] = prop( { get [key]() { return target.props[key]; } }, /** @type {string} */ key, PROPS_IS_UPDATED ); } target.special[key](value); update(target.version); return true; }, getOwnPropertyDescriptor(target, key) { if (target.exclude.includes(key)) return; if (key in target.props) { return { enumerable: true, configurable: true, value: target.props[key] }; } }, deleteProperty(target, key) { if (target.exclude.includes(key)) return true; target.exclude.push(key); update(target.version); return true; }, has(target, key) { if (target.exclude.includes(key)) return false; return key in target.props; }, ownKeys(target) { return Reflect.ownKeys(target.props).filter((key) => !target.exclude.includes(key)); } }; function legacy_rest_props(props, exclude) { return new Proxy({ props, exclude, special: {}, version: source(0) }, legacy_rest_props_handler); } const spread_props_handler = { get(target, key) { let i = target.props.length; while (i--) { let p = target.props[i]; if (is_function(p)) p = p(); if (typeof p === "object" && p !== null && key in p) return p[key]; } }, set(target, key, value) { let i = target.props.length; while (i--) { let p = target.props[i]; if (is_function(p)) p = p(); const desc = get_descriptor(p, key); if (desc && desc.set) { desc.set(value); return true; } } return false; }, getOwnPropertyDescriptor(target, key) { let i = target.props.length; while (i--) { let p = target.props[i]; if (is_function(p)) p = p(); if (typeof p === "object" && p !== null && key in p) { const descriptor = get_descriptor(p, key); if (descriptor && !descriptor.configurable) { descriptor.configurable = true; } return descriptor; } } }, has(target, key) { if (key === STATE_SYMBOL || key === LEGACY_PROPS) return false; for (let p of target.props) { if (is_function(p)) p = p(); if (p != null && key in p) return true; } return false; }, ownKeys(target) { const keys = []; for (let p of target.props) { if (is_function(p)) p = p(); for (const key in p) { if (!keys.includes(key)) keys.push(key); } } return keys; } }; function spread_props(...props) { return new Proxy({ props }, spread_props_handler); } function with_parent_branch(fn) { var effect2 = active_effect; var previous_effect = active_effect; while (effect2 !== null && (effect2.f & (BRANCH_EFFECT | ROOT_EFFECT)) === 0) { effect2 = effect2.parent; } try { set_active_effect(effect2); return fn(); } finally { set_active_effect(previous_effect); } } function prop(props, key, flags, fallback) { var _a2; var immutable = (flags & PROPS_IS_IMMUTABLE) !== 0; var runes = !legacy_mode_flag || (flags & PROPS_IS_RUNES) !== 0; var bindable = (flags & PROPS_IS_BINDABLE) !== 0; var lazy = (flags & PROPS_IS_LAZY_INITIAL) !== 0; var is_store_sub = false; var prop_value; if (bindable) { [prop_value, is_store_sub] = capture_store_binding(() => ( /** @type {V} */ props[key] )); } else { prop_value = /** @type {V} */ props[key]; } var is_entry_props = STATE_SYMBOL in props || LEGACY_PROPS in props; var setter = bindable && (((_a2 = get_descriptor(props, key)) == null ? undefined : _a2.set) ?? (is_entry_props && key in props && ((v) => props[key] = v))) || undefined; var fallback_value = ( /** @type {V} */ fallback ); var fallback_dirty = true; var fallback_used = false; var get_fallback = () => { fallback_used = true; if (fallback_dirty) { fallback_dirty = false; if (lazy) { fallback_value = untrack( /** @type {() => V} */ fallback ); } else { fallback_value = /** @type {V} */ fallback; } } return fallback_value; }; if (prop_value === undefined && fallback !== undefined) { if (setter && runes) { props_invalid_value(); } prop_value = get_fallback(); if (setter) setter(prop_value); } var getter; if (runes) { getter = () => { var value = ( /** @type {V} */ props[key] ); if (value === undefined) return get_fallback(); fallback_dirty = true; fallback_used = false; return value; }; } else { var derived_getter = with_parent_branch( () => (immutable ? derived : derived_safe_equal)(() => ( /** @type {V} */ props[key] )) ); derived_getter.f |= LEGACY_DERIVED_PROP; getter = () => { var value = get$1(derived_getter); if (value !== undefined) fallback_value = /** @type {V} */ undefined; return value === undefined ? fallback_value : value; }; } if ((flags & PROPS_IS_UPDATED) === 0) { return getter; } if (setter) { var legacy_parent = props.$$legacy; return function(value, mutation) { if (arguments.length > 0) { if (!runes || !mutation || legacy_parent || is_store_sub) { setter(mutation ? getter() : value); } return value; } else { return getter(); } }; } var from_child = false; var was_from_child = false; var inner_current_value = /* @__PURE__ */ mutable_source(prop_value); var current_value = with_parent_branch( () => /* @__PURE__ */ derived(() => { var parent_value = getter(); var child_value = get$1(inner_current_value); if (from_child) { from_child = false; was_from_child = true; return child_value; } was_from_child = false; return inner_current_value.v = parent_value; }) ); if (!immutable) current_value.equals = safe_equals; return function(value, mutation) { if (captured_signals !== null) { from_child = was_from_child; getter(); get$1(inner_current_value); } if (arguments.length > 0) { const new_value = mutation ? get$1(current_value) : runes && bindable ? proxy(value) : value; if (!current_value.equals(new_value)) { from_child = true; set(inner_current_value, new_value); if (fallback_used && fallback_value !== undefined) { fallback_value = new_value; } untrack(() => get$1(current_value)); } return value; } return get$1(current_value); }; } function createClassComponent(options) { return new Svelte4Component(options); } class Svelte4Component { /** * @param {ComponentConstructorOptions & { * component: any; * }} options */ constructor(options) { /** @type {any} */ __privateAdd(this, _events); /** @type {Record<string, any>} */ __privateAdd(this, _instance); var _a2; var sources = /* @__PURE__ */ new Map(); var add_source = (key, value) => { var s = /* @__PURE__ */ mutable_source(value); sources.set(key, s); return s; }; const props = new Proxy( { ...options.props || {}, $$events: {} }, { get(target, prop2) { return get$1(sources.get(prop2) ?? add_source(prop2, Reflect.get(target, prop2))); }, has(target, prop2) { if (prop2 === LEGACY_PROPS) return true; get$1(sources.get(prop2) ?? add_source(prop2, Reflect.get(target, prop2))); return Reflect.has(target, prop2); }, set(target, prop2, value) { set(sources.get(prop2) ?? add_source(prop2, value), value); return Reflect.set(target, prop2, value); } } ); __privateSet(this, _instance, (options.hydrate ? hydrate : mount)(options.component, { target: options.target, anchor: options.anchor, props, context: options.context, intro: options.intro ?? false, recover: options.recover })); if (!((_a2 = options == null ? undefined : options.props) == null ? undefined : _a2.$$host) || options.sync === false) { flush_sync(); } __privateSet(this, _events, props.$$events); for (const key of Object.keys(__privateGet(this, _instance))) { if (key === "$set" || key === "$destroy" || key === "$on") continue; define_property(this, key, { get() { return __privateGet(this, _instance)[key]; }, /** @param {any} value */ set(value) { __privateGet(this, _instance)[key] = value; }, enumerable: true }); } __privateGet(this, _instance).$set = /** @param {Record<string, any>} next */ (next2) => { Object.assign(props, next2); }; __privateGet(this, _instance).$destroy = () => { unmount(__privateGet(this, _instance)); }; } /** @param {Record<string, any>} props */ $set(props) { __privateGet(this, _instance).$set(props); } /** * @param {string} event * @param {(...args: any[]) => any} callback * @returns {any} */ $on(event2, callback) { __privateGet(this, _events)[event2] = __privateGet(this, _events)[event2] || []; const cb = (...args) => callback.call(this, ...args); __privateGet(this, _events)[event2].push(cb); return () => { __privateGet(this, _events)[event2] = __privateGet(this, _events)[event2].filter( /** @param {any} fn */ (fn) => fn !== cb ); }; } $destroy() { __privateGet(this, _instance).$destroy(); } } _events = new WeakMap(); _instance = new WeakMap(); let SvelteElement; if (typeof HTMLElement === "function") { SvelteElement = class extends HTMLElement { /** * @param {*} $$componentCtor * @param {*} $$slots * @param {*} use_shadow_dom */ constructor($$componentCtor, $$slots, use_shadow_dom) { super(); /** The Svelte component constructor */ __publicField(this, "$$ctor"); /** Slots */ __publicField(this, "$$s"); /** @type {any} The Svelte component instance */ __publicField(this, "$$c"); /** Whether or not the custom element is connected */ __publicField(this, "$$cn", false); /** @type {Record<string, any>} Component props data */ __publicField(this, "$$d", {}); /** `true` if currently in the process of reflecting component props back to attributes */ __publicField(this, "$$r", false); /** @type {Record<string, CustomElementPropDefinition>} Props definition (name, reflected, type etc) */ __publicField(this, "$$p_d", {}); /** @type {Record<string, EventListenerOrEventListenerObject[]>} Event listeners */ __publicField(this, "$$l", {}); /** @type {Map<EventListenerOrEventListenerObject, Function>} Event listener unsubscribe functions */ __publicField(this, "$$l_u", /* @__PURE__ */ new Map()); /** @type {any} The managed render effect for reflecting attributes */ __publicField(this, "$$me"); this.$$ctor = $$componentCtor; this.$$s = $$slots; if (use_shadow_dom) { this.attachShadow({ mode: "open" }); } } /** * @param {string} type * @param {EventListenerOrEventListenerObject} listener * @param {boolean | AddEventListenerOptions} [options] */ addEventListener(type, listener, options) { this.$$l[type] = this.$$l[type] || []; this.$$l[type].push(listener); if (this.$$c) { const unsub = this.$$c.$on(type, listener); this.$$l_u.set(listener, unsub); } super.addEventListener(type, listener, options); } /** * @param {string} type * @param {EventListenerOrEventListenerObject} listener * @param {boolean | AddEventListenerOptions} [options] */ removeEventListener(type, listener, options) { super.removeEventListener(type, listener, options); if (this.$$c) { const unsub = this.$$l_u.get(listener); if (unsub) { unsub(); this.$$l_u.delete(listener); } } } async connectedCallback() { this.$$cn = true; if (!this.$$c) { let create_slot = function(name) { return (anchor) => { const slot2 = document.createElement("slot"); if (name !== "default") slot2.name = name; append(anchor, slot2); }; }; await Promise.resolve(); if (!this.$$cn || this.$$c) { return; } const $$slots = {}; const existing_slots = get_custom_elements_slots(this); for (const name of this.$$s) { if (name in existing_slots) { if (name === "default" && !this.$$d.children) { this.$$d.children = create_slot(name); $$slots.default = true; } else { $$slots[name] = create_slot(name); } } } for (const attribute of this.attributes) { const name = this.$$g_p(attribute.name); if (!(name in this.$$d)) { this.$$d[name] = get_custom_element_value(name, attribute.value, this.$$p_d, "toProp"); } } for (const key in this.$$p_d) { if (!(key in this.$$d) && this[key] !== undefined) { this.$$d[key] = this[key]; delete this[key]; } } this.$$c = createClassComponent({ component: this.$$ctor, target: this.shadowRoot || this, props: { ...this.$$d, $$slots, $$host: this } }); this.$$me = effect_root(() => { render_effect(() => { var _a2; this.$$r = true; for (const key of object_keys(this.$$c)) { if (!((_a2 = this.$$p_d[key]) == null ? undefined : _a2.reflect)) continue; this.$$d[key] = this.$$c[key]; const attribute_value = get_custom_element_value( key, this.$$d[key], this.$$p_d, "toAttribute" ); if (attribute_value == null) { this.removeAttribute(this.$$p_d[key].attribute || key); } else { this.setAttribute(this.$$p_d[key].attribute || key, attribute_value); } } this.$$r = false; }); }); for (const type in this.$$l) { for (const listener of this.$$l[type]) { const unsub = this.$$c.$on(type, listener); this.$$l_u.set(listener, unsub); } } this.$$l = {}; } } // We don't need this when working within Svelte code, but for compatibility of people using this outside of Svelte // and setting attributes through setAttribute etc, this is helpful /** * @param {string} attr * @param {string} _oldValue * @param {string} newValue */ attributeChangedCallback(attr, _oldValue, newValue) { var _a2; if (this.$$r) return; attr = this.$$g_p(attr); this.$$d[attr] = get_custom_element_value(attr, newValue, this.$$p_d, "toProp"); (_a2 = this.$$c) == null ? undefined : _a2.$set({ [attr]: this.$$d[attr] }); } disconnectedCallback() { this.$$cn = false; Promise.resolve().then(() => { if (!this.$$cn && this.$$c) { this.$$c.$destroy(); this.$$me(); this.$$c = undefined; } }); } /** * @param {string} attribute_name */ $$g_p(attribute_name) { return object_keys(this.$$p_d).find( (key) => this.$$p_d[key].attribute === attribute_name || !this.$$p_d[key].attribute && key.toLowerCase() === attribute_name ) || attribute_name; } }; } function get_custom_element_value(prop2, value, props_definition, transform) { var _a2; const type = (_a2 = props_definition[prop2]) == null ? undefined : _a2.type; value = type === "Boolean" && typeof value !== "boolean" ? value != null : value; if (!transform || !props_definition[prop2]) { return value; } else if (transform === "toAttribute") { switch (type) { case "Object": case "Array": return value == null ? null : JSON.stringify(value); case "Boolean": return value ? "" : null; case "Number": return value == null ? null : value; default: return value; } } else { switch (type) { case "Object": case "Array": return value && JSON.parse(value); case "Boolean": return value; case "Number": return value != null ? +value : value; default: return value; } } } function get_custom_elements_slots(element) { const result = {}; element.childNodes.forEach((node) => { result[ /** @type {Element} node */ node.slot || "default" ] = true; }); return result; } function create_custom_element(Component, props_definition, slots, exports, use_shadow_dom, extend) { let Class = class extends SvelteElement { constructor() { super(Component, slots, use_shadow_dom); this.$$p_d = props_definition; } static get observedAttributes() { return object_keys(props_definition).map( (key) => (props_definition[key].attribute || key).toLowerCase() ); } }; object_keys(props_definition).forEach((prop2) => { define_property(Class.prototype, prop2, { get() { return this.$$c && prop2 in this.$$c ? this.$$c[prop2] : this.$$d[prop2]; }, set(value) { var _a2; value = get_custom_element_value(prop2, value, props_definition); this.$$d[prop2] = value; var component2 = this.$$c; if (component2) { var setter = (_a2 = get_descriptor(component2, prop2)) == null ? undefined : _a2.get; if (setter) { component2[prop2] = value; } else { component2.$set({ [prop2]: value }); } } } }); }); exports.forEach((property) => { define_property(Class.prototype, property, { get() { var _a2; return (_a2 = this.$$c) == null ? undefined : _a2[property]; } }); }); { Class = extend(Class); } Component.element = /** @type {any} */ Class; return Class; } function onMount(fn) { if (component_context === null) { lifecycle_outside_component(); } if (legacy_mode_flag && component_context.l !== null) { init_update_callbacks(component_context).m.push(fn); } else { user_effect(() => { const cleanup = untrack(fn); if (typeof cleanup === "function") return ( /** @type {() => void} */ cleanup ); }); } } function create_custom_event(type, detail, { bubbles = false, cancelable = false } = {}) { return new CustomEvent(type, { detail, bubbles, cancelable }); } function createEventDispatcher() { const active_component_context = component_context; if (active_component_context === null) { lifecycle_outside_component(); } return (type, detail, options) => { var _a2; const events = ( /** @type {Record<string, Function | Function[]>} */ (_a2 = active_component_context.s.$$events) == null ? undefined : _a2[ /** @type {any} */ type ] ); if (events) { const callbacks = is_array(events) ? events.slice() : [events]; const event2 = create_custom_event( /** @type {string} */ type, detail, options ); for (const fn of callbacks) { fn.call(active_component_context.x, event2); } return !event2.defaultPrevented; } return true; }; } function afterUpdate(fn) { if (component_context === null) { lifecycle_outside_component(); } if (component_context.l === null) { lifecycle_legacy_only(); } init_update_callbacks(component_context).a.push(fn); } function init_update_callbacks(context) { var l = ( /** @type {ComponentContextLegacy} */ context.l ); return l.u ?? (l.u = { a: [], b: [], m: [] }); } function subscribe_to_store(store, run2, invalidate) { if (store == null) { run2(undefined); return noop; } const unsub = untrack( () => store.subscribe( run2, // @ts-expect-error invalidate ) ); return unsub.unsubscribe ? () => unsub.unsubscribe() : unsub; } const subscriber_queue = []; function readable(value, start) { return { subscribe: writable(value, start).subscribe }; } function writable(value, start = noop) { let stop = null; const subscribers = /* @__PURE__ */ new Set(); function set2(new_value) { if (safe_not_equal(value, new_value)) { value = new_value; if (stop) { const run_queue = !subscriber_queue.length; for (const subscriber of subscribers) { subscriber[1](); subscriber_queue.push(subscriber, value); } if (run_queue) { for (let i = 0; i < subscriber_queue.length; i += 2) { subscriber_queue[i][0](subscriber_queue[i + 1]); } subscriber_queue.length = 0; } } } } function update2(fn) { set2(fn( /** @type {T} */ value )); } function subscribe(run2, invalidate = noop) { const subscriber = [run2, invalidate]; subscribers.add(subscriber); if (subscribers.size === 1) { stop = start(set2, update2) || noop; } run2( /** @type {T} */ value ); return () => { subscribers.delete(subscriber); if (subscribers.size === 0 && stop) { stop(); stop = null; } }; } return { set: set2, update: update2, subscribe }; } function get(store) { let value; subscribe_to_store(store, (_) => value = _)(); return value; } function increment(source2) { set(source2, source2.v + 1); } function createSubscriber(start) { let subscribers = 0; let version = source(0); let stop; return () => { if (effect_tracking()) { get$1(version); render_effect(() => { if (subscribers === 0) { stop = untrack(() => start(() => increment(version))); } subscribers += 1; return () => { tick().then(() => { subscribers -= 1; if (subscribers === 0) { stop == null ? undefined : stop(); stop = undefined; } }); }; }); } }; } function toStore(get2, set2) { let init_value = get2(); const store = writable(init_value, (set3) => { let ran = init_value !== get2(); const teardown2 = effect_root(() => { render_effect(() => { const value = get2(); if (ran) set3(value); }); }); ran = true; return teardown2; }); if (set2) { return { set: set2, update: (fn) => set2(fn(get2())), subscribe: store.subscribe }; } return { subscribe: store.subscribe }; } 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); }, throw(msg, Err) { if (Err) { throw new Err(`${namePrefix}${msg}`); } else { throw new Error(`${namePrefix}${msg}`); } } }; } const logger = getLogger(); function sleep(delay) { return new Promise((resolve) => { setTimeout(resolve, delay); }); } function replaceInvalidChar(str) { if (typeof str !== "string") throw new TypeError("expect string but got " + typeof str); if (!str) return ""; return str.replace(new RegExp("\\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 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); } function readBlobAsDataUrl(blob) { return new Promise((resolve) => { const reader = new FileReader(); reader.onload = () => { resolve(reader.result); }; reader.readAsDataURL(blob); }); } async function addStyleToShadow(shadowRoot) { { shadowRoot.adoptedStyleSheets = [window._pdlShadowStyle]; } } function getElementText(el) { el.normalize(); if (el.childNodes.length === 0) return ""; const blockNode = [ "ADDRESS", "ARTICLE", "ASIDE", "BLOCKQUOTE", "DD", "DIV", "DL", "DT", "FIELDSET", "FIGCAPTION", "FIGURE", "FOOTER", "FORM", "H1", "H2", "H3", "H4", "H5", "H6", "HEADER", "HR", "LI", "MAIN", "NAV", "OL", "P", "PRE", "SECTION", "TABLE", "UL" ]; let str = ""; for (let i = 0; i < el.childNodes.length; i++) { const node = el.childNodes[i]; if (node.nodeType === Node.TEXT_NODE) { const val = node.nodeValue; (val == null ? undefined : val.trim()) && (str += val); } else if (node.nodeType === Node.ELEMENT_NODE) { if (node.nodeName === "BR") { str += "\n"; continue; } if (!blockNode.includes(node.nodeName)) { const childText = getElementText(node); childText && (str += childText); } else { const childText = getElementText(node); if (childText) { str ? str += "\n" + childText : str += childText; } } } } return str; } function aDownload$1(blob, filename) { const el = document.createElement("a"); el.href = URL.createObjectURL(blob); el.download = filename; el.click(); URL.revokeObjectURL(el.href); } function intersect(a, b) { const set2 = new Set(a); const result = []; b.forEach((item) => { set2.has(item) && result.push(item); }); return result; } function isPlainObject(val) { if (typeof val !== "object") return false; if (Object.prototype.toString.call(val) !== "[object Object]") return false; const proto = Object.getPrototypeOf(val); if (proto === null) return true; const valConstructor = Object.prototype.hasOwnProperty.call(proto, "constructor") && proto.constructor; return typeof valConstructor === "function" && valConstructor instanceof valConstructor && Function.prototype.toString.call(valConstructor) === Function.prototype.toString.call(Object); } function waitDom(selector, timeout = 3e3) { const el = document.querySelector(selector); if (el) { return Promise.resolve(el); } return new Promise((resolve) => { let timer; const observer2 = new MutationObserver(() => { const el2 = document.querySelector(selector); if (el2) { resolve(el2); observer2.disconnect(); if (timer) { clearTimeout(timer); } } }); observer2.observe(document, { childList: true, subtree: true }); timer = setTimeout(() => { resolve(null); observer2.disconnect(); }, timeout); }); } class ChannelEvent { constructor(channelName) { __privateAdd(this, _channel); __privateAdd(this, _event); __privateSet(this, _channel, new BroadcastChannel(channelName)); __privateSet(this, _event, {}); __privateGet(this, _channel).onmessage = (event2) => { const { eventName, args } = event2.data; if (!__privateGet(this, _event)[eventName]) return; __privateGet(this, _event)[eventName].forEach((callback) => callback(...args)); }; } on(eventName, callback) { var _a2; (_a2 = __privateGet(this, _event))[eventName] ?? (_a2[eventName] = /* @__PURE__ */ new Set()); __privateGet(this, _event)[eventName].add(callback); } off(eventName, callback) { if (!__privateGet(this, _event)[eventName]) return; __privateGet(this, _event)[eventName].delete(callback); } emit(eventName, ...args) { const data = { eventName, args }; __privateGet(this, _channel).postMessage(data); } once(eventName, callback) { const onceCallback = (...args) => { callback(...args); this.off(eventName, onceCallback); }; this.on(eventName, onceCallback); } } _channel = new WeakMap(); _event = new WeakMap(); const channelEvent = new ChannelEvent("pixiv-downloader"); const _HistoryDb = class _HistoryDb extends Dexie { constructor() { super("PdlHistory"); __publicField(this, "history"); __publicField(this, "imageEffect"); __publicField(this, "filehandle"); __privateAdd(this, _DIRECTORY_HANDLE_NAME, "directory-handle"); this.version(4).stores({ history: "pid, userId, user, title, *tags", imageEffect: "id", filehandle: "" }); } throwIfInvalidNumber(num) { if (typeof num === "string") { if (num !== "") { num = +num; } else { return logger.throw('Invalid argument: can not be "".', RangeError); } } if (num < 0 || !Number.isSafeInteger(num)) { logger.throw(`Invalid number: ${num}, must be a non-negative integer.`, RangeError); } return num; } async add(historyData) { const { pid, page } = historyData; return this.transaction("rw", this.history, async () => { if (page !== undefined) { this.throwIfInvalidNumber(page); const historyItem = await this.get(pid); if (historyItem && historyItem.page === undefined) { delete historyData.page; this.history.put(historyData); } else { const u8arr = _HistoryDb.updatePageData(page, historyItem == null ? undefined : historyItem.page); this.history.put({ ...historyData, page: u8arr }); } } else { this.history.put(historyData); } }); } import(objArr) { const historyItems = objArr.map((historyObj) => { if (historyObj.page) { return { ...historyObj, page: new Uint8Array(Object.values(historyObj.page)) }; } else { return historyObj; } }); return this.history.bulkPut(historyItems); } async has(pid, page) { if (page === undefined) { return !!await this.get(pid); } else { this.throwIfInvalidNumber(page); const historyItem = await this.get(pid); if (!historyItem) return false; if (!historyItem.page) return true; return _HistoryDb.isPageInData(page, historyItem.page); } } get(pid) { pid = this.throwIfInvalidNumber(pid); return this.history.get(pid); } getAll() { return this.history.toArray(); } generateCsv() { return this.getAll().then((datas) => { const csvData = datas.map((historyData) => { const { pid, userId = "", user = "", title = "", tags = "", comment: comment2 = "" } = historyData; return [String(pid), String(userId), user, title, comment2, tags ? tags.join(",") : tags]; }); csvData.unshift(["id", "userId", "user", "title", "comment", "tags"]); return generateCsv(csvData); }); } clear() { return this.history.clear(); } // Firefox does not support storing `Arraybuffer`, so it will always return `undefined`. async getImageEffect(effectId) { return await this.imageEffect.get(effectId); } addImageEffect(effectData) { return this.imageEffect.put(effectData); } getDirectoryHandle() { return this.filehandle.get(__privateGet(this, _DIRECTORY_HANDLE_NAME)); } setDirectoryHandle(dirHandle2) { return this.filehandle.put(dirHandle2, __privateGet(this, _DIRECTORY_HANDLE_NAME)); } static updatePageData(page, pageData) { const byteIndex = Math.floor(page / 8); const bitIndex = page % 8; if (!pageData) { const newArr = new Uint8Array(byteIndex + 1); newArr[byteIndex] |= 1 << bitIndex; return newArr; } else if (byteIndex > pageData.length - 1) { const newArr = new Uint8Array(byteIndex + 1); newArr.set(pageData); newArr[byteIndex] |= 1 << bitIndex; return newArr; } else { pageData[byteIndex] |= 1 << bitIndex; return pageData; } } static isPageInData(page, pageData) { const byteIndex = Math.floor(page / 8); const bitIndex = page % 8; return !(byteIndex > pageData.length - 1) && (pageData[byteIndex] & 1 << bitIndex) !== 0; } }; _DIRECTORY_HANDLE_NAME = new WeakMap(); let HistoryDb = _HistoryDb; class CachedHistoryDb extends HistoryDb { constructor() { super(); __publicField(this, "cache", /* @__PURE__ */ new Map()); __publicField(this, "initCachePromise"); this.initCachePromise = this.initCache(); this.initChannel(); } async initCache() { logger.time("loadDb"); const historyItems = await this.getAll(); let historyItem; for (let i = 0; historyItem = historyItems[i++]; ) { const { pid, page = null } = historyItem; this.cache.set(pid, page); } logger.timeEnd("loadDb"); } initChannel() { channelEvent.on("db.sync", (items) => { if (Array.isArray(items)) { items.forEach((item) => { this.cache.set(item.pid, item.page); }); logger.info("Sync database cache:", items.length); } else { this.cache.set(items.pid, items.page); } }); channelEvent.on("db.clear", () => { this.cache.clear(); logger.info("clear database cache"); }); } updateCache(item) { if (Array.isArray(item)) { item.forEach((cache) => { this.cache.set(cache.pid, cache.page); }); } else { this.cache.set(item.pid, item.page); } channelEvent.emit("db.sync", item); } clearCache() { this.cache.clear(); channelEvent.emit( "db.clear" /* CLEAR */ ); } async getCache(pid) { pid = this.throwIfInvalidNumber(pid); await this.initCachePromise; return this.cache.get(pid); } async add(historyData) { const { pid, page } = historyData; if (page === undefined) { !await this.has(pid) && this.updateCache({ pid, page: null }); } else { this.throwIfInvalidNumber(page); const pageData = await this.getCache(pid); if (pageData !== null) { this.updateCache({ pid, page: HistoryDb.updatePageData(page, pageData) }); } else { this.updateCache({ pid, page: null }); } } return super.add(historyData); } import(objArr) { const cacheItems = objArr.map((historyObj) => { return { pid: historyObj.pid, page: historyObj.page ? new Uint8Array(Object.values(historyObj.page)) : null }; }); this.updateCache(cacheItems); return super.import(objArr); } async has(pid, page) { pid = this.throwIfInvalidNumber(pid); await this.initCachePromise; if (page === undefined) { return this.cache.has(pid); } else { this.throwIfInvalidNumber(page); const cachesData = await this.getCache(pid); if (cachesData === undefined) return false; if (cachesData === null) return true; return HistoryDb.isPageInData(page, cachesData); } } clear() { this.clearCache(); return super.clear(); } } class ReadableHistoryDb extends CachedHistoryDb { constructor() { super(...arguments); __publicField(this, "subscribers", /* @__PURE__ */ new Set()); } runSubscription() { logger.info("runSubscription", this.subscribers.size); this.subscribers.forEach((subscription) => { subscription(this.cache); }); } async initCache() { await super.initCache(); this.runSubscription(); } initChannel() { super.initChannel(); const runSubscription = this.runSubscription.bind(this); channelEvent.on("db.sync", runSubscription); channelEvent.on("db.clear", runSubscription); } updateCache(item) { super.updateCache(item); this.runSubscription(); } clearCache() { super.clearCache(); this.runSubscription(); } subscribe(subscription) { this.subscribers.add(subscription); subscription(this.cache); return () => { this.subscribers.delete(subscription); }; } } const historyDb = new ReadableHistoryDb(); const btnStyle = ".pdl-thumbnail{position:absolute;display:flex;justify-content:center;align-items:center;margin:0;padding:0;height:32px;width:32px;top:calc((100% - 32px) * var(--pdl-btn-top-percent) / 100 + var(--pdl-btn-top-px));left:calc((100% - 32px) * var(--pdl-btn-left-percent) / 100 + var(--pdl-btn-left-px));border:none;border-radius:4px;overflow:hidden;white-space:nowrap;-webkit-user-select:none;-moz-user-select:none;user-select:none;font-family:system-ui;font-size:13px;font-weight:700;color:#262626;background-color:#ffffff80;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px);z-index:1;pointer-events:auto;cursor:pointer}.pdl-thumbnail:disabled{cursor:not-allowed}.pdl-thumbnail>svg{position:absolute;width:85%;height:85%;fill:currentColor;stroke:currentColor}.pdl-thumbnail>span{opacity:0;transition:opacity .2s}.pdl-thumbnail>span.show{opacity:1}:host([data-type=gallery]) .pdl-thumbnail{--pdl-sticky-safe-area-top: 0px;--pdl-sticky-safe-area-bottom: 0px;position:sticky;top:var(--pdl-sticky-safe-area-top);left:0;right:0}:host([data-type=gallery].bottom) .pdl-thumbnail{top:calc(100% - 32px - var(--pdl-sticky-safe-area-bottom))}:host([data-type=gallery].pixiv) .pdl-thumbnail{--pdl-sticky-safe-area-top: 40px;--pdl-sticky-safe-area-bottom: 48px}:host([data-type=gallery].nijie) .pdl-thumbnail{--pdl-sticky-safe-area-top: 36px;--pdl-sticky-safe-area-bottom: 70px}:host([data-type=pixiv-my-bookmark]) .pdl-thumbnail{top:calc((100% - 32px) * var(--pdl-btn-pixiv-bookmark-top-percent) / 100 + var(--pdl-btn-pixiv-bookmark-top-px));left:calc((100% - 32px) * var(--pdl-btn-pixiv-bookmark-left-percent) / 100 + var(--pdl-btn-pixiv-bookmark-left-px))}:host([data-type=pixiv-history]) .pdl-thumbnail{z-index:auto}:host([data-type=pixiv-presentation]) .pdl-thumbnail{position:fixed;top:50px;right:20px;left:auto}:host([data-type=pixiv-toolbar]) .pdl-thumbnail{position:relative;top:auto;left:auto;color:inherit;background-color:transparent}:host([data-type=pixiv-manga-viewer]) .pdl-thumbnail{top:80%;right:4px;left:auto}:host([data-type=yande-browse]) .pdl-thumbnail{top:320px;right:4px;left:auto}:host([data-type=nijie-illust]) .pdl-thumbnail{display:inline-flex;position:static;height:44px;width:44px;top:auto;left:auto;border-radius:8px;margin:0 8px;vertical-align:top}:host([data-type=nijie-illust]) .pdl-thumbnail>svg{width:70%;height:70%}:host([data-type=rule34vault-post-action]) .pdl-thumbnail{position:relative;top:auto;left:auto;color:inherit;background-color:inherit;width:4em;height:36px}:host([data-type=rule34vault-post-action]) .pdl-thumbnail>svg{height:28px;width:28px}:host([data-type=rule34vault-post-action]) .pdl-thumbnail:hover{filter:brightness(1.2)}:host([data-status]) .pdl-thumbnail{color:#16a34a}:host([data-status=error]) .pdl-thumbnail{color:#ef4444}"; const svgGroup = `<svg xmlns="http://www.w3.org/2000/svg" style="display: none"> <symbol id="pdl-download" viewBox="0 0 512 512"> <path 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" ></path> </symbol> <symbol id="pdl-loading" viewBox="0 0 512 512"> <style> @keyframes pdl-loading { 0% { transform: rotate3d(0, 0, 1, -90deg) rotate3d(1, 0, 0, 0deg); stroke-dashoffset: 1407.43; } 49.99% { transform: rotate3d(0, 0, 1, 90deg) rotate3d(1, 0, 0, 0deg); } 50% { transform: rotate3d(0, 0, 1, 90deg) rotate3d(1, 0, 0, 180deg); stroke-dashoffset: 0; } 100% { transform: rotate3d(0, 0, 1, 270deg) rotate3d(1, 0, 0, 180deg); stroke-dashoffset: 1407.43; } } circle.rotate { transform-origin: 50% 50%; animation: 2.5s infinite ease-in-out pdl-loading; } </style> <circle class="rotate" cx="256" cy="256" r="224" stroke-width="48" fill="none" stroke-dasharray="1407.43" stroke-dashoffset="1055.57" stroke-linecap="round" ></circle> </symbol> <symbol id="pdl-progress" viewBox="0 0 512 512"> <style> circle.progress { transition: stroke-dashoffset 0.2s ease; } </style> <circle class="progress" cx="256" cy="256" r="224" stroke-width="48" fill="none" stroke-dasharray="1407.43" stroke-linecap="round" transform="rotate(-90 256 256)" ></circle> </symbol> <symbol id="pdl-error" viewBox="0 0 512 512"> <path 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" ></path> </symbol> <symbol id="pdl-complete" viewBox="0 0 512 512"> <path 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" ></path> </symbol> </svg> `; const iconTypeMap = { init: "#pdl-download", loading: "#pdl-loading", progress: "#pdl-progress", complete: "#pdl-complete", error: "#pdl-error" }; var ThumbnailBtnStatus = /* @__PURE__ */ ((ThumbnailBtnStatus2) => { ThumbnailBtnStatus2["Init"] = "init"; ThumbnailBtnStatus2["Loading"] = "loading"; ThumbnailBtnStatus2["Progress"] = "progress"; ThumbnailBtnStatus2["Complete"] = "complete"; ThumbnailBtnStatus2["Error"] = "error"; return ThumbnailBtnStatus2; })(ThumbnailBtnStatus || {}); var ThumbnailBtnType = /* @__PURE__ */ ((ThumbnailBtnType2) => { ThumbnailBtnType2["Gallery"] = "gallery"; ThumbnailBtnType2["PixivMyBookmark"] = "pixiv-my-bookmark"; ThumbnailBtnType2["PixivHistory"] = "pixiv-history"; ThumbnailBtnType2["PixivPresentation"] = "pixiv-presentation"; ThumbnailBtnType2["PixivToolbar"] = "pixiv-toolbar"; ThumbnailBtnType2["PixivMangaViewer"] = "pixiv-manga-viewer"; ThumbnailBtnType2["YandeBrowse"] = "yande-browse"; ThumbnailBtnType2["NijieIllust"] = "nijie-illust"; ThumbnailBtnType2["Rule34VaultPostAction"] = "rule34vault-post-action"; return ThumbnailBtnType2; })(ThumbnailBtnType || {}); class ThumbnailButton extends HTMLElement { constructor(props) { super(); __privateAdd(this, _ThumbnailButton_instances); __publicField(this, "btn"); __publicField(this, "status", "init"); __publicField(this, "mediaId"); __publicField(this, "page"); __publicField(this, "type"); __publicField(this, "onClick"); __publicField(this, "unsubscriber"); __publicField(this, "connectedFlag", false); __publicField(this, "shouldObserveDb", true); __publicField(this, "progress", 0); __publicField(this, "dirty", false); __privateAdd(this, _downloadingId, null); __privateAdd(this, _downloadingPage, null); this.dispatchDownload = this.dispatchDownload.bind(this); this.onClick = props.onClick; this.mediaId = this.toValidatedNumber(props.id); this.dataset.id = String(this.mediaId); if (props.type) { this.dataset.type = this.type = props.type; } if (props.page !== undefined) { this.page = this.toValidatedNumber(props.page); this.dataset.page = String(this.page); } props.shouldObserveDb !== undefined && (this.shouldObserveDb = props.shouldObserveDb); if (props.extraData) { for (const key in props.extraData) { this.dataset[key] = props.extraData[key]; } } } static get tagNameLowerCase() { return "pdl-button"; } static get observedAttributes() { return ["data-id", "data-status", "data-page", "data-type", "disabled"]; } attributeChangedCallback(name, _oldValue, newValue) { switch (name) { case "data-id": this.updateId(newValue); break; case "data-status": this.updateIcon(newValue); break; case "data-page": this.updatePage(newValue); break; case "data-type": this.resetType(newValue); break; case "disabled": this.updateDisableStatus(newValue); break; } } toValidatedNumber(num) { if (typeof num === "string") { if (num !== "") { num = +num; } else { return logger.throw('Invalid argument: can not be "".', RangeError); } } if (num < 0 || !Number.isSafeInteger(num)) { return logger.throw(`Invalid number: ${num}, must be a non-negative integer.`, RangeError); } return num; } resetType(newVal) { if (newVal === null && this.type === undefined) return; if (newVal !== this.type) { if (this.type === undefined) { delete this.dataset.type; } else { this.dataset.type = this.type; } logger.error('Changes to "data-type" is not allowed.'); } } updateId(id) { try { if (id === null) throw new Error('Attribute "data-id" is required.'); this.mediaId = this.toValidatedNumber(id); __privateMethod(this, _ThumbnailButton_instances, resetStatus_fn).call(this); __privateGet(this, _downloadingId) && __privateSet(this, _downloadingId, null); this.connectedFlag && this.shouldObserveDb && this.observeDb()(); } catch (error) { logger.error(error); this.dataset.id = String(this.mediaId); } } updateDisableStatus(val) { if (!this.connectedFlag) { this.dirty = true; return; } if (typeof val === "string") { this.btn.setAttribute("disabled", ""); } else { this.btn.removeAttribute("disabled"); } } updatePage(page) { try { this.page = page === null ? void 0 : this.toValidatedNumber(page); __privateGet(this, _downloadingPage) && __privateSet(this, _downloadingPage, null); __privateMethod(this, _ThumbnailButton_instances, resetStatus_fn).call(this); this.connectedFlag && this.shouldObserveDb && this.observeDb()(); } catch (error) { logger.error(error); if (this.page === undefined) { delete this.dataset.page; } else { this.dataset.page = String(this.page); } } } updateIcon(status) { if (status === null) { status = "init"; } else if (status === "init") { delete this.dataset.status; return; } else if (!(status in iconTypeMap)) { this.dataset.status = this.status; return; } this.status = status; if (!this.connectedFlag) { this.dirty = true; return; } const useEl = this.shadowRoot.querySelector("use"); useEl.setAttribute("xlink:href", iconTypeMap[status]); useEl.animate([{ opacity: 0.5 }, { opactiy: 1 }], { duration: 200 }); } render() { let shadowRoot; if ((shadowRoot = this.shadowRoot) && !this.dirty) return; const statusIsProgress = this.status === "progress"; shadowRoot ?? (shadowRoot = this.attachShadow({ mode: "open" })); shadowRoot.innerHTML = `<style>${btnStyle}</style>${svgGroup}<button class="pdl-thumbnail" ${this.hasAttribute("disabled") ? "disabled" : ""}> <svg xmlns="http://www.w3.org/2000/svg" class="pdl-icon" ${statusIsProgress ? `style="stroke-dashoffset: ${this.clacProgressRadial(this.progress)};"` : ""}> <use xlink:href="${iconTypeMap[this.status]}"></use> </svg> ${statusIsProgress ? `<span class="show">${this.progress}</span>` : "<span></span>"} </button>`; } dispatchDownload(evt) { evt == null ? undefined : evt.preventDefault(); evt == null ? undefined : evt.stopPropagation(); this.setAttribute("disabled", ""); this.setStatus( "loading" /* Loading */ ); const id = __privateSet(this, _downloadingId, this.mediaId); const page = __privateSet(this, _downloadingPage, this.page); Promise.resolve(this.onClick(this)).then( () => { if (id === __privateGet(this, _downloadingId) && page === __privateGet(this, _downloadingPage)) { this.setStatus( "complete" /* Complete */ ); this.removeAttribute("disabled"); } }, (err) => { if (err) logger.error(err); if (id === __privateGet(this, _downloadingId) && page === __privateGet(this, _downloadingPage)) { this.setStatus( "error" /* Error */ ); this.removeAttribute("disabled"); } } ).finally(() => { id === __privateGet(this, _downloadingId) && __privateSet(this, _downloadingId, null); page === __privateGet(this, _downloadingPage) && __privateSet(this, _downloadingPage, null); }); } observeDb() { return historyDb.subscribe(async () => { const id = this.mediaId; const page = this.page; const downloaded = await historyDb.has(id, page); if (id !== this.mediaId || page !== this.page) return; if (this.status === "complete") { !downloaded && this.setStatus( "init" /* Init */ ); } else { downloaded && this.setStatus( "complete" /* Complete */ ); } }); } connectedCallback() { this.render(); this.dirty && (this.dirty = false); this.connectedFlag = true; this.btn = this.shadowRoot.querySelector("button"); this.btn.addEventListener("click", this.dispatchDownload); this.shouldObserveDb && (this.unsubscriber = this.observeDb()); } disconnectedCallback() { var _a2, _b2; this.connectedFlag = false; (_a2 = this.btn) == null ? undefined : _a2.removeEventListener("click", this.dispatchDownload); (_b2 = this.unsubscriber) == null ? undefined : _b2.call(this); } clacProgressRadial(progress) { const radius = 224; const circumference = 2 * Math.PI * radius; const offset = circumference - progress / 100 * circumference; return offset; } setProgress(progress, updateProgressbar = true) { if (progress < 0 || progress > 100) throw new RangeError('Value "progress" must between 0-100'); if (this.mediaId !== __privateGet(this, _downloadingId) || this.page !== __privateGet(this, _downloadingPage)) return; this.progress = Math.floor(progress); if (this.status !== "progress") { this.dataset.status = "progress"; } if (!this.connectedFlag) { this.dirty = true; return; } const shadowRoot = this.shadowRoot; const span = shadowRoot.querySelector("span"); span.classList.add("show"); span.textContent = String(this.progress); if (!updateProgressbar) return; const svg = shadowRoot.querySelector("svg.pdl-icon"); svg.style.strokeDashoffset = String(this.clacProgressRadial(progress)); } removeProgress() { this.progress = 0; if (!this.connectedFlag) { this.dirty = true; return; } const shadowRoot = this.shadowRoot; const span = shadowRoot.querySelector("span"); const svg = shadowRoot.querySelector("svg.pdl-icon"); if (!span.classList.contains("show")) return; span.classList.remove("show"); span.addEventListener( "transitionend", () => { span.textContent = ""; }, { once: true } ); svg.style.removeProperty("stroke-dashoffset"); } setStatus(newStatus) { if (newStatus !== this.status) { if (newStatus === "init") { delete this.dataset.status; return; } if (newStatus === "progress") { this.setProgress(0); return; } this.removeProgress(); this.dataset.status = newStatus; } } } _downloadingId = new WeakMap(); _downloadingPage = new WeakMap(); _ThumbnailButton_instances = new WeakSet(); resetStatus_fn = function() { this.hasAttribute("disabled") && this.removeAttribute("disabled"); this.setStatus( "init" /* Init */ ); }; customElements.define(ThumbnailButton.tagNameLowerCase, ThumbnailButton); const wrapperStyle = ".button-wrapper{--pdl-sticky-container-safe-area-top: 0px;--pdl-sticky-container-safe-area-bottom: 0px;--pdl-sticky-container-safe-area-right: 8px;--pdl-sticky-container-safe-area-left: 8px;inset:var(--pdl-sticky-container-safe-area-top) var(--pdl-sticky-container-safe-area-right) var(--pdl-sticky-container-safe-area-bottom) var(--pdl-sticky-container-safe-area-left);position:absolute;direction:ltr;pointer-events:none}.button-wrapper.rtl{direction:rtl}.button-wrapper.pixiv{--pdl-sticky-container-safe-area-top: 40px}.button-wrapper.gelbooru{--pdl-sticky-container-safe-area-bottom: calc(1em + 22px) }.button-wrapper.moebooru_image{--pdl-sticky-container-safe-area-right: calc(5em + 8px) }.button-wrapper.native_video{--pdl-sticky-container-safe-area-bottom: 70px}.button-wrapper.vjs_video{--pdl-sticky-container-safe-area-bottom: 32px}.button-wrapper.fluid_video{--pdl-sticky-container-safe-area-bottom: 56px}"; const gifWorker = (() => GM_getResourceText("gif.js/dist/gif.worker?raw"))(); const workerUrl$2 = URL.createObjectURL(new Blob([gifWorker], { type: "text/javascript" })); function isBlobArray$2(frames2) { return frames2[0] instanceof Blob; } async function gif(frames2, delays, quality, signal, onProgress) { signal == null ? undefined : signal.throwIfAborted(); if (isBlobArray$2(frames2)) { frames2 = await Promise.all(frames2.map((frame) => createImageBitmap(frame))); } signal == null ? undefined : signal.throwIfAborted(); const canvas = document.createElement("canvas"); const width = canvas.width = frames2[0].width; const height = canvas.height = frames2[0].height; const ctx = canvas.getContext("2d", { willReadFrequently: true }); let resolveConvert; let rejectConvert; const convertPromise = new Promise((resolve, reject) => { resolveConvert = resolve; rejectConvert = reject; }); const gif2 = new GIF({ workers: 2, quality, width, height, workerScript: workerUrl$2 }); gif2.on("progress", (progress) => { onProgress == null ? undefined : onProgress(progress * 100); }); gif2.on("abort", () => { rejectConvert(signal == null ? undefined : signal.reason); }); gif2.on("finished", (gifBlob) => { resolveConvert(gifBlob); }); signal == null ? undefined : signal.addEventListener( "abort", () => { gif2.abort(); }, { once: true } ); frames2.forEach((bitmap, i) => { ctx.drawImage(bitmap, 0, 0); gif2.addFrame(ctx, { copy: true, delay: delays[i] }); bitmap.close(); }); gif2.render(); return convertPromise; } const pngWorkerFragment = 'function isBlobArray(frames) {\n return frames[0] instanceof Blob;\n}\nasync function encodeAPNG(frames, delays, cnum) {\n if (isBlobArray(frames)) {\n frames = await Promise.all(frames.map((frame) => createImageBitmap(frame)));\n }\n const width = frames[0].width;\n const height = frames[0].height;\n const canvas = new OffscreenCanvas(width, height);\n const ctx = canvas.getContext("2d", { willReadFrequently: true });\n const u8arrs = [];\n for (const frame of frames) {\n ctx.drawImage(frame, 0, 0);\n frame.close();\n u8arrs.push(ctx.getImageData(0, 0, width, height).data);\n }\n const png = UPNG.encode(u8arrs, width, height, cnum, delays, { loop: 0 });\n if (!png)\n throw new Error("Failed to encode apng.");\n return png;\n}\nfunction decodeApng(ab) {\n const img = UPNG.decode(ab);\n const rgba = UPNG.toRGBA8(img);\n const { width, height } = img;\n const delays = img.frames.map((frame) => frame.delay);\n return { frames: rgba, delays, width, height };\n}\nasync function appendEffect(illustBlob, effect) {\n const illustBitmap = await createImageBitmap(illustBlob);\n const { frames: effectFrames, delays, width, height } = decodeApng(effect);\n const effectBitmaps = await Promise.all(effectFrames.map((buf) => createImageBitmap(new ImageData(new Uint8ClampedArray(buf), width, height))));\n const { width: illustWidth, height: illustHeight } = illustBitmap;\n const illustAspectRatio = illustWidth / illustHeight;\n const effectAspectRatio = width / height;\n let dx;\n let dy;\n let dWidth;\n let dHeight;\n if (effectAspectRatio > illustAspectRatio) {\n dWidth = illustHeight * effectAspectRatio;\n dHeight = illustHeight;\n dx = (illustWidth - dWidth) / 2;\n dy = 0;\n }\n else {\n dWidth = illustWidth;\n dHeight = illustWidth / effectAspectRatio;\n dx = 0;\n dy = (illustHeight - dHeight) / 2;\n }\n const canvas = new OffscreenCanvas(illustWidth, illustHeight);\n const ctx = canvas.getContext("2d", { willReadFrequently: true });\n const finalDatas = [];\n for (const effectBitmap of effectBitmaps) {\n ctx.drawImage(illustBitmap, 0, 0);\n ctx.drawImage(effectBitmap, dx, dy, dWidth, dHeight);\n finalDatas.push(ctx.getImageData(0, 0, illustWidth, illustHeight));\n effectBitmap.close();\n }\n illustBitmap.close();\n const result = finalDatas.map((data) => createImageBitmap(data));\n return {\n frames: await Promise.all(result),\n delays\n };\n}\nself.onmessage = async (evt) => {\n try {\n const data = evt.data;\n if ("effect" in data) {\n const { illust, effect } = data;\n const result = await appendEffect(illust, effect);\n self.postMessage(result, [...result.frames]);\n }\n else {\n const { frames, delays, cnum = 256 } = data;\n const apng = await encodeAPNG(frames, delays, cnum);\n self.postMessage(apng, [apng]);\n }\n }\n catch (error) {\n console.error(error);\n self.postMessage(void 0);\n }\n};\n'; const UPNG = (() => GM_getResourceText("upng-js?raw"))(); const pako = (() => GM_getResourceText("pako/dist/pako.js?raw"))(); 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 = []; async function png(frames2, delays, cnum, signal) { signal == null ? undefined : signal.throwIfAborted(); let resolveConvert; let rejectConvert; const convertPromise = new Promise((resolve, reject) => { resolveConvert = resolve; rejectConvert = reject; }); let worker; if (freeApngWorkers.length) { worker = freeApngWorkers.shift(); logger.info("Reuse apng workers."); } else { worker = new Worker(workerUrl$1); } signal == null ? undefined : signal.addEventListener( "abort", () => { worker.terminate(); rejectConvert(signal == null ? undefined : signal.reason); }, { once: true } ); worker.onmessage = function(e) { freeApngWorkers.push(worker); if (signal == null ? undefined : signal.aborted) return; if (!e.data) { return rejectConvert(new TypeError("Failed to get png data.")); } const pngBlob = new Blob([e.data], { type: "image/png" }); resolveConvert(pngBlob); }; const cfg = { frames: frames2, delays, cnum }; worker.postMessage( cfg, cfg.frames[0] instanceof ImageBitmap ? cfg.frames : [] ); return convertPromise; } function mixPngEffect(illust, seasonalEffect, signal) { signal == null ? undefined : signal.throwIfAborted(); let resolveConvert; let rejectConvert; const convertPromise = new Promise((resolve, reject) => { resolveConvert = resolve; rejectConvert = reject; }); let worker; if (freeApngWorkers.length) { worker = freeApngWorkers.shift(); logger.info("Reuse apng workers."); } else { worker = new Worker(workerUrl$1); } signal == null ? undefined : signal.addEventListener("abort", () => { worker.terminate(); rejectConvert(signal == null ? undefined : signal.reason); }); worker.onmessage = function(e) { worker.terminate(); if (!e.data) { return rejectConvert(new Error("Mix Effect convert Failed.")); } resolveConvert(e.data); }; const cfg = { illust, effect: seasonalEffect }; worker.postMessage(cfg, [seasonalEffect]); return convertPromise; } const webpWorkerFragment = 'let resolveModule;\nconst moduleLoaded = new Promise((resolve) => {\n resolveModule = resolve;\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 resolveModule();\n};\nfunction isBlobArray(frames) {\n return frames[0] instanceof Blob;\n}\nonmessage = async (evt) => {\n await moduleLoaded;\n const { frames, delays, lossless = 0, quality = 95, method = 4 } = evt.data;\n webpApi.init(lossless, quality, method);\n const bitmaps = isBlobArray(frames) ? await Promise.all(frames.map((frame) => createImageBitmap(frame))) : frames;\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 for (let i = 0; i < bitmaps.length; i++) {\n ctx?.drawImage(bitmaps[i], 0, 0);\n bitmaps[i].close();\n const webpBlob = await canvas.convertToBlob({\n type: "image/webp",\n quality: lossless ? 1 : quality / 100\n });\n const buffer = await webpBlob.arrayBuffer();\n const u8a = new Uint8Array(buffer);\n const pointer = webpApi.createBuffer(u8a.length);\n Module.HEAPU8.set(u8a, pointer);\n webpApi.addFrame(pointer, u8a.length, delays[i]);\n postMessage((i + 1) / bitmaps.length * 100);\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 webpWasm = (() => GM_getResourceText("../wasm/toWebpWorker?raw"))(); const workerUrl = URL.createObjectURL( new Blob([webpWasm + webpWorkerFragment], { type: "text/javascript" }) ); const freeWebpWorkers = []; async function webp(frames2, delays, lossless, quality, method, signal, onProgress) { signal == null ? undefined : signal.throwIfAborted(); let worker; if (freeWebpWorkers.length) { worker = freeWebpWorkers.shift(); logger.info("Reuse webp workers."); } else { worker = new Worker(workerUrl); } let resolveConvert; let rejectConvert; const convertPromise = new Promise((resolve, reject) => { resolveConvert = resolve; rejectConvert = reject; }); signal == null ? undefined : signal.addEventListener( "abort", () => { worker.terminate(); rejectConvert(signal == null ? undefined : signal.reason); }, { once: true } ); worker.onmessage = (evt) => { if (signal == null ? undefined : signal.aborted) return; const data = evt.data; if (typeof data !== "object") { onProgress == null ? undefined : onProgress(evt.data); } else { freeWebpWorkers.push(worker); resolveConvert(new Blob([evt.data], { type: "image/webp" })); } }; worker.postMessage( { frames: frames2, delays, lossless: Number(lossless), quality, method }, frames2[0] instanceof ImageBitmap ? frames2 : [] ); return convertPromise; } function isBlobArray$1(frames2) { return frames2[0] instanceof Blob; } async function webm(frames2, delays, bitrate, signal) { signal == null ? undefined : signal.throwIfAborted(); if (isBlobArray$1(frames2)) { frames2 = await Promise.all(frames2.map((frame) => createImageBitmap(frame))); } signal == null ? undefined : signal.throwIfAborted(); const width = frames2[0].width; const height = frames2[0].height; const muxer = new webmMuxer.Muxer({ target: new webmMuxer.ArrayBufferTarget(), video: { codec: "V_VP9", width, height } }); const videoEncoder = new VideoEncoder({ output: (chunk, meta) => muxer.addVideoChunk(chunk, meta), error: (e) => logger.error(e) }); videoEncoder.configure({ codec: "vp09.00.51.08.01.01.01.01.00", width, height, bitrate: bitrate * 1e6 }); let timestamp = 0; delays = delays.map((delay) => delay *= 1e3); const videoFrames = []; signal == null ? undefined : signal.addEventListener( "abort", () => { videoFrames.forEach((frame) => frame.close()); }, { once: true } ); for (const [i, frame] of frames2.entries()) { const videoFrame = new VideoFrame(frame, { duration: delays[i], timestamp }); videoEncoder.encode(videoFrame); videoFrames.push(videoFrame); frame.close(); timestamp += delays[i]; } await videoEncoder.flush(); videoEncoder.close(); videoFrames.forEach((frame) => frame.close()); signal == null ? undefined : signal.throwIfAborted(); muxer.finalize(); const { buffer } = muxer.target; return new Blob([buffer], { type: "video/webm" }); } function isBlobArray(frames2) { return frames2[0] instanceof Blob; } async function mp4(frames2, delays, bitrate, signal) { signal == null ? undefined : signal.throwIfAborted(); if (isBlobArray(frames2)) { frames2 = await Promise.all(frames2.map((frame) => createImageBitmap(frame))); } signal == null ? undefined : signal.throwIfAborted(); let width = frames2[0].width; let height = frames2[0].height; if (width % 2 !== 0) width += 1; if (height % 2 !== 0) height += 1; const muxer = new mp4Muxer.Muxer({ target: new mp4Muxer.ArrayBufferTarget(), video: { codec: "avc", width, height }, fastStart: "in-memory" }); const videoEncoder = new VideoEncoder({ output: (chunk, meta) => muxer.addVideoChunk(chunk, meta), error: (e) => logger.error(e) }); videoEncoder.configure({ codec: "avc1.420034", width, height, bitrate: bitrate * 1e6 }); let timestamp = 0; delays = delays.map((delay) => delay *= 1e3); const videoFrames = []; signal == null ? undefined : signal.addEventListener( "abort", () => { videoFrames.forEach((frame) => frame.close()); }, { once: true } ); for (const [i, frame] of frames2.entries()) { const videoFrame = new VideoFrame(frame, { duration: delays[i], timestamp }); videoEncoder.encode(videoFrame); videoFrames.push(videoFrame); frame.close(); timestamp += delays[i]; } await videoEncoder.flush(); videoEncoder.close(); videoFrames.forEach((frame) => frame.close()); signal == null ? undefined : signal.throwIfAborted(); muxer.finalize(); const { buffer } = muxer.target; return new Blob([buffer], { type: "video/mp4" }); } var ConvertFormat = /* @__PURE__ */ ((ConvertFormat2) => { ConvertFormat2["GIF"] = "gif"; ConvertFormat2["WEBP"] = "webp"; ConvertFormat2["PNG"] = "png"; ConvertFormat2["WEBM"] = "webm"; ConvertFormat2["MP4"] = "mp4"; return ConvertFormat2; })(ConvertFormat || {}); const convertAdapter = { getAdapter(option) { switch (option.format) { case "gif": return (frames2, delays, signal, onProgress) => { return gif(frames2, delays, option.quality, signal, onProgress); }; case "png": return (frames2, delays, signal) => { return png(frames2, delays, option.cnum, signal); }; case "mp4": return (frames2, delays, signal) => { return mp4(frames2, delays, option.bitrate, signal); }; case "webm": return (frames2, delays, signal) => { return webm(frames2, delays, option.bitrate, signal); }; case "webp": return (frames2, delays, signal, onProgress) => { const { lossless, quality, method } = option; return webp(frames2, delays, lossless, quality, method, signal, onProgress); }; default: throw new Error("Unspported format."); } }, getMixEffectFn() { return mixPngEffect; } }; const configStr = localStorage.getItem("pdlSetting"); if (configStr) localStorage.removeItem("pdlSetting"); const legacyConfig = configStr ? JSON.parse(configStr) : {}; function mergeDeepState(patch, stateToPatch) { for (const key of Object.keys(patch)) { if (!(key in stateToPatch)) continue; const patchVal = patch[key]; const stateVal = stateToPatch[key]; if (isPlainObject(patchVal) && isPlainObject(stateVal)) { mergeDeepState(patchVal, stateVal); } else if (patchVal !== undefined) { stateToPatch[key] = patchVal; } } return stateToPatch; } function createPersistedStore(key, initialValue, proto) { let version = 0; let syncVersion = 0; const storageValue = localStorage.getItem(key); let state$1 = state(proxy(storageValue ? mergeDeepState(JSON.parse(storageValue), Object.assign({}, initialValue)) : initialValue)); const writable2 = toStore(() => get$1(state$1), (newState) => set(state$1, proxy(newState))); effect_root(() => { user_effect(() => { if (version === 0) { version = Date.now(); } else if (version < syncVersion) { version = syncVersion; } else { version = Date.now(); channelEvent.emit(`sync:${key}`, { version, data: snapshot(get$1(state$1)) }); } localStorage.setItem(key, JSON.stringify(get$1(state$1))); }); }); channelEvent.on(`sync:${key}`, ({ version: v, data }) => { if (v <= version) return; syncVersion = v; set(state$1, proxy(data)); }); const store = Object.create(proto ? proto : null); Object.defineProperties(store, { $state: { get() { return get$1(state$1); } }, $subscribe: { value: writable2.subscribe }, $update: { value: writable2.update } }); for (const storeKey of Object.keys(get$1(state$1))) { Object.defineProperty(store, storeKey, { enumerable: true, get() { return get$1(state$1)[storeKey]; }, set(v) { get$1(state$1)[storeKey] = v; } }); } return store; } var ButtonStyle = /* @__PURE__ */ ((ButtonStyle2) => { ButtonStyle2["LEFT_PERCENT"] = "--pdl-btn-left-percent"; ButtonStyle2["TOP_PERCENT"] = "--pdl-btn-top-percent"; ButtonStyle2["PIXIV_BOOKMARK_LEFT_PERCENT"] = "--pdl-btn-pixiv-bookmark-left-percent"; ButtonStyle2["PIXIV_BOOKMARK_TOP_PERCENT"] = "--pdl-btn-pixiv-bookmark-top-percent"; ButtonStyle2["LEFT_PX"] = "--pdl-btn-left-px"; ButtonStyle2["TOP_PX"] = "--pdl-btn-top-px"; ButtonStyle2["PIXIV_BOOKMARK_LEFT_PX"] = "--pdl-btn-pixiv-bookmark-left-px"; ButtonStyle2["PIXIV_BOOKMARK_TOP_PX"] = "--pdl-btn-pixiv-bookmark-top-px"; return ButtonStyle2; })(ButtonStyle || {}); var BtnLengthUnit = /* @__PURE__ */ ((BtnLengthUnit2) => { BtnLengthUnit2[BtnLengthUnit2["PERCENT"] = 0] = "PERCENT"; BtnLengthUnit2[BtnLengthUnit2["PX"] = 1] = "PX"; return BtnLengthUnit2; })(BtnLengthUnit || {}); var BtnAlignX = /* @__PURE__ */ ((BtnAlignX2) => { BtnAlignX2[BtnAlignX2["LEFT"] = 0] = "LEFT"; BtnAlignX2[BtnAlignX2["RIGHT"] = 1] = "RIGHT"; return BtnAlignX2; })(BtnAlignX || {}); var BtnAlignY = /* @__PURE__ */ ((BtnAlignY2) => { BtnAlignY2[BtnAlignY2["TOP"] = 0] = "TOP"; BtnAlignY2[BtnAlignY2["BOTTOM"] = 1] = "BOTTOM"; return BtnAlignY2; })(BtnAlignY || {}); const buttonPosition = createPersistedStore( "pdl-button-position", { ["--pdl-btn-left-percent"]: legacyConfig["pdl-btn-left"] ?? 0, /* LEFT_PERCENT */ ["--pdl-btn-left-px"]: 0, /* LEFT_PX */ ["--pdl-btn-top-percent"]: legacyConfig["pdl-btn-top"] ?? 100, /* TOP_PERCENT */ ["--pdl-btn-top-px"]: 0, /* TOP_PX */ ["--pdl-btn-pixiv-bookmark-left-percent"]: legacyConfig["pdl-btn-self-bookmark-left"] ?? 100, /* PIXIV_BOOKMARK_LEFT_PERCENT */ ["--pdl-btn-pixiv-bookmark-left-px"]: 0, /* PIXIV_BOOKMARK_LEFT_PX */ ["--pdl-btn-pixiv-bookmark-top-percent"]: legacyConfig["pdl-btn-self-bookmark-top"] ?? 76, /* PIXIV_BOOKMARK_TOP_PERCENT */ ["--pdl-btn-pixiv-bookmark-top-px"]: 0, /* PIXIV_BOOKMARK_TOP_PX */ thumbnailBtnUnitX: 0, /* PERCENT */ thumbnailBtnUnitY: 0, /* PERCENT */ pixivBookmarkBtnUnitX: 0, /* PERCENT */ pixivBookmarkBtnUnitY: 0, /* PERCENT */ artworkBtnAlignX: 1, /* RIGHT */ artworkBtnAlignY: 0 /* TOP */ }, { setPosition(key, value) { const target = document.documentElement; const oldValue = getComputedStyle(target).getPropertyValue(key); if (!oldValue || oldValue !== value) { target.style.setProperty(key, value); } } } ); effect_root(() => { const btnPosTypeSelection = [ [ () => buttonPosition.thumbnailBtnUnitX, "--pdl-btn-left-px", /* LEFT_PX */ "--pdl-btn-left-percent" /* LEFT_PERCENT */ ], [ () => buttonPosition.thumbnailBtnUnitY, "--pdl-btn-top-px", /* TOP_PX */ "--pdl-btn-top-percent" /* TOP_PERCENT */ ], [ () => buttonPosition.pixivBookmarkBtnUnitX, "--pdl-btn-pixiv-bookmark-left-px", /* PIXIV_BOOKMARK_LEFT_PX */ "--pdl-btn-pixiv-bookmark-left-percent" /* PIXIV_BOOKMARK_LEFT_PERCENT */ ], [ () => buttonPosition.pixivBookmarkBtnUnitY, "--pdl-btn-pixiv-bookmark-top-px", /* PIXIV_BOOKMARK_TOP_PX */ "--pdl-btn-pixiv-bookmark-top-percent" /* PIXIV_BOOKMARK_TOP_PERCENT */ ] ]; btnPosTypeSelection.forEach(([getUnit, pxKey, percentKey]) => { user_effect(() => { let px; let percent; if (getUnit() === 1) { px = buttonPosition[pxKey] + "px"; percent = "0"; } else { px = "0px"; percent = String(buttonPosition[percentKey]); } buttonPosition.setPosition(pxKey, px); buttonPosition.setPosition(percentKey, percent); }); }); }); class ArtworkButton extends HTMLElement { constructor(props) { super(); __publicField(this, "props"); __privateAdd(this, _stickyContainer); __privateAdd(this, _thumbnailButton); __privateAdd(this, _unsubscribers, []); this.props = props; } static get tagNameLowerCase() { return "pdl-artwork-button"; } render() { if (this.shadowRoot) return; const shadowRoot = this.attachShadow({ mode: "open" }); const btnProps = { ...this.props }; const site = btnProps.site; shadowRoot.innerHTML = `<style>${wrapperStyle}</style><div class="button-wrapper${site ? " " + site : ""}"></div>`; delete btnProps.site; const thumbnailButton = new ThumbnailButton({ type: ThumbnailBtnType.Gallery, ...btnProps }); site && thumbnailButton.classList.add(site); const stickyContainer = shadowRoot.querySelector(".button-wrapper"); stickyContainer.appendChild(thumbnailButton); __privateSet(this, _thumbnailButton, thumbnailButton); __privateSet(this, _stickyContainer, stickyContainer); } connectedCallback() { this.render(); const unsubscribeX = toStore(() => buttonPosition.artworkBtnAlignX).subscribe( (artworkBtnAlignX) => { var _a2, _b2; if (artworkBtnAlignX === BtnAlignX.LEFT) { (_a2 = __privateGet(this, _stickyContainer)) == null ? undefined : _a2.classList.remove("rtl"); } else { (_b2 = __privateGet(this, _stickyContainer)) == null ? undefined : _b2.classList.add("rtl"); } } ); const unsbuscribeY = toStore(() => buttonPosition.artworkBtnAlignY).subscribe( (artworkBtnAlignY) => { var _a2, _b2; if (artworkBtnAlignY === BtnAlignY.TOP) { (_a2 = __privateGet(this, _thumbnailButton)) == null ? undefined : _a2.classList.remove("bottom"); } else { (_b2 = __privateGet(this, _thumbnailButton)) == null ? undefined : _b2.classList.add("bottom"); } } ); __privateGet(this, _unsubscribers).push(unsubscribeX, unsbuscribeY); } disconnectedCallback() { for (const unsubscriber of __privateGet(this, _unsubscribers)) { unsubscriber(); } __privateGet(this, _unsubscribers).length = 0; } } _stickyContainer = new WeakMap(); _thumbnailButton = new WeakMap(); _unsubscribers = new WeakMap(); customElements.define(ArtworkButton.tagNameLowerCase, ArtworkButton); var _GM_download = /* @__PURE__ */ (() => typeof GM_download != "undefined" ? GM_download : undefined)(); var _GM_info = /* @__PURE__ */ (() => typeof GM_info != "undefined" ? GM_info : undefined)(); var _GM_registerMenuCommand = /* @__PURE__ */ (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : undefined)(); var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : undefined)(); var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : undefined)(); const PUBLIC_VERSION = "5"; if (typeof window !== "undefined") (window.__svelte || (window.__svelte = { v: /* @__PURE__ */ new Set() })).v.add(PUBLIC_VERSION); const DRAWER_STORE_KEY = "drawerStore"; function initializeDrawerStore() { const drawerStore = drawerService(); return setContext(DRAWER_STORE_KEY, drawerStore); } function drawerService() { const { subscribe, set: set2, update: update2 } = writable({}); return { subscribe, set: set2, update: update2, /** Open the drawer. */ open: (newSettings) => update2(() => { return { open: true, ...newSettings }; }), /** Close the drawer. */ close: () => update2((d) => { d.open = false; return d; }) }; } const MODAL_STORE_KEY = "modalStore"; function getModalStore() { const modalStore = getContext(MODAL_STORE_KEY); if (!modalStore) throw new Error("modalStore is not initialized. Please ensure that `initializeStores()` is invoked in the root layout file of this app!"); return modalStore; } function initializeModalStore() { const modalStore = modalService(); return setContext(MODAL_STORE_KEY, modalStore); } function modalService() { const { subscribe, set: set2, update: update2 } = writable([]); return { subscribe, set: set2, update: update2, /** Append to end of queue. */ trigger: (modal) => update2((mStore) => { mStore.push(modal); return mStore; }), /** Remove first item in queue. */ close: () => update2((mStore) => { if (mStore.length > 0) mStore.shift(); return mStore; }), /** Remove all items from queue. */ clear: () => set2([]) }; } const toastDefaults = { message: "Missing Toast Message", autohide: true, timeout: 5e3 }; const TOAST_STORE_KEY = "toastStore"; function getToastStore() { const toastStore = getContext(TOAST_STORE_KEY); if (!toastStore) throw new Error("toastStore is not initialized. Please ensure that `initializeStores()` is invoked in the root layout file of this app!"); return toastStore; } function initializeToastStore() { const toastStore = toastService(); return setContext(TOAST_STORE_KEY, toastStore); } function randomUUID() { const random = Math.random(); return Number(random).toString(32); } function toastService() { const { subscribe, set: set2, update: update2 } = writable([]); const close = (id) => update2((tStore) => { if (tStore.length > 0) { const index2 = tStore.findIndex((t2) => t2.id === id); const selectedToast = tStore[index2]; if (selectedToast) { if (selectedToast.callback) selectedToast.callback({ id, status: "closed" }); if (selectedToast.timeoutId) clearTimeout(selectedToast.timeoutId); tStore.splice(index2, 1); } } return tStore; }); function handleAutoHide(toast2) { if (toast2.autohide === true) { return setTimeout(() => { close(toast2.id); }, toast2.timeout); } } return { subscribe, close, /** Add a new toast to the queue. */ trigger: (toast2) => { const id = randomUUID(); update2((tStore) => { if (toast2 && toast2.callback) toast2.callback({ id, status: "queued" }); if (toast2.hideDismiss) toast2.autohide = true; const tMerged = { ...toastDefaults, ...toast2, id }; tMerged.timeoutId = handleAutoHide(tMerged); tStore.push(tMerged); return tStore; }); return id; }, /** Remain visible on hover */ freeze: (index2) => update2((tStore) => { if (tStore.length > 0) clearTimeout(tStore[index2].timeoutId); return tStore; }), /** Cancel remain visible on leave */ unfreeze: (index2) => update2((tStore) => { if (tStore.length > 0) tStore[index2].timeoutId = handleAutoHide(tStore[index2]); return tStore; }), /** Remove all toasts from queue */ clear: () => set2([]) }; } function initializeStores() { initializeModalStore(); initializeToastStore(); initializeDrawerStore(); } const stores = {}; function getStorage(type) { return type === "local" ? localStorage : sessionStorage; } function localStorageStore(key, initialValue, options) { const serializer = JSON; const storageType = "local"; function updateStorage(key2, value) { getStorage(storageType).setItem(key2, serializer.stringify(value)); } if (!stores[key]) { const store = writable(initialValue, (set3) => { const json = getStorage(storageType).getItem(key); if (json) { set3(serializer.parse(json)); } { const handleStorage = (event2) => { if (event2.key === key) set3(event2.newValue ? serializer.parse(event2.newValue) : null); }; window.addEventListener("storage", handleStorage); return () => window.removeEventListener("storage", handleStorage); } }); const { subscribe, set: set2 } = store; stores[key] = { set(value) { updateStorage(key, value); set2(value); }, update(updater) { const value = updater(get(store)); updateStorage(key, value); set2(value); }, subscribe }; } return stores[key]; } localStorageStore("modeOsPrefers", false); localStorageStore("modeUserPrefers", undefined); localStorageStore("modeCurrent", false); const reducedMotionQuery = "(prefers-reduced-motion: reduce)"; function prefersReducedMotion() { return window.matchMedia(reducedMotionQuery).matches; } const prefersReducedMotionStore = readable(prefersReducedMotion(), (set2) => { { const setReducedMotion = (event2) => { set2(event2.matches); }; const mediaQueryList = window.matchMedia(reducedMotionQuery); mediaQueryList.addEventListener("change", setReducedMotion); return () => { mediaQueryList.removeEventListener("change", setReducedMotion); }; } }); function focusTrap(node, enabled) { const elemWhitelist = 'a[href]:not([tabindex="-1"]), button:not([tabindex="-1"]), input:not([tabindex="-1"]), textarea:not([tabindex="-1"]), select:not([tabindex="-1"]), details:not([tabindex="-1"]), [tabindex]:not([tabindex="-1"])'; let elemFirst; let elemLast; function onFirstElemKeydown(e) { if (e.shiftKey && e.code === "Tab") { e.preventDefault(); elemLast.focus(); } } function onLastElemKeydown(e) { if (!e.shiftKey && e.code === "Tab") { e.preventDefault(); elemFirst.focus(); } } const sortByTabIndex = (focusableElems) => { return focusableElems.filter((elem) => elem.tabIndex >= 0).sort((a, b) => { if (a.tabIndex === 0 && b.tabIndex > 0) return 1; else if (a.tabIndex > 0 && b.tabIndex === 0) return -1; else return a.tabIndex - b.tabIndex; }); }; const getFocusTrapTarget = (elemFirst2) => { const focusindexElements = [...node.querySelectorAll("[data-focusindex]")]; if (!focusindexElements || focusindexElements.length === 0) return elemFirst2; return focusindexElements.sort((a, b) => { return +a.dataset.focusindex - +b.dataset.focusindex; })[0] || elemFirst2; }; const onScanElements = (fromObserver) => { if (enabled === false) return; const focusableElems = sortByTabIndex(Array.from(node.querySelectorAll(elemWhitelist))); if (focusableElems.length) { elemFirst = focusableElems[0]; elemLast = focusableElems[focusableElems.length - 1]; if (!fromObserver) getFocusTrapTarget(elemFirst).focus(); elemFirst.addEventListener("keydown", onFirstElemKeydown); elemLast.addEventListener("keydown", onLastElemKeydown); } }; onScanElements(false); function onCleanUp() { if (elemFirst) elemFirst.removeEventListener("keydown", onFirstElemKeydown); if (elemLast) elemLast.removeEventListener("keydown", onLastElemKeydown); } const onObservationChange = (mutationRecords, observer3) => { if (mutationRecords.length) { onCleanUp(); onScanElements(true); } return observer3; }; const observer2 = new MutationObserver(onObservationChange); observer2.observe(node, { childList: true, subtree: true }); return { update(newArgs) { enabled = newArgs; newArgs ? onScanElements(false) : onCleanUp(); }, destroy() { onCleanUp(); observer2.disconnect(); } }; } enable_legacy_mode_flag(); const linear = (x) => x; function cubic_out(t2) { const f = t2 - 1; return f * f * f + 1; } function split_css_unit(value) { const split = typeof value === "string" && value.match(/^\s*(-?[\d.]+)([^\s]*)\s*$/); return split ? [parseFloat(split[1]), split[2] || "px"] : [ /** @type {number} */ value, "px" ]; } function fade(node, { delay = 0, duration = 400, easing = linear } = {}) { const o = +getComputedStyle(node).opacity; return { delay, duration, easing, css: (t2) => `opacity: ${t2 * o}` }; } function fly(node, { delay = 0, duration = 400, easing = cubic_out, x = 0, y = 0, opacity = 0 } = {}) { const style = getComputedStyle(node); const target_opacity = +style.opacity; const transform = style.transform === "none" ? "" : style.transform; const od = target_opacity * (1 - opacity); const [x_value, x_unit] = split_css_unit(x); const [y_value, y_unit] = split_css_unit(y); return { delay, duration, easing, css: (t2, u) => ` transform: ${transform} translate(${(1 - t2) * x_value}${x_unit}, ${(1 - t2) * y_value}${y_unit}); opacity: ${target_opacity - od * u}` }; } function slide(node, { delay = 0, duration = 400, easing = cubic_out, axis = "y" } = {}) { const style = getComputedStyle(node); const opacity = +style.opacity; const primary_property = axis === "y" ? "height" : "width"; const primary_property_value = parseFloat(style[primary_property]); const secondary_properties = axis === "y" ? ["top", "bottom"] : ["left", "right"]; const capitalized_secondary_properties = secondary_properties.map( (e) => ( /** @type {'Left' | 'Right' | 'Top' | 'Bottom'} */ `${e[0].toUpperCase()}${e.slice(1)}` ) ); const padding_start_value = parseFloat(style[`padding${capitalized_secondary_properties[0]}`]); const padding_end_value = parseFloat(style[`padding${capitalized_secondary_properties[1]}`]); const margin_start_value = parseFloat(style[`margin${capitalized_secondary_properties[0]}`]); const margin_end_value = parseFloat(style[`margin${capitalized_secondary_properties[1]}`]); const border_width_start_value = parseFloat( style[`border${capitalized_secondary_properties[0]}Width`] ); const border_width_end_value = parseFloat( style[`border${capitalized_secondary_properties[1]}Width`] ); return { delay, duration, easing, css: (t2) => `overflow: hidden;opacity: ${Math.min(t2 * 20, 1) * opacity};${primary_property}: ${t2 * primary_property_value}px;padding-${secondary_properties[0]}: ${t2 * padding_start_value}px;padding-${secondary_properties[1]}: ${t2 * padding_end_value}px;margin-${secondary_properties[0]}: ${t2 * margin_start_value}px;margin-${secondary_properties[1]}: ${t2 * margin_end_value}px;border-${secondary_properties[0]}-width: ${t2 * border_width_start_value}px;border-${secondary_properties[1]}-width: ${t2 * border_width_end_value}px;min-${primary_property}: 0` }; } function scale(node, { delay = 0, duration = 400, easing = cubic_out, start = 0, opacity = 0 } = {}) { const style = getComputedStyle(node); const target_opacity = +style.opacity; const transform = style.transform === "none" ? "" : style.transform; const sd = 1 - start; const od = target_opacity * (1 - opacity); return { delay, duration, easing, css: (_t, u) => ` transform: ${transform} scale(${1 - sd * u}); opacity: ${target_opacity - od * u} ` }; } var root$m = /* @__PURE__ */ template(`<div data-testid="accordion"><!></div>`); function Accordion($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const [$$stores, $$cleanup] = setup_stores(); const $prefersReducedMotionStore = () => store_get(prefersReducedMotionStore, "$prefersReducedMotionStore", $$stores); const classesBase = mutable_state(); let autocollapse = prop($$props, "autocollapse", 8, false); let width = prop($$props, "width", 8, "w-full"); let spacing = prop($$props, "spacing", 8, "space-y-1"); let disabled = prop($$props, "disabled", 8, false); let padding = prop($$props, "padding", 8, "py-2 px-4"); let hover = prop($$props, "hover", 8, "hover:bg-primary-hover-token"); let rounded = prop($$props, "rounded", 8, "rounded-container-token"); let caretOpen = prop($$props, "caretOpen", 8, "rotate-180"); let caretClosed = prop($$props, "caretClosed", 8, ""); let regionControl = prop($$props, "regionControl", 8, ""); let regionPanel = prop($$props, "regionPanel", 8, "space-y-4"); let regionCaret = prop($$props, "regionCaret", 8, ""); let transitions = prop($$props, "transitions", 24, () => !$prefersReducedMotionStore()); let transitionIn = prop($$props, "transitionIn", 8, slide); let transitionInParams = prop($$props, "transitionInParams", 24, () => ({ duration: 200 })); let transitionOut = prop($$props, "transitionOut", 8, slide); let transitionOutParams = prop($$props, "transitionOutParams", 24, () => ({ duration: 200 })); const active = writable(null); setContext("active", active); setContext("autocollapse", autocollapse()); setContext("disabled", disabled()); setContext("padding", padding()); setContext("hover", hover()); setContext("rounded", rounded()); setContext("caretOpen", caretOpen()); setContext("caretClosed", caretClosed()); setContext("regionControl", regionControl()); setContext("regionPanel", regionPanel()); setContext("regionCaret", regionCaret()); setContext("transitions", transitions()); setContext("transitionIn", transitionIn()); setContext("transitionInParams", transitionInParams()); setContext("transitionOut", transitionOut()); setContext("transitionOutParams", transitionOutParams()); legacy_pre_effect( () => (deep_read_state(width()), deep_read_state(spacing()), deep_read_state($$sanitized_props)), () => { set(classesBase, `${width()} ${spacing()} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect_reset(); init(); var div = root$m(); var node = child(div); slot(node, $$props, "default", {}, null); reset(div); template_effect(() => set_class(div, `accordion ${get$1(classesBase) ?? ""}`)); append($$anchor, div); pop(); $$cleanup(); } function dynamicTransition(node, dynParams) { const { transition: transition2, params, enabled } = dynParams; if (enabled) return transition2(node, params); if ("duration" in params) return transition2(node, { duration: 0 }); return { duration: 0 }; } var root_1$j = /* @__PURE__ */ template(`<div class="accordion-lead"><!></div>`); var root_3$b = /* @__PURE__ */ template(`<div><!></div>`); var root_8$1 = /* @__PURE__ */ template(`<div><!></div>`); var root_9$5 = /* @__PURE__ */ template(`<div role="region"><!></div>`); var root$l = /* @__PURE__ */ template(`<div data-testid="accordion-item"><button type="button"><!> <div class="accordion-summary flex-1"><!></div> <!></button> <!></div>`); function AccordionItem($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const [$$stores, $$cleanup] = setup_stores(); const $active = () => store_get(active(), "$active", $$stores); const openState = mutable_state(); const classesBase = mutable_state(); const classesControl = mutable_state(); const classesCaretState = mutable_state(); const classesControlCaret = mutable_state(); const classesControlIcons = mutable_state(); const classesPanel = mutable_state(); const dispatch = createEventDispatcher(); let open = prop($$props, "open", 12, false); let id = prop($$props, "id", 24, () => String(Math.random())); const cBase = ""; const cControl = "text-start w-full flex items-center space-x-4"; const cControlIcons = "fill-current w-3 transition-transform duration-[200ms]"; const cPanel = ""; let autocollapse = prop($$props, "autocollapse", 24, () => getContext("autocollapse")); let active = prop($$props, "active", 24, () => getContext("active")); let disabled = prop($$props, "disabled", 24, () => getContext("disabled")); let padding = prop($$props, "padding", 24, () => getContext("padding")); let hover = prop($$props, "hover", 24, () => getContext("hover")); let rounded = prop($$props, "rounded", 24, () => getContext("rounded")); let caretOpen = prop($$props, "caretOpen", 24, () => getContext("caretOpen")); let caretClosed = prop($$props, "caretClosed", 24, () => getContext("caretClosed")); let regionControl = prop($$props, "regionControl", 24, () => getContext("regionControl")); let regionPanel = prop($$props, "regionPanel", 24, () => getContext("regionPanel")); let regionCaret = prop($$props, "regionCaret", 24, () => getContext("regionCaret")); let transitions = prop($$props, "transitions", 24, () => getContext("transitions")); let transitionIn = prop($$props, "transitionIn", 24, () => getContext("transitionIn")); let transitionInParams = prop($$props, "transitionInParams", 24, () => getContext("transitionInParams")); let transitionOut = prop($$props, "transitionOut", 24, () => getContext("transitionOut")); let transitionOutParams = prop($$props, "transitionOutParams", 24, () => getContext("transitionOutParams")); const svgCaretIcon = ` <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 448 512"> <path d="M201.4 374.6c12.5 12.5 32.8 12.5 45.3 0l160-160c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L224 306.7 86.6 169.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3l160 160z" /> </svg>`; function setActive(event2) { if (autocollapse() === true) { active().set(id()); } else { open(!open()); } onToggle(event2); } function onToggle(event2) { const currentOpenState = autocollapse() ? $active() === id() : open(); dispatch("toggle", { event: event2, id: id(), panelId: `accordion-panel-${id()}`, open: currentOpenState, autocollapse: autocollapse() }); } if (autocollapse() && open()) setActive(); legacy_pre_effect( () => (deep_read_state(open()), deep_read_state(autocollapse())), () => { if (open() && autocollapse()) setActive(); } ); legacy_pre_effect( () => (deep_read_state(autocollapse()), $active(), deep_read_state(id()), deep_read_state(open())), () => { set(openState, autocollapse() ? $active() === id() : open()); } ); legacy_pre_effect(() => deep_read_state($$sanitized_props), () => { set(classesBase, `${cBase} ${$$sanitized_props.class ?? ""}`); }); legacy_pre_effect( () => (deep_read_state(padding()), deep_read_state(hover()), deep_read_state(rounded()), deep_read_state(regionControl())), () => { set(classesControl, `${cControl} ${padding()} ${hover()} ${rounded()} ${regionControl()}`); } ); legacy_pre_effect( () => (get$1(openState), deep_read_state(caretOpen()), deep_read_state(caretClosed())), () => { set(classesCaretState, get$1(openState) ? caretOpen() : caretClosed()); } ); legacy_pre_effect( () => (deep_read_state(regionCaret()), get$1(classesCaretState)), () => { set(classesControlCaret, `${cControlIcons} ${regionCaret()} ${get$1(classesCaretState)}`); } ); legacy_pre_effect(() => deep_read_state(regionCaret()), () => { set(classesControlIcons, `${cControlIcons} ${regionCaret()}`); }); legacy_pre_effect( () => (deep_read_state(padding()), deep_read_state(rounded()), deep_read_state(regionPanel())), () => { set(classesPanel, `${cPanel} ${padding()} ${rounded()} ${regionPanel()}`); } ); legacy_pre_effect_reset(); init(); var div = root$l(); var button2 = child(div); var node = child(button2); { var consequent = ($$anchor2) => { var div_1 = root_1$j(); var node_1 = child(div_1); slot(node_1, $$props, "lead", {}, null); reset(div_1); append($$anchor2, div_1); }; if_block(node, ($$render) => { if ($$slots.lead) $$render(consequent); }); } var div_2 = sibling(node, 2); var node_2 = child(div_2); slot(node_2, $$props, "summary", {}, ($$anchor2) => { var text$1 = text("(summary)"); append($$anchor2, text$1); }); reset(div_2); var node_3 = sibling(div_2, 2); { var consequent_2 = ($$anchor2) => { var div_3 = root_3$b(); var node_4 = child(div_3); { var consequent_1 = ($$anchor3) => { var fragment = comment(); var node_5 = first_child(fragment); slot(node_5, $$props, "iconClosed", {}, ($$anchor4) => { var fragment_1 = comment(); var node_6 = first_child(fragment_1); html(node_6, () => svgCaretIcon); append($$anchor4, fragment_1); }); append($$anchor3, fragment); }; var alternate = ($$anchor3) => { var fragment_2 = comment(); var node_7 = first_child(fragment_2); slot(node_7, $$props, "iconOpen", {}, ($$anchor4) => { var fragment_3 = comment(); var node_8 = first_child(fragment_3); html(node_8, () => svgCaretIcon); append($$anchor4, fragment_3); }); append($$anchor3, fragment_2); }; if_block(node_4, ($$render) => { if (get$1(openState)) $$render(consequent_1); else $$render(alternate, false); }); } reset(div_3); template_effect(() => set_class(div_3, `accordion-summary-icons ${get$1(classesControlIcons) ?? ""}`)); append($$anchor2, div_3); }; var alternate_1 = ($$anchor2) => { var div_4 = root_8$1(); var node_9 = child(div_4); html(node_9, () => svgCaretIcon); reset(div_4); template_effect(() => set_class(div_4, `accordion-summary-caret ${get$1(classesControlCaret) ?? ""}`)); append($$anchor2, div_4); }; if_block(node_3, ($$render) => { if ($$slots.iconClosed || $$slots.iconOpen) $$render(consequent_2); else $$render(alternate_1, false); }); } reset(button2); var node_10 = sibling(button2, 2); { var consequent_3 = ($$anchor2) => { var div_5 = root_9$5(); var node_11 = child(div_5); slot(node_11, $$props, "content", {}, ($$anchor3) => { var text_1 = text("(content)"); append($$anchor3, text_1); }); reset(div_5); template_effect(() => { set_class(div_5, `accordion-panel ${get$1(classesPanel) ?? ""}`); set_attribute(div_5, "id", `accordion-panel-${id() ?? ""}`); set_attribute(div_5, "aria-hidden", !get$1(openState)); set_attribute(div_5, "aria-labelledby", id()); }); transition(1, div_5, () => dynamicTransition, () => ({ transition: transitionIn(), params: transitionInParams(), enabled: transitions() })); transition(2, div_5, () => dynamicTransition, () => ({ transition: transitionOut(), params: transitionOutParams(), enabled: transitions() })); append($$anchor2, div_5); }; if_block(node_10, ($$render) => { if (get$1(openState)) $$render(consequent_3); }); } reset(div); template_effect(() => { set_class(div, `accordion-item ${get$1(classesBase) ?? ""}`); set_class(button2, `accordion-control ${get$1(classesControl) ?? ""}`); set_attribute(button2, "id", id()); set_attribute(button2, "aria-expanded", get$1(openState)); set_attribute(button2, "aria-controls", `accordion-panel-${id() ?? ""}`); button2.disabled = disabled(); }); event("click", button2, setActive); event("click", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keydown", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keyup", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keypress", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor, div); pop(); $$cleanup(); } var root_1$i = /* @__PURE__ */ template(`<div><!></div>`); var root_2$8 = /* @__PURE__ */ template(`<div><!></div>`); var root_3$a = /* @__PURE__ */ template(`<div><!></div>`); var root$k = /* @__PURE__ */ template(`<div data-testid="app-bar" role="toolbar"><div><!> <div><!></div> <!></div> <!></div>`); function AppBar($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const classesBase = mutable_state(); const classesRowMain = mutable_state(); const classesRowHeadline = mutable_state(); const classesSlotLead = mutable_state(); const classesSlotDefault = mutable_state(); const classesSlotTrail = mutable_state(); let background = prop($$props, "background", 8, "bg-surface-100-800-token"); let border = prop($$props, "border", 8, ""); let padding = prop($$props, "padding", 8, "p-4"); let shadow = prop($$props, "shadow", 8, ""); let spacing = prop($$props, "spacing", 8, "space-y-4"); let gridColumns = prop($$props, "gridColumns", 8, "grid-cols-[auto_1fr_auto]"); let gap = prop($$props, "gap", 8, "gap-4"); let regionRowMain = prop($$props, "regionRowMain", 8, ""); let regionRowHeadline = prop($$props, "regionRowHeadline", 8, ""); let slotLead = prop($$props, "slotLead", 8, ""); let slotDefault = prop($$props, "slotDefault", 8, ""); let slotTrail = prop($$props, "slotTrail", 8, ""); let label = prop($$props, "label", 8, ""); let labelledby = prop($$props, "labelledby", 8, ""); const cBase = "flex flex-col"; const cRowMain = "grid items-center"; const cRowHeadline = ""; const cSlotLead = "flex-none flex justify-between items-center"; const cSlotDefault = "flex-auto"; const cSlotTrail = "flex-none flex items-center space-x-4"; legacy_pre_effect( () => (deep_read_state(background()), deep_read_state(border()), deep_read_state(spacing()), deep_read_state(padding()), deep_read_state(shadow()), deep_read_state($$sanitized_props)), () => { set(classesBase, `${cBase} ${background()} ${border()} ${spacing()} ${padding()} ${shadow()} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect( () => (deep_read_state(gridColumns()), deep_read_state(gap()), deep_read_state(regionRowMain())), () => { set(classesRowMain, `${cRowMain} ${gridColumns()} ${gap()} ${regionRowMain()}`); } ); legacy_pre_effect(() => deep_read_state(regionRowHeadline()), () => { set(classesRowHeadline, `${cRowHeadline} ${regionRowHeadline()}`); }); legacy_pre_effect(() => deep_read_state(slotLead()), () => { set(classesSlotLead, `${cSlotLead} ${slotLead()}`); }); legacy_pre_effect(() => deep_read_state(slotDefault()), () => { set(classesSlotDefault, `${cSlotDefault} ${slotDefault()}`); }); legacy_pre_effect(() => deep_read_state(slotTrail()), () => { set(classesSlotTrail, `${cSlotTrail} ${slotTrail()}`); }); legacy_pre_effect_reset(); init(); var div = root$k(); var div_1 = child(div); var node = child(div_1); { var consequent = ($$anchor2) => { var div_2 = root_1$i(); var node_1 = child(div_2); slot(node_1, $$props, "lead", {}, null); reset(div_2); template_effect(() => set_class(div_2, `app-bar-slot-lead ${get$1(classesSlotLead) ?? ""}`)); append($$anchor2, div_2); }; if_block(node, ($$render) => { if ($$slots.lead) $$render(consequent); }); } var div_3 = sibling(node, 2); var node_2 = child(div_3); slot(node_2, $$props, "default", {}, null); reset(div_3); var node_3 = sibling(div_3, 2); { var consequent_1 = ($$anchor2) => { var div_4 = root_2$8(); var node_4 = child(div_4); slot(node_4, $$props, "trail", {}, null); reset(div_4); template_effect(() => set_class(div_4, `app-bar-slot-trail ${get$1(classesSlotTrail) ?? ""}`)); append($$anchor2, div_4); }; if_block(node_3, ($$render) => { if ($$slots.trail) $$render(consequent_1); }); } reset(div_1); var node_5 = sibling(div_1, 2); { var consequent_2 = ($$anchor2) => { var div_5 = root_3$a(); var node_6 = child(div_5); slot(node_6, $$props, "headline", {}, null); reset(div_5); template_effect(() => set_class(div_5, `app-bar-row-headline ${get$1(classesRowHeadline) ?? ""}`)); append($$anchor2, div_5); }; if_block(node_5, ($$render) => { if ($$slots.headline) $$render(consequent_2); }); } reset(div); template_effect(() => { set_class(div, `app-bar ${get$1(classesBase) ?? ""}`); set_attribute(div, "aria-label", label()); set_attribute(div, "aria-labelledby", labelledby()); set_class(div_1, `app-bar-row-main ${get$1(classesRowMain) ?? ""}`); set_class(div_3, `app-bar-slot-default ${get$1(classesSlotDefault) ?? ""}`); }); append($$anchor, div); pop(); } var root$j = /* @__PURE__ */ template(`<div data-testid="file-button"><div class="w-0 h-0 overflow-hidden"><input></div> <button type="button"><!></button></div>`); function FileButton($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); const $$restProps = legacy_rest_props($$sanitized_props, [ "files", "fileInput", "name", "width", "button" ]); push($$props, false); const classesBase = mutable_state(); const classesButton = mutable_state(); let files = prop($$props, "files", 28, () => undefined); let fileInput = prop($$props, "fileInput", 28, () => undefined); let name = prop($$props, "name", 8); let width = prop($$props, "width", 8, ""); let button2 = prop($$props, "button", 8, "btn variant-filled"); function onButtonClick() { if (fileInput()) fileInput().click(); } function prunedRestProps() { delete $$restProps.class; return $$restProps; } legacy_pre_effect(() => deep_read_state($$sanitized_props), () => { set(classesBase, `${$$sanitized_props.class ?? ""}`); }); legacy_pre_effect( () => (deep_read_state(button2()), deep_read_state(width())), () => { set(classesButton, `${button2()} ${width()}`); } ); legacy_pre_effect_reset(); init(); var div = root$j(); var div_1 = child(div); var input = child(div_1); remove_input_defaults(input); const spread_with_call = /* @__PURE__ */ derived_safe_equal(prunedRestProps); let attributes; bind_this(input, ($$value) => fileInput($$value), () => fileInput()); reset(div_1); var button_1 = sibling(div_1, 2); var node = child(button_1); slot(node, $$props, "default", {}, ($$anchor2) => { var text$1 = text("Select a File"); append($$anchor2, text$1); }); reset(button_1); reset(div); template_effect(() => { set_class(div, `file-button ${get$1(classesBase) ?? ""}`); attributes = set_attributes(input, attributes, { type: "file", name: name(), ...get$1(spread_with_call) }); set_class(button_1, `file-button-btn ${get$1(classesButton) ?? ""}`); button_1.disabled = $$restProps.disabled; }); bind_files(input, files); event("change", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("click", button_1, onButtonClick); event("keydown", button_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keyup", button_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keypress", button_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor, div); pop(); } function cubicOut(t2) { const f = t2 - 1; return f * f * f + 1; } function flip(node, { from, to }, params = {}) { var { delay = 0, duration = (d) => Math.sqrt(d) * 120, easing = cubicOut } = params; var style = getComputedStyle(node); var transform = style.transform === "none" ? "" : style.transform; var [ox, oy] = style.transformOrigin.split(" ").map(parseFloat); ox /= node.clientWidth; oy /= node.clientHeight; var zoom = get_zoom(node); var sx = node.clientWidth / to.width / zoom; var sy = node.clientHeight / to.height / zoom; var fx = from.left + from.width * ox; var fy = from.top + from.height * oy; var tx = to.left + to.width * ox; var ty = to.top + to.height * oy; var dx = (fx - tx) * sx; var dy = (fy - ty) * sy; var dsx = from.width / to.width; var dsy = from.height / to.height; return { delay, duration: typeof duration === "function" ? duration(Math.sqrt(dx * dx + dy * dy)) : duration, easing, css: (t2, u) => { var x = u * dx; var y = u * dy; var sx2 = t2 + u * dsx; var sy2 = t2 + u * dsy; return `transform: ${transform} translate(${x}px, ${y}px) scale(${sx2}, ${sy2});`; } }; } function get_zoom(element) { if ("currentCSSZoom" in element) { return ( /** @type {number} */ element.currentCSSZoom ); } var current = element; var zoom = 1; while (current !== null) { zoom *= +getComputedStyle(current).zoom; current = /** @type {Element | null} */ current.parentElement; } return zoom; } var root_1$h = /* @__PURE__ */ template(`<option> </option>`); var root_3$9 = /* @__PURE__ */ template(`<div><button type="button"><span> </span> <span>✕</span></button></div>`); var root_2$7 = /* @__PURE__ */ template(`<div></div>`); var root$i = /* @__PURE__ */ template(`<div><div class="h-0 overflow-hidden"><select multiple tabindex="-1"></select></div> <div><input> <!></div></div>`); function InputChip($$anchor, $$props) { var _a2; const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); const $$restProps = legacy_rest_props($$sanitized_props, [ "addChip", "removeChip", "input", "name", "value", "whitelist", "max", "minlength", "maxlength", "allowUpperCase", "allowDuplicates", "validation", "duration", "required", "chips", "invalid", "padding", "rounded", "regionChipWrapper", "regionChipList", "regionInput", "label", "transitions", "listTransitionIn", "listTransitionInParams", "listTransitionOut", "listTransitionOutParams", "chipTransitionIn", "chipTransitionInParams", "chipTransitionOut", "chipTransitionOutParams" ]); push($$props, false); const [$$stores, $$cleanup] = setup_stores(); const $prefersReducedMotionStore = () => store_get(prefersReducedMotionStore, "$prefersReducedMotionStore", $$stores); const classesInvalid = mutable_state(); const classesBase = mutable_state(); const classesChipWrapper = mutable_state(); const classesChipList = mutable_state(); const classesInput = mutable_state(); const dispatch = createEventDispatcher(); let input = prop($$props, "input", 12, ""); let name = prop($$props, "name", 8); let value = prop($$props, "value", 28, () => []); let whitelist = prop($$props, "whitelist", 24, () => []); let max = prop($$props, "max", 24, () => -1); let minlength = prop($$props, "minlength", 24, () => -1); let maxlength = prop($$props, "maxlength", 24, () => -1); let allowUpperCase = prop($$props, "allowUpperCase", 8, false); let allowDuplicates = prop($$props, "allowDuplicates", 8, false); let validation = prop($$props, "validation", 8, () => true); let duration = prop($$props, "duration", 8, 150); let required = prop($$props, "required", 8, false); let chips = prop($$props, "chips", 8, "variant-filled"); let invalid = prop($$props, "invalid", 8, "input-error"); let padding = prop($$props, "padding", 8, "p-2"); let rounded = prop($$props, "rounded", 8, "rounded-container-token"); let regionChipWrapper = prop($$props, "regionChipWrapper", 8, ""); let regionChipList = prop($$props, "regionChipList", 8, ""); let regionInput = prop($$props, "regionInput", 8, ""); let label = prop($$props, "label", 8, "Chips select"); let transitions = prop($$props, "transitions", 24, () => !$prefersReducedMotionStore()); let listTransitionIn = prop($$props, "listTransitionIn", 8, fly); let listTransitionInParams = prop($$props, "listTransitionInParams", 24, () => ({ duration: 150, opacity: 0, y: -20 })); let listTransitionOut = prop($$props, "listTransitionOut", 8, fly); let listTransitionOutParams = prop($$props, "listTransitionOutParams", 24, () => ({ duration: 150, opacity: 0, y: -20 })); let chipTransitionIn = prop($$props, "chipTransitionIn", 8, scale); let chipTransitionInParams = prop($$props, "chipTransitionInParams", 24, () => ({ duration: 150, opacity: 0 })); let chipTransitionOut = prop($$props, "chipTransitionOut", 8, scale); let chipTransitionOutParams = prop($$props, "chipTransitionOutParams", 24, () => ({ duration: 150, opacity: 0 })); const cBase = "textarea cursor-pointer"; const cChipWrapper = "space-y-4"; const cChipList = "flex flex-wrap gap-2"; const cInputField = "unstyled bg-transparent border-0 !ring-0 p-0 w-full"; let inputValid = mutable_state(true); let chipValues = mutable_state(((_a2 = value()) == null ? undefined : _a2.map((val) => { return { val, id: Math.random() }; })) || []); function resetFormHandler() { value([]); } let selectElement = mutable_state(); onMount(() => { if (!get$1(selectElement).form) return; const externalForm = get$1(selectElement).form; externalForm.addEventListener("reset", resetFormHandler); return () => { externalForm.removeEventListener("reset", resetFormHandler); }; }); function validateCustom(chip) { return validation() === undefined || validation()(chip); } function validateCount() { return max() === -1 || value().length < max(); } function validateLength(chip) { return (minlength() === -1 || chip.length >= minlength()) && (maxlength() === -1 || chip.length <= maxlength()); } function validateWhiteList(chip) { return whitelist().length === 0 || whitelist().includes(chip); } function validateDuplicates(chip) { return allowDuplicates() || !value().includes(chip); } function validate(chip = "") { if (!chip && !input()) return false; chip = chip !== "" ? chip.trim() : input().trim(); return validateCustom(chip) && validateCount() && validateLength(chip) && validateWhiteList(chip) && validateDuplicates(chip); } function addChipCommon(chip) { chip = allowUpperCase() ? chip : chip.toLowerCase(); value().push(chip); value(value()); get$1(chipValues).push({ val: chip, id: Math.random() }); set(chipValues, get$1(chipValues)); } function removeChipCommon(chip) { let chipIndex = value().indexOf(chip); value().splice(chipIndex, 1); value(value()); get$1(chipValues).splice(chipIndex, 1); set(chipValues, get$1(chipValues)); } function onKeyHandler(event2) { if (event2.key !== "Enter") return; event2.preventDefault(); set(inputValid, validate()); if (get$1(inputValid) === false) { dispatch("invalid", { event: event2, input: input() }); return; } addChipCommon(input()); dispatch("add", { event: event2, chipIndex: value().length - 1, chipValue: input() }); input(""); } function removeChipInternally(event2, chipIndex, chipValue) { if ($$restProps.disabled) return; removeChipCommon(chipValue); dispatch("remove", { event: event2, chipIndex, chipValue }); } function addChip(chip) { set(inputValid, validate(chip)); if (get$1(inputValid) === false) { dispatch("invalidManually", { input: chip }); return; } addChipCommon(chip); dispatch("addManually", { chipIndex: value().length - 1, chipValue: chip }); } function removeChip(chip) { if ($$restProps.disabled) return; removeChipCommon(chip); dispatch("removeManually", { chipValue: chip }); } function prunedRestProps() { delete $$restProps.class; return $$restProps; } legacy_pre_effect( () => (get$1(inputValid), deep_read_state(invalid())), () => { set(classesInvalid, get$1(inputValid) === false ? invalid() : ""); } ); legacy_pre_effect( () => (deep_read_state(padding()), deep_read_state(rounded()), deep_read_state($$sanitized_props), get$1(classesInvalid)), () => { set(classesBase, `${cBase} ${padding()} ${rounded()} ${$$sanitized_props.class ?? ""} ${get$1(classesInvalid)}`); } ); legacy_pre_effect(() => deep_read_state(regionChipWrapper()), () => { set(classesChipWrapper, `${cChipWrapper} ${regionChipWrapper()}`); }); legacy_pre_effect(() => deep_read_state(regionChipList()), () => { set(classesChipList, `${cChipList} ${regionChipList()}`); }); legacy_pre_effect(() => deep_read_state(regionInput()), () => { set(classesInput, `${cInputField} ${regionInput()}`); }); legacy_pre_effect( () => (get$1(chipValues), deep_read_state(value())), () => { var _a3; set(chipValues, ((_a3 = value()) == null ? undefined : _a3.map((val, i) => { var _a4; if (((_a4 = get$1(chipValues)[i]) == null ? undefined : _a4.val) === val) return get$1(chipValues)[i]; return { id: Math.random(), val }; })) || []); } ); legacy_pre_effect_reset(); init(); var div = root$i(); var div_1 = child(div); var select = child(div_1); template_effect(() => { value(); invalidate_inner_signals(() => { get$1(selectElement); name(); required(); label(); }); }); each(select, 5, value, index, ($$anchor2, option) => { var option_1 = root_1$h(); var option_1_value = {}; var text2 = child(option_1, true); reset(option_1); template_effect(() => { if (option_1_value !== (option_1_value = get$1(option))) { option_1.value = null == (option_1.__value = get$1(option)) ? "" : get$1(option); } set_text(text2, get$1(option)); }); append($$anchor2, option_1); }); reset(select); bind_this(select, ($$value) => set(selectElement, $$value), () => get$1(selectElement)); reset(div_1); var div_2 = sibling(div_1, 2); var input_1 = child(div_2); remove_input_defaults(input_1); const spread_with_call = /* @__PURE__ */ derived_safe_equal(prunedRestProps); let attributes; var node = sibling(input_1, 2); { var consequent = ($$anchor2) => { var div_3 = root_2$7(); each(div_3, 15, () => get$1(chipValues), ({ id, val }) => id, ($$anchor3, $$item, i) => { let val = () => get$1($$item).val; var div_4 = root_3$9(); var button2 = child(div_4); var span = child(button2); var text_1 = child(span, true); reset(span); next(2); reset(button2); reset(div_4); template_effect(() => { set_class(button2, `chip ${chips() ?? ""}`); set_text(text_1, val()); }); event("click", button2, (e) => removeChipInternally(e, get$1(i), val())); event("click", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keypress", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keydown", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keyup", button2, function($$arg) { bubble_event.call(this, $$props, $$arg); }); transition(1, button2, () => dynamicTransition, () => ({ transition: chipTransitionIn(), params: chipTransitionInParams(), enabled: transitions() })); transition(2, button2, () => dynamicTransition, () => ({ transition: chipTransitionOut(), params: chipTransitionOutParams(), enabled: transitions() })); animation(div_4, () => flip, () => ({ duration: duration() })); append($$anchor3, div_4); }); reset(div_3); template_effect(() => set_class(div_3, `input-chip-list ${get$1(classesChipList) ?? ""}`)); transition(1, div_3, () => dynamicTransition, () => ({ transition: listTransitionIn(), params: listTransitionInParams(), enabled: transitions() })); transition(2, div_3, () => dynamicTransition, () => ({ transition: listTransitionOut(), params: listTransitionOutParams(), enabled: transitions() })); append($$anchor2, div_3); }; if_block(node, ($$render) => { if (get$1(chipValues).length) $$render(consequent); }); } reset(div_2); reset(div); template_effect(() => { set_class(div, `input-chip ${get$1(classesBase) ?? ""}`); toggle_class(div, "opacity-50", $$restProps.disabled); set_attribute(select, "name", name()); select.required = required(); set_attribute(select, "aria-label", label()); set_class(div_2, `input-chip-wrapper ${get$1(classesChipWrapper) ?? ""}`); attributes = set_attributes(input_1, attributes, { type: "text", placeholder: $$restProps.placeholder ?? "Enter values...", class: `input-chip-field ${get$1(classesInput) ?? ""}`, disabled: $$restProps.disabled, ...get$1(spread_with_call) }); }); bind_select_value(select, value); bind_value(input_1, input); event("keydown", input_1, onKeyHandler); event("input", input_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("focus", input_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("blur", input_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor, div); bind_prop($$props, "addChip", addChip); bind_prop($$props, "removeChip", removeChip); var $$pop = pop({ addChip, removeChip }); $$cleanup(); return $$pop; } var root$h = /* @__PURE__ */ template(`<div role="listbox" data-testid="listbox"><!></div>`); function ListBox($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const classesBase = mutable_state(); let multiple = prop($$props, "multiple", 8, false); let disabled = prop($$props, "disabled", 8, false); let spacing = prop($$props, "spacing", 8, "space-y-1"); let rounded = prop($$props, "rounded", 8, "rounded-token"); let active = prop($$props, "active", 8, "variant-filled"); let hover = prop($$props, "hover", 8, "hover:variant-soft"); let padding = prop($$props, "padding", 8, "px-4 py-2"); let regionLead = prop($$props, "regionLead", 8, ""); let regionDefault = prop($$props, "regionDefault", 8, ""); let regionTrail = prop($$props, "regionTrail", 8, ""); let labelledby = prop($$props, "labelledby", 8, ""); setContext("disabled", disabled()); setContext("multiple", multiple()); setContext("rounded", rounded()); setContext("active", active()); setContext("hover", hover()); setContext("padding", padding()); setContext("regionLead", regionLead()); setContext("regionDefault", regionDefault()); setContext("regionTrail", regionTrail()); const cBase = ""; legacy_pre_effect( () => (deep_read_state(spacing()), deep_read_state(rounded()), deep_read_state($$sanitized_props)), () => { set(classesBase, `${cBase} ${spacing()} ${rounded()} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect_reset(); init(); var div = root$h(); var node = child(div); slot(node, $$props, "default", {}, null); reset(div); template_effect(() => { set_class(div, `listbox ${get$1(classesBase) ?? ""}`); set_attribute(div, "aria-labelledby", labelledby()); }); append($$anchor, div); pop(); } var root_1$g = /* @__PURE__ */ template(`<input type="checkbox" tabindex="-1">`); var root_2$6 = /* @__PURE__ */ template(`<input type="radio" tabindex="-1">`); var root_3$8 = /* @__PURE__ */ template(`<div><!></div>`); var root_4$6 = /* @__PURE__ */ template(`<div><!></div>`); var root$g = /* @__PURE__ */ template(`<label><div data-testid="listbox-item" role="option" tabindex="0"><div class="h-0 w-0 overflow-hidden"><!></div> <div><!> <div><!></div> <!></div></div></label>`); function ListBoxItem($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const selected = mutable_state(); const classesActive = mutable_state(); const classesDisabled = mutable_state(); const classesBase = mutable_state(); const classesLabel = mutable_state(); const classesRegionLead = mutable_state(); const classesRegionDefault = mutable_state(); const classesRegionTrail = mutable_state(); const binding_group = []; let group = prop($$props, "group", 12); let name = prop($$props, "name", 8); let value = prop($$props, "value", 8); let disabled = prop($$props, "disabled", 24, () => getContext("disabled")); let multiple = prop($$props, "multiple", 24, () => getContext("multiple")); let rounded = prop($$props, "rounded", 24, () => getContext("rounded")); let active = prop($$props, "active", 24, () => getContext("active")); let hover = prop($$props, "hover", 24, () => getContext("hover")); let padding = prop($$props, "padding", 24, () => getContext("padding")); let regionLead = prop($$props, "regionLead", 24, () => getContext("regionLead")); let regionDefault = prop($$props, "regionDefault", 24, () => getContext("regionDefault")); let regionTrail = prop($$props, "regionTrail", 24, () => getContext("regionTrail")); const cBase = "cursor-pointer -outline-offset-[3px]"; const cDisabled = "opacity-50 !cursor-default"; const cLabel = "flex items-center space-x-4"; let checked = mutable_state(); let elemInput = mutable_state(); function areDeeplyEqual(param1, param2) { if (param1 === param2) return true; if (!(param1 instanceof Object) || !(param2 instanceof Object)) return false; const keys1 = Object.keys(param1); const keys2 = Object.keys(param2); if (keys1.length !== keys2.length) return false; for (const key of keys1) { const value1 = param1[key]; const value2 = param2[key]; if (!areDeeplyEqual(value1, value2)) return false; } return true; } function updateCheckbox(group2) { set(checked, group2.indexOf(value()) >= 0); } function updateGroup(checked2) { const index2 = group().indexOf(value()); if (checked2) { if (index2 < 0) { group().push(value()); group(group()); } } else { if (index2 >= 0) { group().splice(index2, 1); group(group()); } } } function onKeyDown(event2) { if (["Enter", "Space"].includes(event2.code)) { event2.preventDefault(); get$1(elemInput).click(); } } const cRegionLead = ""; const cRegionDefault = "flex-1"; const cRegionTrail = ""; legacy_pre_effect( () => (deep_read_state(multiple()), deep_read_state(group())), () => { if (multiple()) updateCheckbox(group()); } ); legacy_pre_effect( () => (deep_read_state(multiple()), get$1(checked)), () => { if (multiple()) updateGroup(get$1(checked)); } ); legacy_pre_effect( () => (deep_read_state(multiple()), deep_read_state(group()), deep_read_state(value())), () => { set(selected, multiple() ? group().some((groupVal) => areDeeplyEqual(value(), groupVal)) : areDeeplyEqual(group(), value())); } ); legacy_pre_effect( () => (get$1(selected), deep_read_state(active()), deep_read_state(disabled()), deep_read_state(hover())), () => { set(classesActive, get$1(selected) ? active() : !disabled() ? hover() : ""); } ); legacy_pre_effect(() => deep_read_state(disabled()), () => { set(classesDisabled, disabled() ? cDisabled : ""); }); legacy_pre_effect( () => (get$1(classesDisabled), deep_read_state(rounded()), deep_read_state(padding()), get$1(classesActive), deep_read_state($$sanitized_props)), () => { set(classesBase, `${cBase} ${get$1(classesDisabled)} ${rounded()} ${padding()} ${get$1(classesActive)} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect(() => { }, () => { set(classesLabel, `${cLabel}`); }); legacy_pre_effect(() => deep_read_state(regionLead()), () => { set(classesRegionLead, `${cRegionLead} ${regionLead()}`); }); legacy_pre_effect(() => deep_read_state(regionDefault()), () => { set(classesRegionDefault, `${cRegionDefault} ${regionDefault()}`); }); legacy_pre_effect(() => deep_read_state(regionTrail()), () => { set(classesRegionTrail, `${cRegionTrail} ${regionTrail()}`); }); legacy_pre_effect_reset(); init(); var label = root$g(); var div = child(label); var div_1 = child(div); var node = child(div_1); { var consequent = ($$anchor2) => { var input = root_1$g(); remove_input_defaults(input); var input_value; bind_this(input, ($$value) => set(elemInput, $$value), () => get$1(elemInput)); template_effect(() => { input.disabled = disabled(); set_attribute(input, "name", name()); if (input_value !== (input_value = value())) { input.value = null == (input.__value = value()) ? "" : value(); } }); bind_checked(input, () => get$1(checked), ($$value) => set(checked, $$value)); event("click", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("change", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor2, input); }; var alternate = ($$anchor2) => { var input_1 = root_2$6(); remove_input_defaults(input_1); var input_1_value; bind_this(input_1, ($$value) => set(elemInput, $$value), () => get$1(elemInput)); template_effect(() => { input_1.disabled = disabled(); set_attribute(input_1, "name", name()); if (input_1_value !== (input_1_value = value())) { input_1.value = null == (input_1.__value = value()) ? "" : value(); } }); bind_group( binding_group, [], input_1, () => { value(); return group(); }, group ); event("click", input_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("change", input_1, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor2, input_1); }; if_block(node, ($$render) => { if (multiple()) $$render(consequent); else $$render(alternate, false); }); } reset(div_1); var div_2 = sibling(div_1, 2); var node_1 = child(div_2); { var consequent_1 = ($$anchor2) => { var div_3 = root_3$8(); var node_2 = child(div_3); slot(node_2, $$props, "lead", {}, null); reset(div_3); template_effect(() => set_class(div_3, `listbox-label-lead ${get$1(classesRegionLead) ?? ""}`)); append($$anchor2, div_3); }; if_block(node_1, ($$render) => { if ($$slots.lead) $$render(consequent_1); }); } var div_4 = sibling(node_1, 2); var node_3 = child(div_4); slot(node_3, $$props, "default", {}, null); reset(div_4); var node_4 = sibling(div_4, 2); { var consequent_2 = ($$anchor2) => { var div_5 = root_4$6(); var node_5 = child(div_5); slot(node_5, $$props, "trail", {}, null); reset(div_5); template_effect(() => set_class(div_5, `listbox-label-trail ${get$1(classesRegionTrail) ?? ""}`)); append($$anchor2, div_5); }; if_block(node_4, ($$render) => { if ($$slots.trail) $$render(consequent_2); }); } reset(div_2); reset(div); reset(label); template_effect(() => { set_class(div, `listbox-item ${get$1(classesBase) ?? ""}`); set_attribute(div, "aria-selected", get$1(selected)); set_class(div_2, `listbox-label ${get$1(classesLabel) ?? ""}`); set_class(div_4, `listbox-label-content ${get$1(classesRegionDefault) ?? ""}`); }); event("keydown", div, onKeyDown); event("keydown", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keyup", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keypress", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor, label); pop(); } var root$f = /* @__PURE__ */ template(`<div data-testid="progress-bar" role="progressbar"><div></div></div>`); function ProgressBar($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const fillPercent = mutable_state(); const indeterminate = mutable_state(); const classesIndeterminate = mutable_state(); const classesTrack = mutable_state(); const classesMeter = mutable_state(); let value = prop($$props, "value", 24, () => undefined); let min = prop($$props, "min", 8, 0); let max = prop($$props, "max", 8, 100); let height = prop($$props, "height", 8, "h-2"); let rounded = prop($$props, "rounded", 8, "rounded-token"); let transition2 = prop($$props, "transition", 8, "transition-[width]"); let animIndeterminate = prop($$props, "animIndeterminate", 8, "anim-indeterminate"); let meter = prop($$props, "meter", 8, "bg-surface-900-50-token"); let track = prop($$props, "track", 8, "bg-surface-200-700-token"); let labelledby = prop($$props, "labelledby", 8, ""); const cTrack = "w-full overflow-hidden"; const cMeter = "h-full"; legacy_pre_effect( () => (deep_read_state(value()), deep_read_state(min()), deep_read_state(max())), () => { set(fillPercent, value() ? 100 * (value() - min()) / (max() - min()) : 0); } ); legacy_pre_effect(() => deep_read_state(value()), () => { set(indeterminate, value() === undefined || value() < 0); }); legacy_pre_effect( () => (get$1(indeterminate), deep_read_state(animIndeterminate())), () => { set(classesIndeterminate, get$1(indeterminate) ? animIndeterminate() : ""); } ); legacy_pre_effect( () => (deep_read_state(track()), deep_read_state(height()), deep_read_state(rounded()), deep_read_state($$sanitized_props)), () => { set(classesTrack, `${cTrack} ${track()} ${height()} ${rounded()} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect( () => (deep_read_state(meter()), deep_read_state(rounded()), get$1(classesIndeterminate), deep_read_state(transition2())), () => { set(classesMeter, `${cMeter} ${meter()} ${rounded()} ${get$1(classesIndeterminate)} ${transition2()}`); } ); legacy_pre_effect_reset(); init(); var div = root$f(); var div_1 = child(div); reset(div); template_effect(() => { set_class(div, `progress-bar ${get$1(classesTrack) ?? ""} svelte-12wvf64`); set_attribute(div, "aria-labelledby", labelledby()); set_attribute(div, "aria-valuenow", value()); set_attribute(div, "aria-valuemin", min()); set_attribute(div, "aria-valuemax", max() - min()); set_class(div_1, `progress-bar-meter ${get$1(classesMeter) ?? ""} svelte-12wvf64`); set_style(div_1, "width", `${(get$1(indeterminate) ? 100 : get$1(fillPercent)) ?? ""}%`); }); append($$anchor, div); pop(); } var root_1$f = /* @__PURE__ */ ns_template(`<text x="50%" y="50%" text-anchor="middle" dominant-baseline="middle" font-weight="bold"><!></text>`); var root$e = /* @__PURE__ */ template(`<figure data-testid="progress-radial" role="meter"><svg class="rounded-full"><circle cx="50%" cy="50%"></circle><circle cx="50%" cy="50%"></circle><!></svg></figure>`); function ProgressRadial($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const classesBase = mutable_state(); let value = prop($$props, "value", 24, () => undefined); let stroke = prop($$props, "stroke", 8, 40); let font = prop($$props, "font", 8, 56); let strokeLinecap = prop($$props, "strokeLinecap", 8, "butt"); let transition2 = prop($$props, "transition", 8, "transition-[stroke-dashoffset]"); let width = prop($$props, "width", 8, "w-36"); let meter = prop($$props, "meter", 8, "stroke-surface-900 dark:stroke-surface-50"); let track = prop($$props, "track", 8, "stroke-surface-500/30"); let fill = prop($$props, "fill", 8, "fill-token"); let labelledby = prop($$props, "labelledby", 8, ""); const cBase = "progress-radial relative overflow-hidden"; const cBaseTrack = "fill-transparent"; const cBaseMeter = "fill-transparent -rotate-90 origin-[50%_50%]"; const baseSize = 512; const radius = baseSize / 2 - stroke() / 2; let circumference = mutable_state(radius); let dashoffset = mutable_state(); function setProgress(percent) { set(circumference, radius * 2 * Math.PI); set(dashoffset, get$1(circumference) - percent / 100 * get$1(circumference)); } setProgress(0); afterUpdate(() => { setProgress(value() === undefined ? 25 : value()); }); legacy_pre_effect( () => (deep_read_state(width()), deep_read_state($$sanitized_props)), () => { set(classesBase, `${cBase} ${width()} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect_reset(); init(); var figure = root$e(); set_attribute(figure, "aria-valuemin", 0); set_attribute(figure, "aria-valuemax", 100); var svg = child(figure); set_attribute(svg, "viewBox", `0 0 ${baseSize} ${baseSize}`); var circle = child(svg); set_attribute(circle, "r", radius); var circle_1 = sibling(circle); set_attribute(circle_1, "r", radius); var node = sibling(circle_1); { var consequent = ($$anchor2) => { var text2 = root_1$f(); var node_1 = child(text2); slot(node_1, $$props, "default", {}, null); reset(text2); template_effect(() => { set_attribute(text2, "font-size", font()); set_svg_class(text2, `progress-radial-text ${fill() ?? ""}`); }); append($$anchor2, text2); }; if_block(node, ($$render) => { if (value() != undefined && value() >= 0 && $$slots.default) $$render(consequent); }); } reset(svg); reset(figure); template_effect(() => { set_class(figure, `progress-radial ${get$1(classesBase) ?? ""}`); set_attribute(figure, "aria-labelledby", labelledby()); set_attribute(figure, "aria-valuenow", value() || 0); set_attribute(figure, "aria-valuetext", value() ? `${value()}%` : "Indeterminate Spinner"); toggle_class(svg, "animate-spin", value() === undefined); set_svg_class(circle, `progress-radial-track ${cBaseTrack} ${track() ?? ""}`); set_attribute(circle, "stroke-width", stroke()); set_svg_class(circle_1, `progress-radial-meter ${cBaseMeter} ${meter() ?? ""} ${transition2() ?? ""}`); set_attribute(circle_1, "stroke-width", stroke()); set_attribute(circle_1, "stroke-linecap", strokeLinecap()); set_style(circle_1, "stroke-dasharray", `${get$1(circumference) ?? ""} ${get$1(circumference) ?? ""}`); set_style(circle_1, "stroke-dashoffset", get$1(dashoffset)); }); append($$anchor, figure); pop(); } var root$d = /* @__PURE__ */ template(`<div data-testid="radio-group" role="radiogroup"><!></div>`); function RadioGroup($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const classesBase = mutable_state(); let display = prop($$props, "display", 8, "inline-flex"); let flexDirection = prop($$props, "flexDirection", 8, "flex-row"); let gap = prop($$props, "gap", 8, "gap-1"); let background = prop($$props, "background", 8, "bg-surface-200-700-token"); let border = prop($$props, "border", 8, "border-token border-surface-400-500-token"); let rounded = prop($$props, "rounded", 8, "rounded-token"); let padding = prop($$props, "padding", 8, "px-4 py-1"); let active = prop($$props, "active", 8, "variant-filled"); let hover = prop($$props, "hover", 8, "hover:variant-soft"); let color = prop($$props, "color", 8, ""); let fill = prop($$props, "fill", 8, ""); let regionLabel = prop($$props, "regionLabel", 8, ""); let labelledby = prop($$props, "labelledby", 8, ""); setContext("rounded", rounded()); setContext("padding", padding()); setContext("active", active()); setContext("hover", hover()); setContext("color", color()); setContext("fill", fill()); setContext("regionLabel", regionLabel()); const cBase = "p-1"; legacy_pre_effect( () => (deep_read_state(display()), deep_read_state(flexDirection()), deep_read_state(gap()), deep_read_state(background()), deep_read_state(border()), deep_read_state(rounded()), deep_read_state($$sanitized_props)), () => { set(classesBase, `${cBase} ${display()} ${flexDirection()} ${gap()} ${background()} ${border()} ${rounded()} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect_reset(); init(); var div = root$d(); var node = child(div); slot(node, $$props, "default", {}, null); reset(div); template_effect(() => { set_class(div, `radio-group ${get$1(classesBase) ?? ""}`); set_attribute(div, "aria-labelledby", labelledby()); }); append($$anchor, div); pop(); } var root$c = /* @__PURE__ */ template(`<label><div data-testid="radio-item" role="radio" tabindex="0"><div class="h-0 w-0 overflow-hidden"><input></div> <!></div></label>`); function RadioItem($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); const $$restProps = legacy_rest_props($$sanitized_props, [ "group", "name", "value", "title", "label", "rounded", "padding", "active", "hover", "color", "fill", "regionLabel" ]); push($$props, false); const checked = mutable_state(); const classesActive = mutable_state(); const classesDisabled = mutable_state(); const classsBase = mutable_state(); const classesWrapper = mutable_state(); const binding_group = []; let group = prop($$props, "group", 12); let name = prop($$props, "name", 8); let value = prop($$props, "value", 8); let title = prop($$props, "title", 8, ""); let label = prop($$props, "label", 8, ""); let rounded = prop($$props, "rounded", 24, () => getContext("rounded")); let padding = prop($$props, "padding", 24, () => getContext("padding")); let active = prop($$props, "active", 24, () => getContext("active")); let hover = prop($$props, "hover", 24, () => getContext("hover")); let color = prop($$props, "color", 24, () => getContext("color")); let fill = prop($$props, "fill", 24, () => getContext("fill")); let regionLabel = prop($$props, "regionLabel", 24, () => getContext("regionLabel")); const cBase = "flex-auto"; const cWrapper = "text-base text-center cursor-pointer"; const cDisabled = "opacity-50 cursor-not-allowed"; let elemInput = mutable_state(); function onKeyDown(event2) { if (["Enter", "Space"].includes(event2.code)) { event2.preventDefault(); get$1(elemInput).click(); } } function prunedRestProps() { delete $$restProps.class; return $$restProps; } legacy_pre_effect( () => (deep_read_state(value()), deep_read_state(group())), () => { set(checked, value() === group()); } ); legacy_pre_effect( () => (get$1(checked), deep_read_state(active()), deep_read_state(color()), deep_read_state(fill()), deep_read_state(hover())), () => { set(classesActive, get$1(checked) ? `${active()} ${color()} ${fill()}` : hover()); } ); legacy_pre_effect(() => deep_read_state($$sanitized_props), () => { set(classesDisabled, $$sanitized_props.disabled ? cDisabled : ""); }); legacy_pre_effect(() => { }, () => { set(classsBase, `${cBase}`); }); legacy_pre_effect( () => (deep_read_state(padding()), deep_read_state(rounded()), get$1(classesActive), get$1(classesDisabled), deep_read_state($$sanitized_props)), () => { set(classesWrapper, `${cWrapper} ${padding()} ${rounded()} ${get$1(classesActive)} ${get$1(classesDisabled)} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect_reset(); init(); var label_1 = root$c(); var div = child(label_1); var div_1 = child(div); var input = child(div_1); remove_input_defaults(input); const spread_with_call = /* @__PURE__ */ derived_safe_equal(prunedRestProps); let attributes; bind_this(input, ($$value) => set(elemInput, $$value), () => get$1(elemInput)); reset(div_1); var node = sibling(div_1, 2); slot(node, $$props, "default", {}, null); reset(div); reset(label_1); template_effect(() => { set_class(label_1, `radio-label ${get$1(classsBase) ?? ""} ${regionLabel() ?? ""}`); set_class(div, `radio-item ${get$1(classesWrapper) ?? ""}`); set_attribute(div, "aria-checked", get$1(checked)); set_attribute(div, "aria-label", label()); set_attribute(div, "title", title()); attributes = set_attributes(input, attributes, { type: "radio", name: name(), value: value(), ...get$1(spread_with_call), tabindex: "-1" }); }); bind_group( binding_group, [], input, () => { value(); return group(); }, group ); event("click", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("change", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keydown", div, onKeyDown); event("keydown", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keyup", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keypress", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor, label_1); pop(); } var root_1$e = /* @__PURE__ */ template(`<label><!></label>`); var root_3$7 = /* @__PURE__ */ template(`<option></option>`); var root_2$5 = /* @__PURE__ */ template(`<datalist class="range-slider-ticks"></datalist>`); var root_4$5 = /* @__PURE__ */ template(`<div class="range-slider-trail"><!></div>`); var root$b = /* @__PURE__ */ template(`<div data-testid="range-slider"><!> <div><input> <!></div> <!></div>`); function RangeSlider($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); const $$restProps = legacy_rest_props($$sanitized_props, [ "name", "id", "value", "min", "max", "step", "ticked", "accent", "label" ]); push($$props, false); const classesBase = mutable_state(); const classesInput = mutable_state(); let name = prop($$props, "name", 8); let id = prop($$props, "id", 24, () => String(Math.random())); let value = prop($$props, "value", 12, 0); let min = prop($$props, "min", 8, 0); let max = prop($$props, "max", 8, 100); let step = prop($$props, "step", 8, 1); let ticked = prop($$props, "ticked", 8, false); let accent = prop($$props, "accent", 8, "accent-surface-900 dark:accent-surface-50"); let label = prop($$props, "label", 8, ""); const cBase = "space-y-2"; const cBaseLabel = ""; const cBaseContent = "flex justify-center py-2"; const cBaseInput = "w-full h-2"; let tickmarks = mutable_state(); function setTicks() { if (ticked() == false) return; set(tickmarks, Array.from({ length: max() - min() + 1 }, (_, i) => i + min())); } if (ticked()) setTicks(); afterUpdate(() => { setTicks(); }); function prunedRestProps() { delete $$restProps.class; return $$restProps; } legacy_pre_effect(() => deep_read_state($$sanitized_props), () => { set(classesBase, `${cBase} ${$$sanitized_props.class ?? ""}`); }); legacy_pre_effect(() => deep_read_state(accent()), () => { set(classesInput, `${cBaseInput} ${accent()}`); }); legacy_pre_effect_reset(); init(); var div = root$b(); var node = child(div); { var consequent = ($$anchor2) => { var label_1 = root_1$e(); set_class(label_1, `range-slider-label ${cBaseLabel}`); var node_1 = child(label_1); slot(node_1, $$props, "default", {}, null); reset(label_1); template_effect(() => set_attribute(label_1, "for", id())); append($$anchor2, label_1); }; if_block(node, ($$render) => { if ($$slots.default) $$render(consequent); }); } var div_1 = sibling(node, 2); set_class(div_1, `range-content ${cBaseContent}`); var input = child(div_1); remove_input_defaults(input); const spread_with_call = /* @__PURE__ */ derived_safe_equal(prunedRestProps); let attributes; var node_2 = sibling(input, 2); { var consequent_1 = ($$anchor2) => { var datalist = root_2$5(); each(datalist, 5, () => get$1(tickmarks), index, ($$anchor3, tm) => { var option = root_3$7(); var option_value = {}; template_effect(() => { if (option_value !== (option_value = get$1(tm))) { option.value = null == (option.__value = get$1(tm)) ? "" : get$1(tm); } set_attribute(option, "label", get$1(tm)); }); append($$anchor3, option); }); reset(datalist); template_effect(() => set_attribute(datalist, "id", `tickmarks-${id() ?? ""}`)); append($$anchor2, datalist); }; if_block(node_2, ($$render) => { if (ticked() && get$1(tickmarks) && get$1(tickmarks).length) $$render(consequent_1); }); } reset(div_1); var node_3 = sibling(div_1, 2); { var consequent_2 = ($$anchor2) => { var div_2 = root_4$5(); var node_4 = child(div_2); slot(node_4, $$props, "trail", {}, null); reset(div_2); append($$anchor2, div_2); }; if_block(node_3, ($$render) => { if ($$slots.trail) $$render(consequent_2); }); } reset(div); template_effect(() => { set_class(div, `range-slider ${get$1(classesBase) ?? ""}`); attributes = set_attributes(input, attributes, { type: "range", id: id(), name: name(), class: `range-slider-input ${get$1(classesInput) ?? ""}`, list: `tickmarks-${id() ?? ""}`, "aria-label": label(), min: min(), max: max(), step: step(), ...get$1(spread_with_call) }); }); bind_value(input, value); event("click", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("change", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("blur", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor, div); pop(); } var root_1$d = /* @__PURE__ */ template(`<div class="slide-toggle-text ml-3"><!></div>`); var root$a = /* @__PURE__ */ template(`<div data-testid="slide-toggle" role="switch" tabindex="0"><label><input> <div><div></div></div> <!></label></div>`); function SlideToggle($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); const $$restProps = legacy_rest_props($$sanitized_props, [ "name", "checked", "size", "background", "active", "border", "rounded", "label" ]); push($$props, false); const cTrackActive = mutable_state(); const cThumbBackground = mutable_state(); const cThumbPos = mutable_state(); const classesDisabled = mutable_state(); const classesBase = mutable_state(); const classesLabel = mutable_state(); const classesTrack = mutable_state(); const classesThumb = mutable_state(); const dispatch = createEventDispatcher(); let name = prop($$props, "name", 8); let checked = prop($$props, "checked", 12, false); let size = prop($$props, "size", 8, "md"); let background = prop($$props, "background", 8, "bg-surface-400 dark:bg-surface-700"); let active = prop($$props, "active", 8, "bg-surface-900 dark:bg-surface-300"); let border = prop($$props, "border", 8, ""); let rounded = prop($$props, "rounded", 8, "rounded-full"); let label = prop($$props, "label", 8, ""); const cBase = "inline-block"; const cLabel = "unstyled flex items-center"; const cTrack = "flex transition-all duration-[200ms] cursor-pointer"; const cThumb = "w-[50%] h-full scale-[0.8] transition-all duration-[200ms] shadow"; let trackSize = mutable_state(); switch (size()) { case "sm": set(trackSize, "w-12 h-6"); break; case "lg": set(trackSize, "w-20 h-10"); break; default: set(trackSize, "w-16 h-8"); } function onKeyDown(event2) { if (["Enter", "Space"].includes(event2.code)) { event2.preventDefault(); dispatch("keyup", event2); const inputElem = event2.currentTarget.firstChild; inputElem.click(); } } function prunedRestProps() { delete $$restProps.class; return $$restProps; } legacy_pre_effect( () => (deep_read_state(checked()), deep_read_state(active()), deep_read_state(background())), () => { set(cTrackActive, checked() ? active() : `${background()} cursor-pointer`); } ); legacy_pre_effect(() => deep_read_state(checked()), () => { set(cThumbBackground, checked() ? "bg-white/75" : "bg-white"); }); legacy_pre_effect(() => deep_read_state(checked()), () => { set(cThumbPos, checked() ? "translate-x-full" : ""); }); legacy_pre_effect(() => deep_read_state($$sanitized_props), () => { set(classesDisabled, $$sanitized_props.disabled === true ? "opacity-50" : "hover:brightness-[105%] dark:hover:brightness-110 cursor-pointer"); }); legacy_pre_effect( () => (deep_read_state(rounded()), get$1(classesDisabled), deep_read_state($$sanitized_props)), () => { set(classesBase, `${cBase} ${rounded()} ${get$1(classesDisabled)} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect(() => { }, () => { set(classesLabel, `${cLabel}`); }); legacy_pre_effect( () => (deep_read_state(border()), deep_read_state(rounded()), get$1(trackSize), get$1(cTrackActive)), () => { set(classesTrack, `${cTrack} ${border()} ${rounded()} ${get$1(trackSize)} ${get$1(cTrackActive)}`); } ); legacy_pre_effect( () => (deep_read_state(rounded()), get$1(cThumbBackground), get$1(cThumbPos)), () => { set(classesThumb, `${cThumb} ${rounded()} ${get$1(cThumbBackground)} ${get$1(cThumbPos)}`); } ); legacy_pre_effect_reset(); init(); var div = root$a(); var label_1 = child(div); var input = child(label_1); remove_input_defaults(input); const spread_with_call = /* @__PURE__ */ derived_safe_equal(prunedRestProps); let attributes; var div_1 = sibling(input, 2); var div_2 = child(div_1); reset(div_1); var node = sibling(div_1, 2); { var consequent = ($$anchor2) => { var div_3 = root_1$d(); var node_1 = child(div_3); slot(node_1, $$props, "default", {}, null); reset(div_3); append($$anchor2, div_3); }; if_block(node, ($$render) => { if ($$slots.default) $$render(consequent); }); } reset(label_1); reset(div); template_effect(() => { set_attribute(div, "id", label()); set_class(div, `slide-toggle ${get$1(classesBase) ?? ""}`); set_attribute(div, "aria-label", label()); set_attribute(div, "aria-checked", checked()); set_class(label_1, `slide-toggle-label ${get$1(classesLabel) ?? ""}`); attributes = set_attributes(input, attributes, { type: "checkbox", class: "slide-toggle-input hidden", name: name(), ...get$1(spread_with_call), disabled: $$sanitized_props.disabled }); set_class(div_1, `slide-toggle-track ${get$1(classesTrack) ?? ""}`); toggle_class(div_1, "cursor-not-allowed", $$sanitized_props.disabled); set_class(div_2, `slide-toggle-thumb ${get$1(classesThumb) ?? ""}`); toggle_class(div_2, "cursor-not-allowed", $$sanitized_props.disabled); }); bind_checked(input, checked); event("click", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keydown", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keyup", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keypress", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("mouseover", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("change", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("focus", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("blur", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keydown", div, onKeyDown); append($$anchor, div); pop(); } var root_1$c = /* @__PURE__ */ template(`<div role="tabpanel" tabindex="0"><!></div>`); var root$9 = /* @__PURE__ */ template(`<div data-testid="tab-group"><div role="tablist"><!></div> <!></div>`); function TabGroup($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const classesBase = mutable_state(); const classesList = mutable_state(); const classesPanel = mutable_state(); let justify = prop($$props, "justify", 8, "justify-start"); let border = prop($$props, "border", 8, "border-b border-surface-400-500-token"); let active = prop($$props, "active", 8, "border-b-2 border-surface-900-50-token"); let hover = prop($$props, "hover", 8, "hover:variant-soft"); let flex = prop($$props, "flex", 8, "flex-none"); let padding = prop($$props, "padding", 8, "px-4 py-2"); let rounded = prop($$props, "rounded", 8, "rounded-tl-container-token rounded-tr-container-token"); let spacing = prop($$props, "spacing", 8, "space-y-1"); let regionList = prop($$props, "regionList", 8, ""); let regionPanel = prop($$props, "regionPanel", 8, ""); let labelledby = prop($$props, "labelledby", 8, ""); let panel = prop($$props, "panel", 8, ""); setContext("active", active()); setContext("hover", hover()); setContext("flex", flex()); setContext("padding", padding()); setContext("rounded", rounded()); setContext("spacing", spacing()); const cBase = "space-y-4"; const cList = "flex overflow-x-auto hide-scrollbar"; const cPanel = ""; legacy_pre_effect(() => deep_read_state($$sanitized_props), () => { set(classesBase, `${cBase} ${$$sanitized_props.class ?? ""}`); }); legacy_pre_effect( () => (deep_read_state(justify()), deep_read_state(border()), deep_read_state(regionList())), () => { set(classesList, `${cList} ${justify()} ${border()} ${regionList()}`); } ); legacy_pre_effect(() => deep_read_state(regionPanel()), () => { set(classesPanel, `${cPanel} ${regionPanel()}`); }); legacy_pre_effect_reset(); init(); var div = root$9(); var div_1 = child(div); var node = child(div_1); slot(node, $$props, "default", {}, null); reset(div_1); var node_1 = sibling(div_1, 2); { var consequent = ($$anchor2) => { var div_2 = root_1$c(); var node_2 = child(div_2); slot(node_2, $$props, "panel", {}, null); reset(div_2); template_effect(() => { set_class(div_2, `tab-panel ${get$1(classesPanel) ?? ""}`); set_attribute(div_2, "aria-labelledby", panel()); }); append($$anchor2, div_2); }; if_block(node_1, ($$render) => { if ($$slots.panel) $$render(consequent); }); } reset(div); template_effect(() => { set_class(div, `tab-group ${get$1(classesBase) ?? ""}`); set_class(div_1, `tab-list ${get$1(classesList) ?? ""}`); set_attribute(div_1, "aria-labelledby", labelledby()); }); append($$anchor, div); pop(); } var root_1$b = /* @__PURE__ */ template(`<div class="tab-lead"><!></div>`); var root$8 = /* @__PURE__ */ template(`<label><div data-testid="tab" role="tab"><div class="h-0 w-0 overflow-hidden"><input></div> <div><!> <div class="tab-label"><!></div></div></div></label>`); function Tab($$anchor, $$props) { const $$slots = sanitize_slots($$props); const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); const $$restProps = legacy_rest_props($$sanitized_props, [ "group", "name", "value", "title", "controls", "regionTab", "active", "hover", "flex", "padding", "rounded", "spacing" ]); push($$props, false); const selected = mutable_state(); const classesActive = mutable_state(); const classesBase = mutable_state(); const classesInterface = mutable_state(); const classesTab = mutable_state(); const binding_group = []; let group = prop($$props, "group", 12); let name = prop($$props, "name", 8); let value = prop($$props, "value", 8); let title = prop($$props, "title", 8, ""); let controls = prop($$props, "controls", 8, ""); let regionTab = prop($$props, "regionTab", 8, ""); let active = prop($$props, "active", 24, () => getContext("active")); let hover = prop($$props, "hover", 24, () => getContext("hover")); let flex = prop($$props, "flex", 24, () => getContext("flex")); let padding = prop($$props, "padding", 24, () => getContext("padding")); let rounded = prop($$props, "rounded", 24, () => getContext("rounded")); let spacing = prop($$props, "spacing", 24, () => getContext("spacing")); const cBase = "text-center cursor-pointer transition-colors duration-100"; const cInterface = ""; let elemInput = mutable_state(); function onKeyDown(event2) { if (["Enter", "Space"].includes(event2.code)) { event2.preventDefault(); get$1(elemInput).click(); } else if (event2.code === "ArrowRight") { const tabList = get$1(elemInput).closest(".tab-list"); if (!tabList) return; const tabs = Array.from(tabList.querySelectorAll(".tab")); const currTab = get$1(elemInput).closest(".tab"); if (!currTab) return; const currIndex = tabs.indexOf(currTab); const nextIndex = currIndex + 1 >= tabs.length ? 0 : currIndex + 1; const nextTab = tabs[nextIndex]; const nextTabInput = nextTab == null ? undefined : nextTab.querySelector("input"); if (nextTab && nextTabInput) { nextTabInput.click(); nextTab.focus(); } } else if (event2.code === "ArrowLeft") { const tabList = get$1(elemInput).closest(".tab-list"); if (!tabList) return; const tabs = Array.from(tabList.querySelectorAll(".tab")); const currTab = get$1(elemInput).closest(".tab"); if (!currTab) return; const currIndex = tabs.indexOf(currTab); const nextIndex = currIndex - 1 < 0 ? tabs.length - 1 : currIndex - 1; const nextTab = tabs[nextIndex]; const nextTabInput = nextTab == null ? undefined : nextTab.querySelector("input"); if (nextTab && nextTabInput) { nextTabInput.click(); nextTab.focus(); } } } function prunedRestProps() { delete $$restProps.class; return $$restProps; } legacy_pre_effect( () => (deep_read_state(value()), deep_read_state(group())), () => { set(selected, value() === group()); } ); legacy_pre_effect( () => (get$1(selected), deep_read_state(active()), deep_read_state(hover())), () => { set(classesActive, get$1(selected) ? active() : hover()); } ); legacy_pre_effect( () => (deep_read_state(flex()), deep_read_state(padding()), deep_read_state(rounded()), get$1(classesActive), deep_read_state($$sanitized_props)), () => { set(classesBase, `${cBase} ${flex()} ${padding()} ${rounded()} ${get$1(classesActive)} ${$$sanitized_props.class ?? ""}`); } ); legacy_pre_effect(() => deep_read_state(spacing()), () => { set(classesInterface, `${cInterface} ${spacing()}`); }); legacy_pre_effect(() => deep_read_state(regionTab()), () => { set(classesTab, `${regionTab()}`); }); legacy_pre_effect_reset(); init(); var label = root$8(); var div = child(label); var div_1 = child(div); var input = child(div_1); remove_input_defaults(input); const spread_with_call = /* @__PURE__ */ derived_safe_equal(prunedRestProps); let attributes; bind_this(input, ($$value) => set(elemInput, $$value), () => get$1(elemInput)); reset(div_1); var div_2 = sibling(div_1, 2); var node = child(div_2); { var consequent = ($$anchor2) => { var div_3 = root_1$b(); var node_1 = child(div_3); slot(node_1, $$props, "lead", {}, null); reset(div_3); append($$anchor2, div_3); }; if_block(node, ($$render) => { if ($$slots.lead) $$render(consequent); }); } var div_4 = sibling(node, 2); var node_2 = child(div_4); slot(node_2, $$props, "default", {}, null); reset(div_4); reset(div_2); reset(div); reset(label); template_effect(() => { set_class(label, clsx(get$1(classesBase))); set_attribute(label, "title", title()); set_class(div, `tab ${get$1(classesTab) ?? ""}`); set_attribute(div, "aria-controls", controls()); set_attribute(div, "aria-selected", get$1(selected)); set_attribute(div, "tabindex", get$1(selected) ? 0 : -1); attributes = set_attributes(input, attributes, { type: "radio", name: name(), value: value(), ...get$1(spread_with_call), tabindex: "-1" }); set_class(div_2, `tab-interface ${get$1(classesInterface) ?? ""}`); }); bind_group( binding_group, [], input, () => { value(); return group(); }, group ); event("click", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("change", input, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keydown", div, onKeyDown); event("keydown", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keyup", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); event("keypress", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }); append($$anchor, label); pop(); } var root_4$4 = /* @__PURE__ */ template(`<header><!></header>`); var root_5$3 = /* @__PURE__ */ template(`<article><!></article>`); var root_6$3 = /* @__PURE__ */ template(`<img alt="Modal">`); var root_7 = /* @__PURE__ */ template(`<footer><button type="button"> </button></footer>`); var root_9$4 = /* @__PURE__ */ template(`<footer><button type="button"> </button> <button type="button"> </button></footer>`); var root_11$3 = /* @__PURE__ */ template(`<form class="space-y-4"><input> <footer><button type="button"> </button> <button type="submit"> </button></footer></form>`); var root_3$6 = /* @__PURE__ */ template(`<div data-testid="modal" role="dialog" aria-modal="true"><!> <!> <!> <!></div>`); var root_12$2 = /* @__PURE__ */ template(`<div data-testid="modal-component" role="dialog" aria-modal="true"><!></div>`); var root_2$4 = /* @__PURE__ */ template(`<div data-testid="modal-backdrop"><div><!></div></div>`); function Modal($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const [$$stores, $$cleanup] = setup_stores(); const $prefersReducedMotionStore = () => store_get(prefersReducedMotionStore, "$prefersReducedMotionStore", $$stores); const $modalStore = () => store_get(modalStore, "$modalStore", $$stores); const cPosition = mutable_state(); const classesBackdrop = mutable_state(); const classesTransitionLayer = mutable_state(); const classesModal = mutable_state(); const parent = mutable_state(); const dispatch = createEventDispatcher(); let components = prop($$props, "components", 24, () => ({})); let position = prop($$props, "position", 8, "items-center"); let background = prop($$props, "background", 8, "bg-surface-100-800-token"); let width = prop($$props, "width", 8, "w-modal"); let height = prop($$props, "height", 8, "h-auto"); let padding = prop($$props, "padding", 8, "p-4"); let spacing = prop($$props, "spacing", 8, "space-y-4"); let rounded = prop($$props, "rounded", 8, "rounded-container-token"); let shadow = prop($$props, "shadow", 8, "shadow-xl"); let zIndex = prop($$props, "zIndex", 8, "z-[999]"); let buttonNeutral = prop($$props, "buttonNeutral", 8, "variant-ghost-surface"); let buttonPositive = prop($$props, "buttonPositive", 8, "variant-filled"); let buttonTextCancel = prop($$props, "buttonTextCancel", 12, "Cancel"); let buttonTextConfirm = prop($$props, "buttonTextConfirm", 12, "Confirm"); let buttonTextSubmit = prop($$props, "buttonTextSubmit", 12, "Submit"); let regionBackdrop = prop($$props, "regionBackdrop", 8, ""); let regionHeader = prop($$props, "regionHeader", 8, "text-2xl font-bold"); let regionBody = prop($$props, "regionBody", 8, "max-h-[200px] overflow-hidden"); let regionFooter = prop($$props, "regionFooter", 8, "flex justify-end space-x-2"); let transitions = prop($$props, "transitions", 24, () => !$prefersReducedMotionStore()); let transitionIn = prop($$props, "transitionIn", 8, fly); let transitionInParams = prop($$props, "transitionInParams", 24, () => ({ duration: 150, opacity: 0, x: 0, y: 100 })); let transitionOut = prop($$props, "transitionOut", 8, fly); let transitionOutParams = prop($$props, "transitionOutParams", 24, () => ({ duration: 150, opacity: 0, x: 0, y: 100 })); const cBackdrop = "fixed top-0 left-0 right-0 bottom-0 bg-surface-backdrop-token p-4"; const cTransitionLayer = "w-full h-fit min-h-full overflow-y-auto flex justify-center"; const cModal = "block overflow-y-auto"; const cModalImage = "w-full h-auto"; let promptValue = mutable_state(); const buttonTextDefaults = { buttonTextCancel: buttonTextCancel(), buttonTextConfirm: buttonTextConfirm(), buttonTextSubmit: buttonTextSubmit() }; let currentComponent = mutable_state(); let registeredInteractionWithBackdrop = false; let modalElement = mutable_state(); let windowHeight = mutable_state(); let backdropOverflow = mutable_state("overflow-y-hidden"); const modalStore = getModalStore(); function handleModals(modals) { if (modals[0].type === "prompt") set(promptValue, modals[0].value); buttonTextCancel(modals[0].buttonTextCancel || buttonTextDefaults.buttonTextCancel); buttonTextConfirm(modals[0].buttonTextConfirm || buttonTextDefaults.buttonTextConfirm); buttonTextSubmit(modals[0].buttonTextSubmit || buttonTextDefaults.buttonTextSubmit); set(currentComponent, typeof modals[0].component === "string" ? components()[modals[0].component] : modals[0].component); } function onModalHeightChange(modal) { var _a2; let modalHeight = modal == null ? undefined : modal.clientHeight; if (!modalHeight) modalHeight = (_a2 = modal == null ? undefined : modal.firstChild) == null ? undefined : _a2.clientHeight; if (!modalHeight) return; if (modalHeight > get$1(windowHeight)) { set(backdropOverflow, "overflow-y-auto"); } else { set(backdropOverflow, "overflow-y-hidden"); } } function onBackdropInteractionBegin(event2) { if (!(event2.target instanceof Element)) return; const classList = event2.target.classList; if (classList.contains("modal-backdrop") || classList.contains("modal-transition")) { registeredInteractionWithBackdrop = true; } } function onBackdropInteractionEnd(event2) { if (!(event2.target instanceof Element)) return; const classList = event2.target.classList; if ((classList.contains("modal-backdrop") || classList.contains("modal-transition")) && registeredInteractionWithBackdrop) { if ($modalStore()[0].response) $modalStore()[0].response(undefined); modalStore.close(); dispatch("backdrop", event2); } registeredInteractionWithBackdrop = false; } function onClose() { if ($modalStore()[0].response) $modalStore()[0].response(false); modalStore.close(); } function onConfirm() { if ($modalStore()[0].response) $modalStore()[0].response(true); modalStore.close(); } function onPromptSubmit(event2) { event2.preventDefault(); if ($modalStore()[0].response) { if ($modalStore()[0].valueAttr !== undefined && "type" in $modalStore()[0].valueAttr && $modalStore()[0].valueAttr.type === "number") $modalStore()[0].response(parseInt(get$1(promptValue))); else $modalStore()[0].response(get$1(promptValue)); } modalStore.close(); } function onKeyDown(event2) { if (!$modalStore().length) return; if (event2.code === "Escape") onClose(); } legacy_pre_effect(() => $modalStore(), () => { if ($modalStore().length) handleModals($modalStore()); }); legacy_pre_effect(() => get$1(modalElement), () => { onModalHeightChange(get$1(modalElement)); }); legacy_pre_effect( () => ($modalStore(), deep_read_state(position())), () => { var _a2; set(cPosition, ((_a2 = $modalStore()[0]) == null ? undefined : _a2.position) ?? position()); } ); legacy_pre_effect( () => (deep_read_state(regionBackdrop()), deep_read_state(zIndex()), deep_read_state($$sanitized_props), $modalStore()), () => { var _a2; set(classesBackdrop, `${cBackdrop} ${regionBackdrop()} ${zIndex()} ${$$sanitized_props.class ?? ""} ${((_a2 = $modalStore()[0]) == null ? undefined : _a2.backdropClasses) ?? ""}`); } ); legacy_pre_effect(() => get$1(cPosition), () => { set(classesTransitionLayer, `${cTransitionLayer} ${get$1(cPosition) ?? ""}`); }); legacy_pre_effect( () => (deep_read_state(background()), deep_read_state(width()), deep_read_state(height()), deep_read_state(padding()), deep_read_state(spacing()), deep_read_state(rounded()), deep_read_state(shadow()), $modalStore()), () => { var _a2; set(classesModal, `${cModal} ${background()} ${width()} ${height()} ${padding()} ${spacing()} ${rounded()} ${shadow()} ${((_a2 = $modalStore()[0]) == null ? undefined : _a2.modalClasses) ?? ""}`); } ); legacy_pre_effect( () => (deep_read_state(position()), deep_read_state(background()), deep_read_state(width()), deep_read_state(height()), deep_read_state(padding()), deep_read_state(spacing()), deep_read_state(rounded()), deep_read_state(shadow()), deep_read_state(buttonNeutral()), deep_read_state(buttonPositive()), deep_read_state(buttonTextCancel()), deep_read_state(buttonTextConfirm()), deep_read_state(buttonTextSubmit()), deep_read_state(regionBackdrop()), deep_read_state(regionHeader()), deep_read_state(regionBody()), deep_read_state(regionFooter())), () => { set(parent, { position: position(), // --- background: background(), width: width(), height: height(), padding: padding(), spacing: spacing(), rounded: rounded(), shadow: shadow(), // --- buttonNeutral: buttonNeutral(), buttonPositive: buttonPositive(), buttonTextCancel: buttonTextCancel(), buttonTextConfirm: buttonTextConfirm(), buttonTextSubmit: buttonTextSubmit(), // --- regionBackdrop: regionBackdrop(), regionHeader: regionHeader(), regionBody: regionBody(), regionFooter: regionFooter(), // --- onClose }); } ); legacy_pre_effect_reset(); init(); var fragment = comment(); event("keydown", $window, onKeyDown); var node = first_child(fragment); { var consequent_8 = ($$anchor2) => { var fragment_1 = comment(); var node_1 = first_child(fragment_1); key_block(node_1, $modalStore, ($$anchor3) => { var div = root_2$4(); var div_1 = child(div); var node_2 = child(div_1); { var consequent_6 = ($$anchor4) => { var div_2 = root_3$6(); var node_3 = child(div_2); { var consequent = ($$anchor5) => { var header = root_4$4(); var node_4 = child(header); html(node_4, () => $modalStore()[0].title); reset(header); template_effect(() => set_class(header, `modal-header ${regionHeader() ?? ""}`)); append($$anchor5, header); }; if_block(node_3, ($$render) => { var _a2; if ((_a2 = $modalStore()[0]) == null ? undefined : _a2.title) $$render(consequent); }); } var node_5 = sibling(node_3, 2); { var consequent_1 = ($$anchor5) => { var article = root_5$3(); var node_6 = child(article); html(node_6, () => $modalStore()[0].body); reset(article); template_effect(() => set_class(article, `modal-body ${regionBody() ?? ""}`)); append($$anchor5, article); }; if_block(node_5, ($$render) => { var _a2; if ((_a2 = $modalStore()[0]) == null ? undefined : _a2.body) $$render(consequent_1); }); } var node_7 = sibling(node_5, 2); { var consequent_2 = ($$anchor5) => { var img = root_6$3(); set_class(img, `modal-image ${cModalImage}`); template_effect(() => { var _a2; return set_attribute(img, "src", (_a2 = $modalStore()[0]) == null ? undefined : _a2.image); }); append($$anchor5, img); }; if_block(node_7, ($$render) => { var _a2, _b2; if (((_a2 = $modalStore()[0]) == null ? undefined : _a2.image) && typeof ((_b2 = $modalStore()[0]) == null ? undefined : _b2.image) === "string") $$render(consequent_2); }); } var node_8 = sibling(node_7, 2); { var consequent_3 = ($$anchor5) => { var footer = root_7(); var button2 = child(footer); var text2 = child(button2, true); reset(button2); reset(footer); template_effect(() => { set_class(footer, `modal-footer ${regionFooter() ?? ""}`); set_class(button2, `btn ${buttonNeutral() ?? ""}`); set_text(text2, buttonTextCancel()); }); event("click", button2, onClose); append($$anchor5, footer); }; var alternate_1 = ($$anchor5) => { var fragment_2 = comment(); var node_9 = first_child(fragment_2); { var consequent_4 = ($$anchor6) => { var footer_1 = root_9$4(); var button_1 = child(footer_1); var text_1 = child(button_1, true); reset(button_1); var button_2 = sibling(button_1, 2); var text_2 = child(button_2, true); reset(button_2); reset(footer_1); template_effect(() => { set_class(footer_1, `modal-footer ${regionFooter() ?? ""}`); set_class(button_1, `btn ${buttonNeutral() ?? ""}`); set_text(text_1, buttonTextCancel()); set_class(button_2, `btn ${buttonPositive() ?? ""}`); set_text(text_2, buttonTextConfirm()); }); event("click", button_1, onClose); event("click", button_2, onConfirm); append($$anchor6, footer_1); }; var alternate = ($$anchor6) => { var fragment_3 = comment(); var node_10 = first_child(fragment_3); { var consequent_5 = ($$anchor7) => { var form = root_11$3(); var input = child(form); remove_input_defaults(input); let attributes; var footer_2 = sibling(input, 2); var button_3 = child(footer_2); var text_3 = child(button_3, true); reset(button_3); var button_4 = sibling(button_3, 2); var text_4 = child(button_4, true); reset(button_4); reset(footer_2); reset(form); template_effect(() => { attributes = set_attributes(input, attributes, { class: "modal-prompt-input input", name: "prompt", type: "text", ...$modalStore()[0].valueAttr }); set_class(footer_2, `modal-footer ${regionFooter() ?? ""}`); set_class(button_3, `btn ${buttonNeutral() ?? ""}`); set_text(text_3, buttonTextCancel()); set_class(button_4, `btn ${buttonPositive() ?? ""}`); set_text(text_4, buttonTextSubmit()); }); bind_value(input, () => get$1(promptValue), ($$value) => set(promptValue, $$value)); event("click", button_3, onClose); event("submit", form, onPromptSubmit); append($$anchor7, form); }; if_block( node_10, ($$render) => { if ($modalStore()[0].type === "prompt") $$render(consequent_5); }, true ); } append($$anchor6, fragment_3); }; if_block( node_9, ($$render) => { if ($modalStore()[0].type === "confirm") $$render(consequent_4); else $$render(alternate, false); }, true ); } append($$anchor5, fragment_2); }; if_block(node_8, ($$render) => { if ($modalStore()[0].type === "alert") $$render(consequent_3); else $$render(alternate_1, false); }); } reset(div_2); bind_this(div_2, ($$value) => set(modalElement, $$value), () => get$1(modalElement)); template_effect(() => { set_class(div_2, `modal ${get$1(classesModal) ?? ""}`); set_attribute(div_2, "aria-label", $modalStore()[0].title ?? ""); }); append($$anchor4, div_2); }; var alternate_3 = ($$anchor4) => { var div_3 = root_12$2(); var node_11 = child(div_3); { var consequent_7 = ($$anchor5) => { var fragment_4 = comment(); var node_12 = first_child(fragment_4); component(node_12, () => { var _a2; return (_a2 = get$1(currentComponent)) == null ? undefined : _a2.ref; }, ($$anchor6, $$component) => { $$component($$anchor6, spread_props(() => { var _a2; return (_a2 = get$1(currentComponent)) == null ? undefined : _a2.props; }, { get parent() { return get$1(parent); }, children: ($$anchor7, $$slotProps) => { var fragment_5 = comment(); var node_13 = first_child(fragment_5); html(node_13, () => { var _a2; return (_a2 = get$1(currentComponent)) == null ? undefined : _a2.slot; }); append($$anchor7, fragment_5); }, $$slots: { default: true } })); }); append($$anchor5, fragment_4); }; var alternate_2 = ($$anchor5) => { var fragment_6 = comment(); var node_14 = first_child(fragment_6); component(node_14, () => { var _a2; return (_a2 = get$1(currentComponent)) == null ? undefined : _a2.ref; }, ($$anchor6, $$component) => { $$component($$anchor6, spread_props(() => { var _a2; return (_a2 = get$1(currentComponent)) == null ? undefined : _a2.props; }, { get parent() { return get$1(parent); } })); }); append($$anchor5, fragment_6); }; if_block(node_11, ($$render) => { var _a2; if ((_a2 = get$1(currentComponent)) == null ? undefined : _a2.slot) $$render(consequent_7); else $$render(alternate_2, false); }); } reset(div_3); bind_this(div_3, ($$value) => set(modalElement, $$value), () => get$1(modalElement)); template_effect(() => { var _a2; set_class(div_3, `modal contents ${((_a2 = $modalStore()[0]) == null ? undefined : _a2.modalClasses) ?? "" ?? ""}`); set_attribute(div_3, "aria-label", $modalStore()[0].title ?? ""); }); append($$anchor4, div_3); }; if_block(node_2, ($$render) => { if ($modalStore()[0].type !== "component") $$render(consequent_6); else $$render(alternate_3, false); }); } reset(div_1); reset(div); effect(() => event("mousedown", div, onBackdropInteractionBegin)); effect(() => event("mouseup", div, onBackdropInteractionEnd)); effect(() => event( "touchstart", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }, undefined, true )); effect(() => event( "touchend", div, function($$arg) { bubble_event.call(this, $$props, $$arg); }, undefined, true )); action(div, ($$node, $$action_arg) => focusTrap == null ? undefined : focusTrap($$node, $$action_arg), () => true); template_effect(() => { set_class(div, `modal-backdrop ${get$1(classesBackdrop) ?? ""} ${get$1(backdropOverflow) ?? ""}`); set_class(div_1, `modal-transition ${get$1(classesTransitionLayer) ?? ""}`); }); transition(5, div_1, () => dynamicTransition, () => ({ transition: transitionIn(), params: transitionInParams(), enabled: transitions() })); transition(6, div_1, () => dynamicTransition, () => ({ transition: transitionOut(), params: transitionOutParams(), enabled: transitions() })); transition(7, div, () => dynamicTransition, () => ({ transition: fade, params: { duration: 150 }, enabled: transitions() })); append($$anchor3, div); }); append($$anchor2, fragment_1); }; if_block(node, ($$render) => { if ($modalStore().length > 0) $$render(consequent_8); }); } bind_window_size("innerHeight", ($$value) => set(windowHeight, $$value)); append($$anchor, fragment); pop(); $$cleanup(); } var root_4$3 = /* @__PURE__ */ template(`<button><!></button>`); var root_5$2 = /* @__PURE__ */ template(`<button aria-label="Dismiss toast"> </button>`); var root_3$5 = /* @__PURE__ */ template(`<div><!> <!></div>`); var root_2$3 = /* @__PURE__ */ template(`<div aria-live="polite"><div data-testid="toast"><div class="text-base"><!></div> <!></div></div>`); var root_1$a = /* @__PURE__ */ template(`<div data-testid="snackbar-wrapper"><div></div></div>`); function Toast($$anchor, $$props) { const $$sanitized_props = legacy_rest_props($$props, [ "children", "$$slots", "$$events", "$$legacy" ]); push($$props, false); const [$$stores, $$cleanup] = setup_stores(); const $prefersReducedMotionStore = () => store_get(prefersReducedMotionStore, "$prefersReducedMotionStore", $$stores); const $toastStore = () => store_get(toastStore, "$toastStore", $$stores); const classesWrapper = mutable_state(); const classesSnackbar = mutable_state(); const classesToast = mutable_state(); const filteredToasts = mutable_state(); const toastStore = getToastStore(); let position = prop($$props, "position", 8, "b"); let max = prop($$props, "max", 8, 3); let background = prop($$props, "background", 8, "variant-filled-secondary"); let width = prop($$props, "width", 8, "max-w-[640px]"); let color = prop($$props, "color", 8, ""); let padding = prop($$props, "padding", 8, "p-4"); let spacing = prop($$props, "spacing", 8, "space-x-4"); let rounded = prop($$props, "rounded", 8, "rounded-container-token"); let shadow = prop($$props, "shadow", 8, "shadow-lg"); let zIndex = prop($$props, "zIndex", 8, "z-[888]"); let buttonAction = prop($$props, "buttonAction", 8, "btn variant-filled"); let buttonDismiss = prop($$props, "buttonDismiss", 8, "btn-icon btn-icon-sm variant-filled"); let buttonDismissLabel = prop($$props, "buttonDismissLabel", 8, "✕"); let transitions = prop($$props, "transitions", 24, () => !$prefersReducedMotionStore()); let transitionIn = prop($$props, "transitionIn", 8, fly); let transitionInParams = prop($$props, "transitionInParams", 24, () => ({ duration: 250 })); let transitionOut = prop($$props, "transitionOut", 8, fly); let transitionOutParams = prop($$props, "transitionOutParams", 24, () => ({ duration: 250 })); const cWrapper = "flex fixed top-0 left-0 right-0 bottom-0 pointer-events-none"; const cSnackbar = "flex flex-col gap-y-2"; const cToast = "flex justify-between items-center pointer-events-auto"; const cToastActions = "flex items-center space-x-2"; let cPosition = mutable_state(); let cAlign = mutable_state(); let animAxis = mutable_state({ x: 0, y: 0 }); switch (position()) { case "t": set(cPosition, "justify-center items-start"); set(cAlign, "items-center"); set(animAxis, { x: 0, y: -100 }); break; case "b": set(cPosition, "justify-center items-end"); set(cAlign, "items-center"); set(animAxis, { x: 0, y: 100 }); break; case "l": set(cPosition, "justify-start items-center"); set(cAlign, "items-start"); set(animAxis, { x: -100, y: 0 }); break; case "r": set(cPosition, "justify-end items-center"); set(cAlign, "items-end"); set(animAxis, { x: 100, y: 0 }); break; case "tl": set(cPosition, "justify-start items-start"); set(cAlign, "items-start"); set(animAxis, { x: -100, y: 0 }); break; case "tr": set(cPosition, "justify-end items-start"); set(cAlign, "items-end"); set(animAxis, { x: 100, y: 0 }); break; case "bl": set(cPosition, "justify-start items-end"); set(cAlign, "items-start"); set(animAxis, { x: -100, y: 0 }); break; case "br": set(cPosition, "justify-end items-end"); set(cAlign, "items-end"); set(animAxis, { x: 100, y: 0 }); break; } function onAction(index2) { var _a2, _b2; (_b2 = (_a2 = $toastStore()[index2]) == null ? undefined : _a2.action) == null ? undefined : _b2.response(); toastStore.close($toastStore()[index2].id); } function onMouseEnter(index2) { var _a2; if ((_a2 = $toastStore()[index2]) == null ? undefined : _a2.hoverable) { toastStore.freeze(index2); set(classesSnackbar, get$1(classesSnackbar) + " scale-[105%]"); } } function onMouseLeave(index2) { var _a2; if ((_a2 = $toastStore()[index2]) == null ? undefined : _a2.hoverable) { toastStore.unfreeze(index2); set(classesSnackbar, get$1(classesSnackbar).replace(" scale-[105%]", "")); } } let wrapperVisible = mutable_state(false); legacy_pre_effect( () => (get$1(cPosition), deep_read_state(zIndex()), deep_read_state($$sanitized_props)), () => { set(classesWrapper, `${cWrapper} ${get$1(cPosition)} ${zIndex()} ${$$sanitized_props.class || ""}`); } ); legacy_pre_effect( () => (get$1(cAlign), deep_read_state(padding())), () => { set(classesSnackbar, `${cSnackbar} ${get$1(cAlign)} ${padding()}`); } ); legacy_pre_effect( () => (deep_read_state(width()), deep_read_state(color()), deep_read_state(padding()), deep_read_state(spacing()), deep_read_state(rounded()), deep_read_state(shadow())), () => { set(classesToast, `${cToast} ${width()} ${color()} ${padding()} ${spacing()} ${rounded()} ${shadow()}`); } ); legacy_pre_effect( () => ($toastStore(), deep_read_state(max())), () => { set(filteredToasts, Array.from($toastStore()).slice(0, max())); } ); legacy_pre_effect(() => get$1(filteredToasts), () => { if (get$1(filteredToasts).length) { set(wrapperVisible, true); } }); legacy_pre_effect_reset(); init(); var fragment = comment(); var node = first_child(fragment); { var consequent_3 = ($$anchor2) => { var div = root_1$a(); var div_1 = child(div); each(div_1, 15, () => get$1(filteredToasts), (t2) => t2, ($$anchor3, t2, i) => { var div_2 = root_2$3(); var div_3 = child(div_2); var div_4 = child(div_3); var node_1 = child(div_4); html(node_1, () => get$1(t2).message); reset(div_4); var node_2 = sibling(div_4, 2); { var consequent_2 = ($$anchor4) => { var div_5 = root_3$5(); set_class(div_5, `toast-actions ${cToastActions}`); var node_3 = child(div_5); { var consequent = ($$anchor5) => { var button2 = root_4$3(); var node_4 = child(button2); html(node_4, () => get$1(t2).action.label); reset(button2); template_effect(() => set_class(button2, clsx(buttonAction()))); event("click", button2, () => onAction(get$1(i))); append($$anchor5, button2); }; if_block(node_3, ($$render) => { if (get$1(t2).action) $$render(consequent); }); } var node_5 = sibling(node_3, 2); { var consequent_1 = ($$anchor5) => { var button_1 = root_5$2(); var text2 = child(button_1, true); reset(button_1); template_effect(() => { set_class(button_1, clsx(buttonDismiss())); set_text(text2, buttonDismissLabel()); }); event("click", button_1, () => toastStore.close(get$1(t2).id)); append($$anchor5, button_1); }; if_block(node_5, ($$render) => { if (!get$1(t2).hideDismiss) $$render(consequent_1); }); } reset(div_5); append($$anchor4, div_5); }; if_block(node_2, ($$render) => { if (get$1(t2).action || !get$1(t2).hideDismiss) $$render(consequent_2); }); } reset(div_3); reset(div_2); template_effect(() => { set_attribute(div_2, "role", get$1(t2).hideDismiss ? "alert" : "alertdialog"); set_class(div_3, `toast ${get$1(classesToast) ?? ""} ${get$1(t2).background ?? background() ?? ""} ${get$1(t2).classes ?? "" ?? ""}`); }); animation(div_2, () => flip, () => ({ duration: transitions() ? 250 : 0 })); transition(5, div_2, () => dynamicTransition, () => ({ transition: transitionIn(), params: { x: get$1(animAxis).x, y: get$1(animAxis).y, ...transitionInParams() }, enabled: transitions() })); transition(6, div_2, () => dynamicTransition, () => ({ transition: transitionOut(), params: { x: get$1(animAxis).x, y: get$1(animAxis).y, ...transitionOutParams() }, enabled: transitions() })); event("outroend", div_2, () => { const outroFinishedForLastToastOnQueue = get$1(filteredToasts).length === 0; if (outroFinishedForLastToastOnQueue) set(wrapperVisible, false); }); event("mouseenter", div_2, () => onMouseEnter(get$1(i))); event("mouseleave", div_2, () => onMouseLeave(get$1(i))); append($$anchor3, div_2); }); reset(div_1); reset(div); template_effect(() => { set_class(div, `snackbar-wrapper ${get$1(classesWrapper) ?? ""}`); set_class(div_1, `snackbar ${get$1(classesSnackbar) ?? ""}`); }); append($$anchor2, div); }; if_block(node, ($$render) => { if (get$1(filteredToasts).length > 0 || get$1(wrapperVisible)) $$render(consequent_3); }); } append($$anchor, fragment); pop(); $$cleanup(); } const creditCode = ""; const githubMark = `<svg viewBox="0 0 98 96" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" clip-rule="evenodd" d="M48.854 0C21.839 0 0 22 0 49.217c0 21.756 13.993 40.172 33.405 46.69 2.427.49 3.316-1.059 3.316-2.362 0-1.141-.08-5.052-.08-9.127-13.59 2.934-16.42-5.867-16.42-5.867-2.184-5.704-5.42-7.17-5.42-7.17-4.448-3.015.324-3.015.324-3.015 4.934.326 7.523 5.052 7.523 5.052 4.367 7.496 11.404 5.378 14.235 4.074.404-3.178 1.699-5.378 3.074-6.6-10.839-1.141-22.243-5.378-22.243-24.283 0-5.378 1.94-9.778 5.014-13.2-.485-1.222-2.184-6.275.486-13.038 0 0 4.125-1.304 13.426 5.052a46.97 46.97 0 0 1 12.214-1.63c4.125 0 8.33.571 12.213 1.63 9.302-6.356 13.427-5.052 13.427-5.052 2.67 6.763.97 11.816.485 13.038 3.155 3.422 5.015 7.822 5.015 13.2 0 18.905-11.404 23.06-22.324 24.283 1.78 1.548 3.316 4.481 3.316 9.126 0 6.6-.08 11.897-.08 13.526 0 1.304.89 2.853 3.316 2.364 19.412-6.52 33.405-24.935 33.405-46.691C97.707 22 75.788 0 48.854 0z"/></svg>`; var root_1$9 = /* @__PURE__ */ template(`<div data-theme="skeleton"><section class="overflow-hidden h-full"><!></section> <div class="absolute top-2 right-2 z-1 select-none flex items-center gap-1"><a target="_blank" href="https://github.com/drunkg00se/Pixiv-Downloader" class="w-5 fill-current"><!></a> <button class="btn-icon btn-icon-sm bg-transparent font-bold hover:text-xl">✕</button></div></div>`); function ModalWrapper($$anchor, $$props) { push($$props, true); const [$$stores, $$cleanup] = setup_stores(); const $modalStore = () => store_get(modalStore, "$modalStore", $$stores); let padding = prop($$props, "padding", 3, "py-6 px-8"), width = prop($$props, "width", 3, "w-full md:max-w-screen-sm lg:max-w-screen-md xl:max-w-screen-lg"), height = prop($$props, "height", 3, ""), classProp = prop($$props, "class", 3, ""); const classes = /* @__PURE__ */ derived(() => `${padding()} ${width()} ${height()} ${classProp()}`); const modalStore = getModalStore(); var fragment = comment(); var node = first_child(fragment); { var consequent = ($$anchor2) => { var div = root_1$9(); var section = child(div); var node_1 = child(section); snippet(node_1, () => $$props.children); reset(section); var div_1 = sibling(section, 2); var a = child(div_1); var node_2 = child(a); html(node_2, () => githubMark); reset(a); var button2 = sibling(a, 2); button2.__click = function(...$$args) { var _a2; (_a2 = $$props.parent.onClose) == null ? undefined : _a2.apply(this, $$args); }; reset(div_1); reset(div); template_effect(() => set_class(div, `relative rounded-container-token shadow-xl bg-scroll ${get$1(classes) ?? ""}`)); append($$anchor2, div); }; if_block(node, ($$render) => { if ($modalStore()[0]) $$render(consequent); }); } append($$anchor, fragment); pop(); $$cleanup(); } delegate(["click"]); const setting$1 = { save_to: { title: "Save To", label: { directory: "Save Path", filename: "Filename" }, options: { use_fsa: "Use FileSystemAccess API", fsa_directory: "Select directory", fsa_filename_conflict: "When filename conflicts", tag_language: "Tag Language", tag_language_tips: "Tags without translation may still be in another language" }, button: { choose_fsa_directory: "Browse" }, radio: { filename_conflict_option_uniquify: "Uniquify", filename_conflict_option_overwrite: "Overwrite", filename_conflict_option_prompt: "Prompt" }, placeholder: { sub_directory_unused: "Leave folder name blank if not saving to a subdirectory", vm_not_supported: "Not supported by Violentmonkey", need_browser_api: "Browser API required", filename_requried: "Required" } }, ugoira: { title: "Ugoira", label: { format: "Ugoira Format", quality: "Ugoira Quality" }, options: { select_format: "Convert Ugoira to selected format", gif_tips: "Lower values produce better colors, but slow processing significantly", webm_tips: "0 (worst) to 99 (best)", webp_lossy: "Lossless Webp", webp_quality: "Image Quality", webp_quality_tips: "For lossy, 0 gives the smallest size and 100 the largest. For lossless, 0 is the fastest but gives larger files compared to the slowest, but best, 100.", webp_method: "Compression Method", webp_method_tips: "Quality/speed trade-off (0=fast, 6=slower-better)", png_tips: "Number of colors in the result; 0: all colors (lossless PNG)" } }, history: { title: "History", label: { scheduled_backups: "Scheduled Backups", "export": "Export", "import": "Import", clear: "Clear" }, options: { scheduled_backups: "Automatically back up download history at selected intervals.", export_as_json: "Export download history as JSON file", export_as_csv: "Export download history as CSV file", import_json: "Import JSON format download history", clear_history: "Clear download history" }, button: { "export": "Export", "import": "Import", clear: "Clear" }, select: { backup_interval_never: "Never", backup_interval_every_day: "Every day", backup_interval_every_7_day: "7 days", backup_interval_every_30_day: "30 days" }, text: { confirm_clear_history: "Do you really want to clear history?" } }, button_position: { title: "Button", label: { common: "Common", my_bookmark: "My Bookmark", gallery: "Gallery" }, options: { horizontal_position: "Horizontal Position", vertical_position: "Vertical Position", horizontal_alignment: "Horizontal Alignment", vertical_alignment: "Vertical Alignment" } }, authorization: { title: "Auth" }, others: { title: "Others", options: { show_setting_button: "Show Setting Button", bundle_multipage_illust: "Bundle multipage illustrations into a zip file", bundle_manga: "Bundle manga into a zip file", like_illust_when_downloading: "Like the artwork when downloading", add_bookmark_when_downloading: "Bookmark artwork when downloading", add_bookmark_with_tags: "Add tags when bookmarking", add_bookmark_private_r18: "Bookmark R-18 artwork as private", option_does_not_apply_to_batch_download: "This option does not apply to batch download" } }, feedback: { title: "Feedback", label: { feedback: "Feedback", donate: "Donate" }, text: { feedback_desc: 'If you encounter any issues or have suggestions for improvements, feel free to provide feedback <a href="https://github.com/drunkg00se/Pixiv-Downloader/issues" target="_blank" class=" anchor">here.</a>', give_me_a_star: 'If the script is helpful to you, please <a href="https://github.com/drunkg00se/Pixiv-Downloader" target="_blank" class="anchor">click here and give me a ⭐Star on GitHub.</a>', donate_desc: "Or, buy me a cola. ^_^" } } }; const downloader$2 = { category: { tab_name: "Category", filter: { exclude_downloaded: "Exclude Downloaded", exclude_blacklist: "Exclude Blacklist", image: "Image", video: "Video", download_all_pages: "All Pages", download_selected_pages: "Custom Pages", pixiv_illust: "Illustration", pixiv_manga: "Manga", pixiv_ugoira: "Ugoira" } }, tag_filter: { tab_name: "Tags", placeholder: { blacklist_tag: "Blacklist: Exclude works with these tags.", whitelist_tag: "Whitelist: Only download works with these tags." } }, others: { tab_name: "others", options: { retry_failed: "Retry failed image downloads." } }, download_type: { stop: "Stop", pixiv_works: "Works", pixiv_bookmark: "Bookmarks", pixiv_bookmark_public: "Public", pixiv_bookmark_private: "Private", pixiv_follow_latest_all: "All", pixiv_follow_latest_r18: "R-18", pixiv_series: "Series", moebooru_posts: "Posts", moebooru_pool: "Pool", moebooru_popular_1d: "1d", moebooru_popular_1w: "1w", moebooru_popular_1m: "1m", moebooru_popular_1y: "1y", moebooru_popular_date: "Popular" }, file_handle_not_found: { title: "Duplicate / Invalid File Name", button: { save: "Save", cancel: "Cancel" } }, failed_downloads: "Failed downloads" }; const button$1 = { setting: "Setting" }; const changelog$1 = { feedback: "Feedback / Report an issue", credit: "Click here to support me!", give_me_a_star: "Give me a ⭐Star on GitHub.", buy_me_a_drink: "Or, buy me a vanilla-flavored iced cola. ^_^" }; const toast$1 = { message: { pick_directory_handle: "Select a folder to save", file_handle_not_found: "The file name is duplicated or invalid. Please save it manually.", request_directory_handle_permission: "Please allow permission to edit files and folders." }, actionLabel: { browse: "Browse", save: "Save", allow_permission: "Allow" } }; const en = { setting: setting$1, downloader: downloader$2, button: button$1, changelog: changelog$1, toast: toast$1 }; const setting = { save_to: { title: "保存至", label: { directory: "保存位置", filename: "文件名" }, options: { use_fsa: "使用FileSystemAccess API", fsa_directory: "选择保存文件夹", fsa_filename_conflict: "当文件名重复时", tag_language: "使用的标签语言", tag_language_tips: "无翻译的标签仍可能是其他语言" }, button: { choose_fsa_directory: "浏览" }, radio: { filename_conflict_option_uniquify: "重命名", filename_conflict_option_overwrite: "覆盖", filename_conflict_option_prompt: "提示" }, placeholder: { sub_directory_unused: "如不需要保存到子目录,此行留空即可", vm_not_supported: "Violentmonkey不支持", need_browser_api: '请将下载模式设置为"Browser Api"', filename_requried: "必填" } }, ugoira: { title: "动图转换", label: { format: "动图格式", quality: "动图质量" }, options: { select_format: "将动图转换到所选格式", gif_tips: "数值越低颜色越好,但处理速度显著减慢", webm_tips: "0最差,99最好", webp_lossy: "无损转换", webp_quality: "图片质量", webp_quality_tips: "有损:0表示文件最小,100表示文件最大。无损:0最快,但文件较大,100最慢,但质量最好。", webp_method: "压缩方法", webp_method_tips: "0=快,6=慢但效果更好", png_tips: "颜色数量。0:所有颜色(无损PNG)" } }, history: { title: "下载历史", label: { scheduled_backups: "定期备份", "export": "导出", "import": "导入", clear: "清理" }, options: { scheduled_backups: "以选定的时间间隔自动备份下载历史", export_as_json: "将下载历史导出为JSON文件", export_as_csv: "将下载历史导出为CSV文件", import_json: "导入JSON格式下载历史", clear_history: "清除下载历史" }, button: { "export": "导出记录", "import": "导入记录", clear: "清除记录" }, select: { backup_interval_never: "不备份", backup_interval_every_day: "每天", backup_interval_every_7_day: "每7天", backup_interval_every_30_day: "每30天" }, text: { confirm_clear_history: "真的要清除历史记录吗?" } }, button_position: { title: "按钮位置", label: { common: "通用", my_bookmark: "我的收藏", gallery: "画廊" }, options: { horizontal_position: "水平位置", vertical_position: "垂直位置", horizontal_alignment: "水平对齐", vertical_alignment: "垂直对齐" } }, authorization: { title: "网站认证" }, others: { title: "其它", options: { show_setting_button: "显示设置按钮", bundle_multipage_illust: "将多页插图打包为zip压缩包", bundle_manga: "将漫画作品打包为zip压缩包", like_illust_when_downloading: "下载作品时点赞", add_bookmark_when_downloading: "下载作品时收藏", add_bookmark_with_tags: "收藏时添加作品标签", add_bookmark_private_r18: "将R-18作品收藏到不公开类别", option_does_not_apply_to_batch_download: "批量下载不适用" } }, feedback: { title: "反馈 / 赞赏", label: { feedback: "反馈", donate: "赞赏" }, text: { feedback_desc: '如果你在使用中发现了问题或有改进建议,欢迎到<a href="https://github.com/drunkg00se/Pixiv-Downloader/issues" target="_blank" class="anchor">此链接</a>反馈。', give_me_a_star: '如果脚本有帮助到你,<a href="https://github.com/drunkg00se/Pixiv-Downloader" target="_blank" class="anchor">欢迎点此在GitHub中给我一个 ⭐Star。</a>', donate_desc: "或者,扫码请我喝杯可乐 ^_^" } } }; const downloader$1 = { category: { tab_name: "类别", filter: { exclude_downloaded: "排除已下载", exclude_blacklist: "排除黑名单", image: "图片", video: "视频", download_all_pages: "下载所有页", download_selected_pages: "自定义页数", pixiv_illust: "插画", pixiv_manga: "漫画", pixiv_ugoira: "动图" } }, tag_filter: { tab_name: "标签", placeholder: { blacklist_tag: "黑名单,将排除含有以下标签的作品。", whitelist_tag: "白名单,只下载包含以下标签的作品。" } }, others: { tab_name: "其它", options: { retry_failed: "对首次下载失败的图片进行重试" } }, download_type: { stop: "停止", pixiv_works: "作品", pixiv_bookmark: "收藏", pixiv_bookmark_public: "公开收藏", pixiv_bookmark_private: "不公开收藏", pixiv_follow_latest_all: "全部", pixiv_follow_latest_r18: "R-18", pixiv_series: "系列", moebooru_posts: "投稿", moebooru_pool: "图集", moebooru_popular_1d: "日", moebooru_popular_1w: "周", moebooru_popular_1m: "月", moebooru_popular_1y: "年", moebooru_popular_date: "人气" }, file_handle_not_found: { title: "重复文件名 / 无效文件名", button: { save: "保存", cancel: "取消" } }, failed_downloads: "下载失败" }; const button = { setting: "设置" }; const changelog = { feedback: "有问题or想建议?这里反馈", credit: "脚本还行?请点这里支持我!", give_me_a_star: "在GitHub中给我一个 ⭐Star,", buy_me_a_drink: "或者,扫码请我喝杯香草味冰可乐。^_^" }; const toast = { message: { pick_directory_handle: "请选择保存文件夹", file_handle_not_found: "文件名重复或无效,请手动保存", request_directory_handle_permission: "请授予权限以修改文件/文件夹" }, actionLabel: { browse: "浏览", save: "保存", allow_permission: "同意" } }; const zh = { setting, downloader: downloader$1, button, changelog, toast }; function createI18n(option) { let locale2 = state(proxy(option.locale)); const setlocale2 = (newlocale) => { if (newlocale === get$1(locale2) || !Object.keys(option.message).includes(newlocale)) return; set(locale2, proxy(newlocale)); }; const t2 = (jsonPath) => { const paths = jsonPath.split("."); let msg = option.message[get$1(locale2)]; for (const path of paths) { msg = msg[path]; if (msg === undefined) return "null"; } return msg; }; return { t: t2, setlocale: setlocale2, locale: { get current() { return get$1(locale2); } }, get availableLocales() { return Object.keys(option.message); } }; } const message = { zh, en }; const browserLocale = navigator.language.split("-")[0]; const { t, setlocale, locale, availableLocales } = createI18n({ locale: browserLocale in message ? browserLocale : "en", message }); var on_click$2 = (_, showCreditCode) => set(showCreditCode, !get$1(showCreditCode)); var root_1$8 = /* @__PURE__ */ template( `<header class="modal-header text-2xl font-bold"></header> <article class="modal-body mt-4"><p>本次更新提高了FileSystemAccess API保存文件的可用性,现在应该可以替代篡改猴"浏览器API"下载模式,请大家尝试一下,反馈问题。</p> <h4 class=" text-xl mt-2">新增</h4> <ul class="list-disc list-inside leading-loose"><li>现在启用“使用FileSystemAccess API”,授予浏览器文件权限后,不需要再次选择文件夹。</li> <li>启用“使用FileSystemAccess API”后,当图片文件名无效时,脚本现在会弹出提示,而不是默认失败(请在文件保存框中修改文件名,直接保存仍然会抛出错误)。</li> <li>启用“使用FileSystemAccess API”后,批量下载时,会列出无法保存的图片,请手动保存或取消。</li> <li>批量下载完成后,会列出下载失败的图片及其原因。</li> <li>为脚本的一些操作增加了提示信息。</li></ul> <h4 class=" text-xl mt-2">修复</h4> <ul class="list-disc list-inside leading-loose"><li>Large files throw security error (<a href="https://github.com/drunkg00se/Pixiv-Downloader/issues/39" target="_blank" class="anchor">#39</a>)</li> <li>修复在rule34vault中进行前进/后退导航时,头像/批量下载项未正确更新的问题。</li></ul> <h4 class=" text-xl mt-2">已知问题</h4> <ul class="list-disc list-inside leading-loose"><li>使用篡改猴并启用浏览器API时,Firefox 137下载较大图片会出现下载成功但无法保存的问题。如您在使用中遇到此问题,请回滚到版本136直至篡改猴正式版更新至v5.4或开发版更新至v5.4.6226。</li> <li>使用篡改猴开发版v5.4.6226并启用浏览器API时,无法创建子文件夹,请尝试使用FileSystemAccess API。</li></ul></article> <footer class="modal-footer mt-4"><div class="flex justify-between items-center text-sm"><button> </button> <a target="_blank" href="https://github.com/drunkg00se/Pixiv-Downloader/issues"> </a></div> <div><div class="flex justify-center items-center min-h-0 gap-14 overflow-hidden"><img alt="credit" class="rounded-full"> <p class="flex flex-col h-full justify-evenly"><a href="https://github.com/drunkg00se/Pixiv-Downloader" target="_blank" class="anchor"> </a> <span> </span></p></div></div></footer>`, 1 ); function Changelog($$anchor, $$props) { push($$props, true); const anchorFocus = `focus:!outline-none focus:decoration-wavy`; const anchor = `leading-loose anchor underline-offset-2 ${anchorFocus}`; let showCreditCode = state(false); const gridRows = /* @__PURE__ */ derived(() => get$1(showCreditCode) ? "grid-rows-[1fr] mt-2" : "grid-rows-[0fr]"); ModalWrapper($$anchor, { get parent() { return $$props.parent; }, children: ($$anchor2, $$slotProps) => { var fragment_1 = root_1$8(); var header = first_child(fragment_1); header.textContent = `Pixiv Downloader ${"1.10.0"}`; var footer = sibling(header, 4); var div = child(footer); var button2 = child(div); set_class(button2, clsx(anchor)); button2.__click = [on_click$2, showCreditCode]; var text2 = child(button2, true); template_effect(() => set_text(text2, t("changelog.credit"))); reset(button2); var a = sibling(button2, 2); set_class(a, clsx(anchor)); var text_1 = child(a, true); template_effect(() => set_text(text_1, t("changelog.feedback"))); reset(a); reset(div); var div_1 = sibling(div, 2); var div_2 = child(div_1); var img = child(div_2); set_attribute(img, "src", creditCode); var p = sibling(img, 2); var a_1 = child(p); var text_2 = child(a_1, true); template_effect(() => set_text(text_2, t("changelog.give_me_a_star"))); reset(a_1); var span = sibling(a_1, 2); var text_3 = child(span, true); template_effect(() => set_text(text_3, t("changelog.buy_me_a_drink"))); reset(span); reset(p); reset(div_2); reset(div_1); reset(footer); template_effect(() => set_class(div_1, `grid transition-[grid-template-rows] duration-[400ms] ${get$1(gridRows) ?? ""}`)); append($$anchor2, fragment_1); }, $$slots: { default: true } }); pop(); } delegate(["click"]); const downloadSvg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><path 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"></path></svg>`; const stopSvg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><path 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"></path></svg> `; const playSvg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12,20C7.59,20 4,16.41 4,12C4,7.59 7.59,4 12,4C16.41,4 20,7.59 20,12C20,16.41 16.41,20 12,20M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2M10,16.5L16,12L10,7.5V16.5Z" /></svg>`; const stopOutLineSvg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2M12,4C16.41,4 20,7.59 20,12C20,16.41 16.41,20 12,20C7.59,20 4,16.41 4,12C4,7.59 7.59,4 12,4M9,9V15H15V9" /></svg>`; const downloadMultipleSvg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M9,1V7H5L12,14L19,7H15V1H9M5,16V18H19V16H5M5,20V22H19V20H5Z" /></svg>`; const inputValidation = (node, value) => { user_effect(() => { node.value = String(value.get()); }); user_effect(() => { const off = on(node, "input", function validate() { const isValid = this.reportValidity(); if (isValid) { this.classList.remove("input-error"); value.set(+this.value); } else { this.classList.add("input-error"); } }); return () => { off(); }; }); }; const storeStr = localStorage.getItem("pdl-downloader_option"); if (storeStr) localStorage.removeItem("pdl-downloader_option"); const legacyStore = storeStr ? JSON.parse(storeStr) : {}; const batchDownloaderStore = createPersistedStore("pdl-batch-downloader", { selectedFilters: legacyStore.selectedFilters ?? null, blacklistTag: legacyStore.blacklistTag ?? [], whitelistTag: legacyStore.whitelistTag ?? [], downloadAllPages: legacyStore.downloadAllPages ?? true, pageStart: legacyStore.pageStart ?? 1, pageEnd: legacyStore.pageEnd ?? 1, retryFailed: legacyStore.retryFailed ?? false }); let ReactiveValue$1 = (_a = class { /** * * @param {() => T} fn * @param {(update: () => void) => void} onsubscribe */ constructor(fn, onsubscribe) { __privateAdd(this, _fn); __privateAdd(this, _subscribe); __privateSet(this, _fn, fn); __privateSet(this, _subscribe, createSubscriber(onsubscribe)); } get current() { __privateGet(this, _subscribe).call(this); return __privateGet(this, _fn).call(this); } }, _fn = new WeakMap(), _subscribe = new WeakMap(), _a); const parenthesis_regex = /\(.+\)/; class MediaQuery extends ReactiveValue$1 { /** * @param {string} query A media query string * @param {boolean} [fallback] Fallback value for the server */ constructor(query, fallback) { let final_query = parenthesis_regex.test(query) ? query : `(${query})`; const q = window.matchMedia(final_query); super( () => q.matches, (update2) => on(q, "change", update2) ); } } class ReactiveValue { constructor(fn, onsubscribe) { __privateAdd(this, _fn2); __privateAdd(this, _subscribe2); __privateSet(this, _fn2, fn); __privateSet(this, _subscribe2, createSubscriber(onsubscribe)); } get current() { __privateGet(this, _subscribe2).call(this); return __privateGet(this, _fn2).call(this); } } _fn2 = new WeakMap(); _subscribe2 = new WeakMap(); const EVENT_FILE_HANDLE_NOT_FOUND = "event-file-handle-not-fount"; const EVENT_DIR_HANDLE_NOT_FOUND = "event-dir-handle-not-fount"; const EVENT_REQUEST_USER_ACTIVATION = "event-request-user-activation"; class RequestError extends Error { constructor(url, status) { super(status + " " + url); __publicField(this, "url"); __publicField(this, "status"); this.name = "RequestError"; this.url = url; this.status = status; } } class CancelError extends Error { constructor() { super("User aborted."); this.name = "CancelError"; } } class JsonDataError extends Error { constructor(msg) { super(msg); this.name = "JsonDataError"; } } class TimoutError extends Error { constructor(msg) { super(msg); this.name = "TimoutError"; } } class PermissionError extends Error { constructor() { super("Permission denied."); this.name = "PermissionError"; } } class InvalidPostError extends Error { constructor(id) { super(`Invalid post id: ${id}.`); this.name = "InvalidPostError"; } } var root_3$4 = /* @__PURE__ */ template(`<!> <!> <!>`, 1); var root_12$1 = /* @__PURE__ */ template(`<label class="btn !py-2 rounded-none !transform-none cursor-pointer variant-soft-surface has-[:checked]:!variant-filled-primary text-sm w-full"><div class="w-0 h-0 overflow-hidden hidden"><input type="checkbox"></div> <div class="!m-0"> </div></label>`); var root_11$2 = /* @__PURE__ */ template(`<div class="flex justify-end items-center my-4"><div class="btn-group w-full"></div></div>`); var root_13$1 = /* @__PURE__ */ template(`<!> <!>`, 1); var root_10$1 = /* @__PURE__ */ template(`<!> <div class="flex justify-between items-center my-4 gap-4"><div class="flex-grow w-full"><!></div> <div class="flex justify-between items-center gap-4 w-full"><label class="input-group input-group-divider flex [&>input]:!min-w-0 [&>input]:!border-transparent border-surface-400/20 dark:border-surface-500/20 bg-surface-400/20 dark:bg-surface-500/20"><div class="input-group-shim !px-1 flex-none"><i class="w-6 fill-current"><!></i></div> <input class="w-20 pr-0 text-surface-700-200-token text-sm" type="number" min="1" step="1" required></label> <label class="input-group input-group-divider flex [&>input]:!min-w-0 [&>input]:!border-transparent border-surface-400/20 dark:border-surface-500/20 bg-surface-400/20 dark:bg-surface-500/20"><div class="input-group-shim !px-1 flex-none"><i class="w-6 fill-current"><!></i></div> <input class="w-20 pr-0 text-surface-700-200-token text-sm" type="number" min="1" step="1" required></label></div></div>`, 1); var root_17$1 = /* @__PURE__ */ template(`<!> <!>`, 1); var root_19$1 = /* @__PURE__ */ template(`<div class="flex justify-between items-center text-base text-surface-700-200-token"><p> </p> <!></div>`); var root_2$2 = /* @__PURE__ */ template(`<div class="downloader-filter"><!> <hr class="!border-t-1 my-4"></div>`); var on_click$1 = (_, startDownload, id) => { startDownload(id()); }; var root_23 = /* @__PURE__ */ template(`<button class="btn rounded-none !transform-none !variant-filled-primary"><i class="w-5"><!></i> <span> </span></button>`); var root_21 = /* @__PURE__ */ template(`<div class=" flex-none btn-group self-start"></div>`); var on_click_1 = (__1, startDownload, batchDownloadEntries) => { var _a2; startDownload(((_a2 = get$1(batchDownloadEntries)) == null ? undefined : _a2[0][0]) ?? ""); }; var root_25 = /* @__PURE__ */ template(`<button class="btn variant-filled-primary self-start"><i class="w-5"><!></i> <span> </span></button>`); var root_20 = /* @__PURE__ */ template(`<div class="flex justify-end flex-grow w-full gap-4"><div class="flex flex-grow flex-col justify-between overflow-hidden text-surface-700-200-token"><p class="truncate"> </p> <p class="break-words"> </p></div> <!></div>`); var on_click_2 = (__2, abort) => { abort(); }; var root_26 = /* @__PURE__ */ template(`<div class="flex flex-grow w-full gap-6 items-center"><div class="flex flex-grow flex-col justify-between h-full overflow-hidden"><!> <div class="flex items-center justify-between gap-4 basis-0 text-surface-700-200-token"><p class="truncate"> </p> <p class=" flex-none"> </p></div></div> <button class="btn variant-filled-primary"><i class="w-5"><!></i> <span> </span></button></div>`); var root_28 = /* @__PURE__ */ template(`<li><div class="flex-1 flex flex-col overflow-hidden"><span class="text-base truncate"> </span> <span class="text-sm text-surface-400 truncate"> </span></div> <div><button class="btn variant-soft-surface"> </button> <button class="btn variant-soft-primary"> </button></div></li>`); var root_27 = /* @__PURE__ */ template(`<div class="text-surface-700-200-token mt-4"><h3 class="text-sm"> </h3> <ul class="relative list px-4 *:py-2 mt-2 bg-white/30 dark:bg-black/15 rounded-container-token *:!rounded-none max-h-80 scrollbar-track-transparent scrollbar-thumb-slate-400/50 scrollbar-corner-transparent scrollbar-thin overflow-y-auto divide-y-[1px] *:border-surface-300-600-token" style="scrollbar-gutter: stable"></ul></div>`); var root_31 = /* @__PURE__ */ template(`<span class="badge variant-soft-error"> </span>`); var root_34 = /* @__PURE__ */ template(`<li><span class="flex-[0_0_20%] truncate"> </span> <span class="text-error-400"> </span></li>`); var root_33 = /* @__PURE__ */ template(`<ul class="list max-h-80 scrollbar-track-transparent scrollbar-thumb-slate-400/50 scrollbar-corner-transparent scrollbar-thin overflow-y-auto" style="scrollbar-gutter: stable"></ul>`); var root_1$7 = /* @__PURE__ */ template(`<div data-theme="skeleton" class="card px-4 pb-4 fixed right-20 top-36 w-[600px] *:text-sm shadow-xl bg-scroll"><!> <div class="flex flex-col relative mt-4 gap-4"><!></div> <!> <!></div>`); var on_click_3 = (__3, showMenu) => { set(showMenu, !get$1(showMenu)); }; var root_36 = /* @__PURE__ */ template(`<img alt="batch download" class="object-cover object-center size-full">`); var root_37 = /* @__PURE__ */ template(`<div class="!absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2"><!></div>`); var root_38 = /* @__PURE__ */ template(`<div class="flex flex-col justify-center items-center px-3 font-bold text-[12px] leading-[14px] overflow-hidden text-surface-700-200-token"><span class=" truncate max-w-full"> </span> <hr class="!border-t-1 my-[1px] self-stretch !border-surface-700 dark:!border-surface-200"> <span class=" truncate max-w-full"> </span></div>`); var root_40 = /* @__PURE__ */ template(`<i class="w-6 fill-slate-700 dark:fill-slate-200 mix-blend-hard-light mt-4"><!></i>`); var root_35 = /* @__PURE__ */ template(`<div class="size-14 rounded-full fixed right-4 top-36 drop-shadow-xl cursor-pointer hover:brightness-110 backdrop-blur-sm"><div data-theme="skeleton" class="avatar absolute -z-10 size-14 rounded-full overflow-hidden bg-scroll transition-opacity duration-[250ms]"><!></div> <!> <div class="size-14 flex justify-center items-center relative"><!></div></div>`); var root$7 = /* @__PURE__ */ template(`<!> <!>`, 1); function Downloader$1($$anchor, $$props) { push($$props, true); const binding_group_1 = []; const { artworkCount, successd, failed, excluded, downloading, log, hasTask, batchDownload, abort, getConcurrency, setConcurrency } = $$props.useBatchDownload(); const reactiveUrl = new ReactiveValue(() => location.href, (update2) => { if ("navigation" in window) { let prevUrl; let nextUrl; navigation.addEventListener("navigatesuccess", () => { prevUrl !== nextUrl && update2(); }); navigation.addEventListener("navigate", (evt) => { prevUrl = location.href; nextUrl = evt.destination.url; }); } else { const rewrite = (type) => { const oriHistory = history[type]; return function(...args) { const currentUrl = location.href; const res = oriHistory.apply(this, args); const navigateUrl = location.href; currentUrl !== navigateUrl && update2(); return res; }; }; history.pushState = rewrite("pushState"); history.replaceState = rewrite("replaceState"); } }); initFilterStore(); let startDownloadEl; let stopDownloadEl; let avatarProgressEl; let avatarDownloadIcon; let fileHandleNotFoundTasks = state(proxy([])); const avatarCache = {}; let avatarSrc = state(""); let menuTabSet = state(0); let showMenu = state(false); const batchDownloadEntries = /* @__PURE__ */ derived(() => { if (!$$props.downloaderConfig) return null; const url = reactiveUrl.current; logger.info("Navigating to ", url); const { pageOption } = $$props.downloaderConfig; const generatorOptionEntries = []; for (const key in pageOption) { const item = pageOption[key]; const { match: matchPattern } = item; if (typeof matchPattern === "string") { url.match(matchPattern) && generatorOptionEntries.push([key, item]); } else if (typeof matchPattern === "function") { matchPattern(url) && generatorOptionEntries.push([key, item]); } else { matchPattern.test(url) && generatorOptionEntries.push([key, item]); } } if (generatorOptionEntries.length) { return generatorOptionEntries; } return null; }); const showDownloader = /* @__PURE__ */ derived(() => downloading.current || !!get$1(batchDownloadEntries)); const processed = /* @__PURE__ */ derived(() => successd.current.length + failed.current.length + excluded.current.length); const downloadProgress = /* @__PURE__ */ derived(() => artworkCount.current ? get$1(processed) / artworkCount.current * 100 : undefined); const downloadResult = /* @__PURE__ */ derived(() => !downloading.current && artworkCount.current ? `Completed: ${successd.current.length}. Failed: ${failed.current.length}. Excluded: ${excluded.current.length}.` : ""); user_effect(() => { if (!$$props.downloaderConfig.avatar) return; if (typeof $$props.downloaderConfig.avatar === "string") { set(avatarSrc, proxy($$props.downloaderConfig.avatar)); return; } const url = reactiveUrl.current; if (downloading.current) { return; } if (url in avatarCache) { const imageSrc = avatarCache[url]; if (get$1(avatarSrc) !== imageSrc) { set(avatarSrc, proxy(imageSrc)); } return; } const urlOrPromise = $$props.downloaderConfig.avatar(url); if (typeof urlOrPromise === "string") { avatarCache[url] = urlOrPromise; if (get$1(avatarSrc) !== urlOrPromise) { set(avatarSrc, proxy(urlOrPromise)); } return; } urlOrPromise.then((imageSrc) => { avatarCache[url] = imageSrc; if (get$1(avatarSrc) !== imageSrc) { set(avatarSrc, proxy(imageSrc)); } }); }); user_effect(() => { if (!get$1(showDownloader)) set(showMenu, false); }); user_effect(() => { if (downloading.current) { globalThis.addEventListener("beforeunload", beforeUnloadHandler); } else { globalThis.removeEventListener("beforeunload", beforeUnloadHandler); } }); user_effect(() => { if (downloading.current) { globalThis.addEventListener(EVENT_FILE_HANDLE_NOT_FOUND, fileHandleNotFoundHandler); } else { globalThis.removeEventListener(EVENT_FILE_HANDLE_NOT_FOUND, fileHandleNotFoundHandler); untrack(() => { if (get$1(fileHandleNotFoundTasks).length !== 0) { get$1(fileHandleNotFoundTasks).forEach((detail) => detail.abort()); set(fileHandleNotFoundTasks, proxy([])); setConcurrency(); } }); } }); function initFilterStore() { if (batchDownloaderStore.selectedFilters === null) { batchDownloaderStore.selectedFilters = $$props.downloaderConfig.filterOption.filters.filter((filter) => filter.checked).map((filter) => filter.id); } } function beforeUnloadHandler(evt) { evt.preventDefault(); evt.returnValue = true; } function fileHandleNotFoundHandler(evt) { if (!downloading.current) return; const customEvent = evt; const { signal } = customEvent.detail; if (!signal || !hasTask(signal)) return; get$1(fileHandleNotFoundTasks).push(customEvent.detail); setConcurrency(getConcurrency() + 1); } function startDownload(id) { return batchDownload(id).catch(logger.error); } var fragment = root$7(); var node = first_child(fragment); { var consequent_12 = ($$anchor2) => { var div = root_1$7(); var node_1 = child(div); { var consequent_5 = ($$anchor3) => { var div_1 = root_2$2(); var node_2 = child(div_1); TabGroup(node_2, { regionList: "text-surface-700-200-token", class: "text-sm", children: ($$anchor4, $$slotProps) => { var fragment_1 = root_3$4(); var node_3 = first_child(fragment_1); Tab(node_3, { name: "category", value: 0, get group() { return get$1(menuTabSet); }, set group($$value) { set(menuTabSet, proxy($$value)); }, children: ($$anchor5, $$slotProps2) => { next(); var text$1 = text(); template_effect(() => set_text(text$1, t("downloader.category.tab_name"))); append($$anchor5, text$1); }, $$slots: { default: true } }); var node_4 = sibling(node_3, 2); { var consequent = ($$anchor5) => { Tab($$anchor5, { name: "tag_filter", value: 1, get group() { return get$1(menuTabSet); }, set group($$value) { set(menuTabSet, proxy($$value)); }, children: ($$anchor6, $$slotProps2) => { next(); var text_1 = text(); template_effect(() => set_text(text_1, t("downloader.tag_filter.tab_name"))); append($$anchor6, text_1); }, $$slots: { default: true } }); }; if_block(node_4, ($$render) => { if (!!$$props.downloaderConfig.filterOption.enableTagFilter) $$render(consequent); }); } var node_5 = sibling(node_4, 2); Tab(node_5, { name: "tag_filter", value: 2, get group() { return get$1(menuTabSet); }, set group($$value) { set(menuTabSet, proxy($$value)); }, children: ($$anchor5, $$slotProps2) => { next(); var text_2 = text(); template_effect(() => set_text(text_2, t("downloader.others.tab_name"))); append($$anchor5, text_2); }, $$slots: { default: true } }); append($$anchor4, fragment_1); }, $$slots: { default: true, panel: ($$anchor4, $$slotProps) => { var fragment_6 = comment(); var node_6 = first_child(fragment_6); { var consequent_2 = ($$anchor5) => { var fragment_7 = root_10$1(); var node_7 = first_child(fragment_7); { var consequent_1 = ($$anchor6) => { var div_2 = root_11$2(); var div_3 = child(div_2); each(div_3, 21, () => $$props.downloaderConfig.filterOption.filters, index, ($$anchor7, $$item) => { let id = () => get$1($$item).id; let name = () => get$1($$item).name; var label = root_12$1(); var div_4 = child(label); var input = child(div_4); remove_input_defaults(input); var input_value; reset(div_4); var div_5 = sibling(div_4, 2); var text_3 = child(div_5, true); template_effect(() => set_text(text_3, typeof name() === "function" ? name()() : name())); reset(div_5); reset(label); template_effect(() => { if (input_value !== (input_value = id())) { input.value = null == (input.__value = id()) ? "" : id(); } }); bind_group( binding_group_1, [], input, () => { id(); return batchDownloaderStore.selectedFilters; }, ($$value) => batchDownloaderStore.selectedFilters = $$value ); append($$anchor7, label); }); reset(div_3); reset(div_2); append($$anchor6, div_2); }; if_block(node_7, ($$render) => { if ($$props.downloaderConfig.filterOption.filters.length) $$render(consequent_1); }); } var div_6 = sibling(node_7, 2); var div_7 = child(div_6); var node_8 = child(div_7); RadioGroup(node_8, { regionLabel: "text-surface-700-200-token", active: "variant-filled-primary", background: "bg-surface-400/20 dark:bg-surface-500/20", border: "", display: "flex", children: ($$anchor6, $$slotProps2) => { var fragment_8 = root_13$1(); var node_9 = first_child(fragment_8); RadioItem(node_9, { class: "text-sm !py-[7px]", name: "justify", value: true, get group() { return batchDownloaderStore.downloadAllPages; }, set group($$value) { batchDownloaderStore.downloadAllPages = $$value; }, children: ($$anchor7, $$slotProps3) => { next(); var text_4 = text(); template_effect(() => set_text(text_4, t("downloader.category.filter.download_all_pages"))); append($$anchor7, text_4); }, $$slots: { default: true } }); var node_10 = sibling(node_9, 2); RadioItem(node_10, { class: "text-sm !py-[7px]", name: "justify", value: false, get group() { return batchDownloaderStore.downloadAllPages; }, set group($$value) { batchDownloaderStore.downloadAllPages = $$value; }, children: ($$anchor7, $$slotProps3) => { next(); var text_5 = text(); template_effect(() => set_text(text_5, t("downloader.category.filter.download_selected_pages"))); append($$anchor7, text_5); }, $$slots: { default: true } }); append($$anchor6, fragment_8); }, $$slots: { default: true } }); reset(div_7); var div_8 = sibling(div_7, 2); var label_1 = child(div_8); var div_9 = child(label_1); var i_1 = child(div_9); var node_11 = child(i_1); html(node_11, () => playSvg); reset(i_1); reset(div_9); var input_1 = sibling(div_9, 2); action(input_1, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return batchDownloaderStore.pageStart; }, set(v) { batchDownloaderStore.pageStart = v; } })); reset(label_1); var label_2 = sibling(label_1, 2); var div_10 = child(label_2); var i_2 = child(div_10); var node_12 = child(i_2); html(node_12, () => stopOutLineSvg); reset(i_2); reset(div_10); var input_2 = sibling(div_10, 2); action(input_2, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return batchDownloaderStore.pageEnd; }, set(v) { batchDownloaderStore.pageEnd = v; } })); reset(label_2); reset(div_8); reset(div_6); template_effect(() => { input_1.disabled = batchDownloaderStore.downloadAllPages; input_2.disabled = batchDownloaderStore.downloadAllPages; }); append($$anchor5, fragment_7); }; var alternate_1 = ($$anchor5) => { var fragment_11 = comment(); var node_13 = first_child(fragment_11); { var consequent_3 = ($$anchor6) => { var fragment_12 = root_17$1(); var node_14 = first_child(fragment_12); var placeholder = /* @__PURE__ */ derived(() => t("downloader.tag_filter.placeholder.blacklist_tag")); InputChip(node_14, { allowUpperCase: true, name: "blacklist", chips: "variant-filled-primary", get placeholder() { return get$1(placeholder); }, get value() { return batchDownloaderStore.blacklistTag; }, set value($$value) { batchDownloaderStore.blacklistTag = $$value; } }); var node_15 = sibling(node_14, 2); var placeholder_1 = /* @__PURE__ */ derived(() => t("downloader.tag_filter.placeholder.whitelist_tag")); InputChip(node_15, { allowUpperCase: true, name: "whitelist", chips: "variant-filled-primary", get placeholder() { return get$1(placeholder_1); }, class: "my-4", get value() { return batchDownloaderStore.whitelistTag; }, set value($$value) { batchDownloaderStore.whitelistTag = $$value; } }); append($$anchor6, fragment_12); }; var alternate = ($$anchor6) => { var fragment_13 = comment(); var node_16 = first_child(fragment_13); { var consequent_4 = ($$anchor7) => { var div_11 = root_19$1(); var p = child(div_11); var text_6 = child(p, true); template_effect(() => set_text(text_6, t("downloader.others.options.retry_failed"))); reset(p); var node_17 = sibling(p, 2); SlideToggle(node_17, { size: "sm", name: "download-retry", get checked() { return batchDownloaderStore.retryFailed; }, set checked($$value) { batchDownloaderStore.retryFailed = $$value; } }); reset(div_11); append($$anchor7, div_11); }; if_block( node_16, ($$render) => { if (get$1(menuTabSet) === 2) $$render(consequent_4); }, true ); } append($$anchor6, fragment_13); }; if_block( node_13, ($$render) => { if (get$1(menuTabSet) === 1) $$render(consequent_3); else $$render(alternate, false); }, true ); } append($$anchor5, fragment_11); }; if_block(node_6, ($$render) => { if (get$1(menuTabSet) === 0) $$render(consequent_2); else $$render(alternate_1, false); }); } append($$anchor4, fragment_6); } } }); next(2); reset(div_1); transition(3, div_1, () => slide); append($$anchor3, div_1); }; if_block(node_1, ($$render) => { if (!downloading.current) $$render(consequent_5); }); } var div_12 = sibling(node_1, 2); var node_18 = child(div_12); { var consequent_9 = ($$anchor3) => { var div_13 = root_20(); var div_14 = child(div_13); var p_1 = child(div_14); var text_7 = child(p_1, true); reset(p_1); var p_2 = sibling(p_1, 2); var text_8 = child(p_2, true); reset(p_2); reset(div_14); var node_19 = sibling(div_14, 2); { var consequent_7 = ($$anchor4) => { var div_15 = root_21(); each(div_15, 21, () => get$1(batchDownloadEntries), index, ($$anchor5, $$item) => { let id = () => get$1($$item)[0]; let item = () => get$1($$item)[1]; var fragment_14 = comment(); var node_20 = first_child(fragment_14); { var consequent_6 = ($$anchor6) => { var button2 = root_23(); button2.__click = [on_click$1, startDownload, id]; var i_3 = child(button2); var node_21 = child(i_3); html(node_21, () => downloadSvg); reset(i_3); var span = sibling(i_3, 2); var text_9 = child(span, true); template_effect(() => set_text(text_9, typeof item().name === "function" ? item().name() : item().name)); reset(span); reset(button2); append($$anchor6, button2); }; if_block(node_20, ($$render) => { if ("fn" in item()) $$render(consequent_6); }); } append($$anchor5, fragment_14); }); reset(div_15); append($$anchor4, div_15); }; var alternate_2 = ($$anchor4) => { var fragment_15 = comment(); var node_22 = first_child(fragment_15); { var consequent_8 = ($$anchor5) => { var button_1 = root_25(); button_1.__click = [ on_click_1, startDownload, batchDownloadEntries ]; var i_4 = child(button_1); var node_23 = child(i_4); html(node_23, () => downloadSvg); reset(i_4); var span_1 = sibling(i_4, 2); var text_10 = child(span_1, true); template_effect(() => set_text(text_10, typeof get$1(batchDownloadEntries)[0][1].name === "function" ? get$1(batchDownloadEntries)[0][1].name() : get$1(batchDownloadEntries)[0][1].name)); reset(span_1); reset(button_1); append($$anchor5, button_1); }; if_block( node_22, ($$render) => { if (get$1(batchDownloadEntries) && "fn" in get$1(batchDownloadEntries)[0][1]) $$render(consequent_8); }, true ); } append($$anchor4, fragment_15); }; if_block(node_19, ($$render) => { if (get$1(batchDownloadEntries) && get$1(batchDownloadEntries).length > 1) $$render(consequent_7); else $$render(alternate_2, false); }); } reset(div_13); bind_this(div_13, ($$value) => startDownloadEl = $$value, () => startDownloadEl); template_effect(() => { var _a2, _b2; set_text(text_7, get$1(downloadResult)); toggle_class(p_2, "text-error-500", ((_a2 = log.current) == null ? undefined : _a2.type) === "Error"); set_text(text_8, ((_b2 = log.current) == null ? undefined : _b2.message) ?? ""); }); event("introstart", div_13, () => ( // required when the transition reverses startDownloadEl.classList.remove("absolute") )); event("outrostart", div_13, () => startDownloadEl.classList.add("absolute")); transition(3, div_13, () => fade, () => ({ duration: 250 })); append($$anchor3, div_13); }; var alternate_3 = ($$anchor3) => { var div_16 = root_26(); var div_17 = child(div_16); var node_24 = child(div_17); ProgressBar(node_24, { height: "h-4", rounded: "rounded-md", meter: "bg-primary-500", track: "bg-primary-500/30", max: 100, get value() { return get$1(downloadProgress); } }); var div_18 = sibling(node_24, 2); var p_3 = child(div_18); var text_11 = child(p_3, true); reset(p_3); var p_4 = sibling(p_3, 2); var text_12 = child(p_4, true); reset(p_4); reset(div_18); reset(div_17); var button_2 = sibling(div_17, 2); button_2.__click = [on_click_2, abort]; var i_5 = child(button_2); var node_25 = child(i_5); html(node_25, () => stopSvg); reset(i_5); var span_2 = sibling(i_5, 2); var text_13 = child(span_2, true); template_effect(() => set_text(text_13, t("downloader.download_type.stop"))); reset(span_2); reset(button_2); reset(div_16); bind_this(div_16, ($$value) => stopDownloadEl = $$value, () => stopDownloadEl); template_effect(() => { var _a2, _b2; toggle_class(p_3, "text-error-500", ((_a2 = log.current) == null ? undefined : _a2.type) === "Error"); set_text(text_11, ((_b2 = log.current) == null ? undefined : _b2.message) ?? ""); set_text(text_12, artworkCount.current ? `${get$1(processed)} / ${artworkCount.current}` : ""); }); event("introstart", div_16, () => stopDownloadEl.classList.remove("absolute")); event("outrostart", div_16, () => stopDownloadEl.classList.add("absolute")); transition(3, div_16, () => fade, () => ({ duration: 250 })); append($$anchor3, div_16); }; if_block(node_18, ($$render) => { if (!downloading.current) $$render(consequent_9); else $$render(alternate_3, false); }); } reset(div_12); var node_26 = sibling(div_12, 2); { var consequent_10 = ($$anchor3) => { var div_19 = root_27(); var h3 = child(div_19); var text_14 = child(h3, true); template_effect(() => set_text(text_14, t("downloader.file_handle_not_found.title"))); reset(h3); var ul = sibling(h3, 2); each(ul, 22, () => get$1(fileHandleNotFoundTasks), (task) => task, ($$anchor4, task, i) => { var li = root_28(); const filenameIndex = /* @__PURE__ */ derived(() => task.path.lastIndexOf("/")); const hasSubpath = /* @__PURE__ */ derived(() => get$1(filenameIndex) !== -1); var div_20 = child(li); var span_3 = child(div_20); var text_15 = child(span_3, true); template_effect(() => set_text(text_15, get$1(hasSubpath) ? task.path.slice(get$1(filenameIndex) + 1) : task.path)); reset(span_3); var span_4 = sibling(span_3, 2); var text_16 = child(span_4, true); template_effect(() => set_text(text_16, get$1(hasSubpath) ? task.path.slice(0, get$1(filenameIndex)) : "")); reset(span_4); reset(div_20); var div_21 = sibling(div_20, 2); var button_3 = child(div_21); button_3.__click = () => { task.abort(); get$1(fileHandleNotFoundTasks).splice(get$1(i), 1); setConcurrency(getConcurrency() - 1); }; var text_17 = child(button_3, true); template_effect(() => set_text(text_17, t("downloader.file_handle_not_found.button.cancel"))); reset(button_3); var button_4 = sibling(button_3, 2); button_4.__click = async () => { await task.getFileHandle(); get$1(fileHandleNotFoundTasks).splice(get$1(i), 1); setConcurrency(getConcurrency() - 1); }; var text_18 = child(button_4, true); template_effect(() => set_text(text_18, t("downloader.file_handle_not_found.button.save"))); reset(button_4); reset(div_21); reset(li); transition(3, li, () => slide, () => ({ duration: 250 })); append($$anchor4, li); }); reset(ul); reset(div_19); transition(3, div_19, () => slide, () => ({ duration: 250 })); append($$anchor3, div_19); }; if_block(node_26, ($$render) => { if (get$1(fileHandleNotFoundTasks).length) $$render(consequent_10); }); } var node_27 = sibling(node_26, 2); { var consequent_11 = ($$anchor3) => { const invalidIdFirst = /* @__PURE__ */ derived(() => failed.current.toSorted((item) => item.reason instanceof InvalidPostError ? -1 : 1)); Accordion($$anchor3, { class: "text-surface-700-200-token mt-4", children: ($$anchor4, $$slotProps) => { { const lead = ($$anchor5) => { var span_5 = root_31(); var text_19 = child(span_5, true); reset(span_5); template_effect(() => set_text(text_19, get$1(invalidIdFirst).length)); append($$anchor5, span_5); }; const summary = ($$anchor5) => { next(); var text_20 = text(); template_effect(() => set_text(text_20, t("downloader.failed_downloads"))); append($$anchor5, text_20); }; const content = ($$anchor5) => { var ul_1 = root_33(); each(ul_1, 20, () => get$1(invalidIdFirst), (item) => item, ($$anchor6, item) => { var li_1 = root_34(); var span_6 = child(li_1); var text_21 = child(span_6, true); reset(span_6); var span_7 = sibling(span_6, 2); var text_22 = child(span_7, true); reset(span_7); reset(li_1); template_effect(() => { set_text(text_21, item.id); set_text(text_22, item.reason); }); append($$anchor6, li_1); }); reset(ul_1); append($$anchor5, ul_1); }; AccordionItem($$anchor4, { class: "bg-white/30 dark:bg-black/15 rounded-container-token", lead, summary, content, $$slots: { lead: true, summary: true, content: true } }); } }, $$slots: { default: true } }); }; if_block(node_27, ($$render) => { if (!downloading.current && failed.current.length) $$render(consequent_11); }); } reset(div); transition(3, div, () => fly, () => ({ x: 50, opacity: 0 })); append($$anchor2, div); }; if_block(node, ($$render) => { if (get$1(showMenu) && get$1(showDownloader)) $$render(consequent_12); }); } var node_28 = sibling(node, 2); { var consequent_17 = ($$anchor2) => { var div_22 = root_35(); div_22.__click = [on_click_3, showMenu]; var div_23 = child(div_22); var node_29 = child(div_23); { var consequent_13 = ($$anchor3) => { var img = root_36(); template_effect(() => set_attribute(img, "src", get$1(avatarSrc))); append($$anchor3, img); }; if_block(node_29, ($$render) => { if (get$1(avatarSrc)) $$render(consequent_13); }); } reset(div_23); var node_30 = sibling(div_23, 2); { var consequent_14 = ($$anchor3) => { var div_24 = root_37(); var node_31 = child(div_24); ProgressRadial(node_31, { width: "w-16", stroke: 32, meter: "stroke-primary-500", track: "stroke-primary-500/30", fill: "fill-primary-500", strokeLinecap: "butt", get value() { return get$1(downloadProgress); } }); reset(div_24); transition(3, div_24, () => fade, () => ({ duration: 250 })); append($$anchor3, div_24); }; if_block(node_30, ($$render) => { if (downloading.current && !get$1(showMenu)) $$render(consequent_14); }); } var div_25 = sibling(node_30, 2); var node_32 = child(div_25); { var consequent_15 = ($$anchor3) => { var div_26 = root_38(); var span_8 = child(div_26); var text_23 = child(span_8, true); reset(span_8); var span_9 = sibling(span_8, 4); var text_24 = child(span_9, true); reset(span_9); reset(div_26); bind_this(div_26, ($$value) => avatarProgressEl = $$value, () => avatarProgressEl); template_effect(() => { set_text(text_23, get$1(processed)); set_text(text_24, artworkCount.current); }); event("introstart", div_26, () => avatarProgressEl.classList.remove("absolute")); event("outrostart", div_26, () => avatarProgressEl.classList.add("absolute")); transition(3, div_26, () => fade, () => ({ duration: 250 })); append($$anchor3, div_26); }; var alternate_4 = ($$anchor3) => { var fragment_19 = comment(); var node_33 = first_child(fragment_19); { var consequent_16 = ($$anchor4) => { var i_6 = root_40(); var node_34 = child(i_6); html(node_34, () => downloadMultipleSvg); reset(i_6); bind_this(i_6, ($$value) => avatarDownloadIcon = $$value, () => avatarDownloadIcon); event("introstart", i_6, () => avatarDownloadIcon.classList.remove("absolute")); event("outrostart", i_6, () => avatarDownloadIcon.classList.add("absolute")); transition(3, i_6, () => fade, () => ({ duration: 250 })); append($$anchor4, i_6); }; if_block( node_33, ($$render) => { if (!get$1(showMenu)) $$render(consequent_16); }, true ); } append($$anchor3, fragment_19); }; if_block(node_32, ($$render) => { if (downloading.current && artworkCount.current && !get$1(showMenu)) $$render(consequent_15); else $$render(alternate_4, false); }); } reset(div_25); reset(div_22); template_effect(() => { toggle_class(div_23, "opacity-70", !get$1(showMenu)); toggle_class(div_23, "blur-[1px]", !get$1(showMenu)); }); transition(3, div_22, () => fly, () => ({ opacity: 0, x: 50 })); append($$anchor2, div_22); }; if_block(node_28, ($$render) => { if (get$1(showDownloader)) $$render(consequent_17); }); } append($$anchor, fragment); pop(); } delegate(["click"]); const cog = `<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>`; 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; }, // firefox使用tampermonkey,downloadmod为默认时也支持子目录,但GM_info显示为原生的“native”,而“原生”不支持子目录 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"; }, videoFrameSupported() { return typeof _unsafeWindow.VideoFrame === "function"; }, isPixiv() { return location.hostname === "www.pixiv.net"; }, isYande() { return location.hostname === "yande.re"; }, isRule34() { return location.hostname === "rule34.xxx"; }, isNijie() { return location.hostname === "nijie.info"; } }; const check = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><title>check</title><path d="M21,7L9,19L3.5,13.5L4.91,12.09L9,16.17L19.59,5.59L21,7Z" /></svg>`; const folderSvg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M10,4H4C2.89,4 2,4.89 2,6V18A2,2 0 0,0 4,20H20A2,2 0 0,0 22,18V8C22,6.89 21.1,6 20,6H12L10,4Z" /></svg>`; const fileSvg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M13,9V3.5L18.5,9M6,2C4.89,2 4,2.89 4,4V20A2,2 0 0,0 6,22H18A2,2 0 0,0 20,20V8L14,2H6Z" /></svg>`; 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: new RegExp("(?<=users\\/)([0-9]+)\\/(artworks|illustrations|manga|bookmarks(?=\\/artworks))(?:\\/artworks)?\\/?([^?]*)"), searchPage: /\/tags\/.*\/(artworks|illustrations|manga)/, activityHref: /illust_id=(\d+)/, originSrcPageNum: new RegExp("(?<=_p)\\d+"), followLatest: /\/bookmark_new_illust(?:_r18)?\.php/, historyPage: /\/history\.php/, historyThumbnailsId: /\d+(?=_)/, series: /\/user\/([0-9]+)\/series\/([0-9]+)/, unlisted: new RegExp("(?<=artworks\\/unlisted\\/)[A-Za-z0-9]+"), imageExt: /bmp|jp(e)?g|jfif|png|tif(f)?|gif|svg|ico|webp|heif|heic|raw|cr2|nef|arw|dng|avif/i, videoExt: /mp4|avi|mkv|mov|wmv|flv|webm|mpeg|mpg|3gp|m4v|ts|vob|ogv|rm|rmvb|m2ts|mxf|asf|swf/i }; var FilenameConflictAction = /* @__PURE__ */ ((FilenameConflictAction2) => { FilenameConflictAction2["UNIQUIFY"] = "uniquify"; FilenameConflictAction2["OVERWRITE"] = "overwrite"; FilenameConflictAction2["PROMPT"] = "prompt"; return FilenameConflictAction2; })(FilenameConflictAction || {}); let dirHandle = null; let dirHandleStatus = "dirHandle.unpick"; const occupiedFilename = /* @__PURE__ */ new Set(); const pendingList = []; channelEvent.on("dirHandle.picking", () => { logger.info("channelEvnet:DirHandleStatus.PICKING"); dirHandleStatus = "dirHandle.picking"; }); channelEvent.on("dirHandle.unpick", async () => { logger.info("channelEvnet:DirHandleStatus.UNPICK"); dirHandleStatus = "dirHandle.unpick"; dirHandle = null; rejectPendingList(new Error("Directory handle not found.")); }); channelEvent.on( "dirHandle.picked", async (handler) => { logger.info("channelEvnet:DirHandleStatus.PICKED"); dirHandleStatus = "dirHandle.picked"; dirHandle = handler; if (await dirHandle.queryPermission({ mode: "readwrite" }) === "granted") { resolvePendingList(dirHandle); } else { rejectPendingList(new PermissionError()); } } ); async function getPersistedDirHanle() { return await historyDb.getDirectoryHandle() ?? null; } function setPersistedDirHanle(dirHandle2) { historyDb.setDirectoryHandle(dirHandle2); } function resolvePendingList(dirHandle2) { if (!pendingList.length) return; for (const [resolve] of pendingList) { resolve(dirHandle2); } pendingList.length = 0; } function rejectPendingList(reason) { if (!pendingList.length) return; for (const [, reject] of pendingList) { reject(reason); } pendingList.length = 0; } function setDirHandleStatus(status, dirHandle2) { if (status === "dirHandle.picking") { dirHandleStatus = "dirHandle.picking"; channelEvent.emit( "dirHandle.picking" /* PICKING */ ); return; } if (status === "dirHandle.unpick") { dirHandleStatus = "dirHandle.unpick"; channelEvent.emit( "dirHandle.unpick" /* UNPICK */ ); return; } if (!dirHandle2) throw new Error("directory handle should not be null."); dirHandleStatus = "dirHandle.picked"; channelEvent.emit( "dirHandle.picked", dirHandle2 ); } async function verifyPermission(fileHandle) { const opts = { mode: "readwrite" }; if (await fileHandle.queryPermission(opts) === "granted") { return true; } if (navigator.userActivation.isActive) { if (await fileHandle.requestPermission(opts) === "granted") { return true; } } else { if (dirHandleStatus !== "dirHandle.picking") { setDirHandleStatus( "dirHandle.picking" /* PICKING */ ); } const success = await new Promise((resolve) => { globalThis.dispatchEvent( new CustomEvent(EVENT_REQUEST_USER_ACTIVATION, { detail: { onAction() { resolve(true); }, onClosed() { resolve(false); } } }) ); }); if (!success) { setDirHandleStatus("dirHandle.picked", fileHandle); return false; } const permissionState = await fileHandle.requestPermission(opts); setDirHandleStatus("dirHandle.picked", fileHandle); if (success && permissionState === "granted") { return true; } } return false; } async function getDirHandleByUserAction(userAction) { try { if (dirHandleStatus !== "dirHandle.picking") { setDirHandleStatus( "dirHandle.picking" /* PICKING */ ); } dirHandle = await userAction(); setDirHandleStatus("dirHandle.picked", dirHandle); resolvePendingList(dirHandle); setPersistedDirHanle(dirHandle); } catch (error) { if (dirHandle) { logger.warn(error); const hasPermission = await verifyPermission(dirHandle); setDirHandleStatus("dirHandle.picked", dirHandle); if (!hasPermission) { const error2 = new PermissionError(); rejectPendingList(error2); throw error2; } resolvePendingList(dirHandle); } else { setDirHandleStatus( "dirHandle.unpick" /* UNPICK */ ); rejectPendingList(error); throw error; } } return dirHandle; } async function getRootDirHandle(signal) { if (dirHandleStatus === "dirHandle.unpick") { setDirHandleStatus( "dirHandle.picking" /* PICKING */ ); if (dirHandle = await getPersistedDirHanle()) { const hasPermission = await verifyPermission(dirHandle); setDirHandleStatus("dirHandle.picked", dirHandle); if (!hasPermission) { const error = new PermissionError(); rejectPendingList(error); throw error; } resolvePendingList(dirHandle); return dirHandle; } return await getDirHandleByUserAction(() => { return new Promise((resolve, reject) => { globalThis.dispatchEvent( new CustomEvent(EVENT_DIR_HANDLE_NOT_FOUND, { detail: { getFileHandle: async () => { try { const dirHandle2 = await _unsafeWindow.showDirectoryPicker({ id: "pdl-dir-handle", mode: "readwrite" }); resolve(dirHandle2); } catch (error) { reject(error); } }, abort: () => reject(new CancelError()) } }) ); signal == null ? void 0 : signal.addEventListener( "abort", () => { reject(signal.reason); }, { once: true } ); pendingList.push([resolve, reject]); }); }); } if (dirHandleStatus === "dirHandle.picking") { const prom = new Promise((resolve, reject) => { pendingList.push([resolve, reject]); }); return await prom; } if (!dirHandle) throw new Error("DirHandle should not be null."); if (!await verifyPermission(dirHandle)) { const error = new PermissionError(); rejectPendingList(error); throw error; } return dirHandle; } async function getFileHandle(dirHandle2, filename, filenameConflictAction) { if (filenameConflictAction === "overwrite") { return dirHandle2.getFileHandle(filename, { create: true }); } if (!occupiedFilename.has(filename)) { occupiedFilename.add(filename); try { await dirHandle2.getFileHandle(filename); } catch (error) { return await dirHandle2.getFileHandle(filename, { create: true }); } finally { occupiedFilename.delete(filename); } } if (filenameConflictAction === "prompt") { throw new DOMException("", "NotFoundError"); } const extIndex = filename.lastIndexOf("."); const ext = filename.slice(extIndex + 1); const name = filename.slice(0, extIndex); for (let suffix = 1; suffix < 101; suffix++) { const newFilename2 = `${name} (${suffix}).${ext}`; if (occupiedFilename.has(newFilename2)) { continue; } occupiedFilename.add(newFilename2); try { await dirHandle2.getFileHandle(newFilename2); } catch (error) { const fileHandle = await dirHandle2.getFileHandle(newFilename2, { create: true }); logger.info(`Rename ${filename}' to ${newFilename2}`); return fileHandle; } finally { occupiedFilename.delete(newFilename2); } } const newFilename = `${name} - ${(/* @__PURE__ */ new Date()).toISOString().replaceAll(":", "")}.${ext}`; return dirHandle2.getFileHandle(newFilename, { create: true }); } async function saveFile$1(filenameConflictAction, blob, path, signal) { const rootDirHandle = await getRootDirHandle(signal); signal == null ? undefined : signal.throwIfAborted(); let currentDirHandle = rootDirHandle; let fileHandle = null; let writableStream = null; const filenameIndex = path.lastIndexOf("/"); const filename = filenameIndex === -1 ? path : path.slice(filenameIndex + 1); try { if (filenameIndex !== -1) { const relativePaths = path.slice(0, filenameIndex).split("/"); for (const dir of relativePaths) { currentDirHandle = await currentDirHandle.getDirectoryHandle(dir, { create: true }); } } fileHandle = await getFileHandle(currentDirHandle, filename, filenameConflictAction); writableStream = await fileHandle.createWritable(); } catch (error) { if (!(error instanceof DOMException && error.name === "NotFoundError")) { throw error; } if (fileHandle !== null) { currentDirHandle.removeEntry(fileHandle.name); } fileHandle = await new Promise((resolve, reject) => { const extIndex = filename.lastIndexOf("."); const ext = filename.slice(extIndex + 1); const mimeType = regexp.imageExt.test(ext) ? `image/${ext}` : `video/${ext}`; globalThis.dispatchEvent( new CustomEvent(EVENT_FILE_HANDLE_NOT_FOUND, { detail: { signal, path, getFileHandle: async () => { try { const fileHandle2 = await _unsafeWindow.showSaveFilePicker({ startIn: currentDirHandle, suggestedName: filename, types: [ { description: "Media", accept: { [mimeType]: "." + ext } } ] }); resolve(fileHandle2); } catch (error2) { reject(error2); } }, abort: () => reject(new CancelError()) } }) ); signal == null ? undefined : signal.addEventListener( "abort", () => { reject(signal.reason); }, { once: true } ); }); } try { writableStream ?? (writableStream = await fileHandle.createWritable()); await writableStream.write(blob); await writableStream.close(); } catch (error) { currentDirHandle.removeEntry(fileHandle.name); throw error; } } function selectRootDirHandle() { return getDirHandleByUserAction(() => { return _unsafeWindow.showDirectoryPicker({ id: "pdl-dir-handle", mode: "readwrite" }); }); } function gmDownload(blob, path, signal) { return new Promise((resolve, reject) => { signal == null ? undefined : signal.throwIfAborted(); const imgUrl = URL.createObjectURL(blob); const abortObj = _GM_download({ url: URL.createObjectURL(blob), name: path, onerror: (error) => { URL.revokeObjectURL(imgUrl); if (signal == null ? undefined : signal.aborted) { resolve(); } else { logger.error(error); reject(new Error(`FileSave error: ${path}`)); } }, onload: () => { URL.revokeObjectURL(imgUrl); resolve(); } }); signal == null ? undefined : signal.addEventListener("abort", () => abortObj.abort(), { once: true }); }); } function aDownload(blob, 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); } function gmDownloadDataUrl(blob, path, signal) { return readBlobAsDataUrl(blob).then((dataUrl) => { signal == null ? undefined : signal.throwIfAborted(); return new Promise((resolve, reject) => { const abortObj = _GM_download({ url: dataUrl, name: path, onerror: (error) => { if (signal == null ? undefined : signal.aborted) { resolve(); } else { logger.error(error); reject(new Error(`FileSave error: ${path}`)); } }, onload: () => { resolve(); } }); signal == null ? undefined : signal.addEventListener("abort", () => abortObj.abort(), { once: true }); }); }); } let saveFile; const blobAvailable = env.isBlobDlAvaliable(); const subPathAvailable = env.isSupportSubpath(); if (subPathAvailable) { if (!blobAvailable) { saveFile = gmDownloadDataUrl; } else { saveFile = gmDownload; } } else { saveFile = aDownload; logger.warn("Download function is not fully supported:", _GM_info.scriptHandler, _GM_info.version); } const fileSaveAdapters = { isFileSystemAccessAvailable: env.isFileSystemAccessAvaliable(), getAdapter(useFileSystemAccessApi, filenameConflictAction = FilenameConflictAction.UNIQUIFY) { if (this.isFileSystemAccessAvailable && useFileSystemAccessApi) { return saveFile$1.bind(undefined, filenameConflictAction); } else { return saveFile; } } }; function getDefaultExportFromCjs(x) { return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, "default") ? x["default"] : x; } var eventemitter3 = { exports: {} }; (function(module) { var has = Object.prototype.hasOwnProperty, prefix = "~"; function Events() { } if (Object.create) { Events.prototype = /* @__PURE__ */ Object.create(null); if (!new Events().__proto__) prefix = false; } function EE(fn, context, once) { this.fn = fn; this.context = context; this.once = once || false; } function addListener(emitter, event2, fn, context, once) { if (typeof fn !== "function") { throw new TypeError("The listener must be a function"); } var listener = new EE(fn, context || emitter, once), evt = prefix ? prefix + event2 : event2; if (!emitter._events[evt]) emitter._events[evt] = listener, emitter._eventsCount++; else if (!emitter._events[evt].fn) emitter._events[evt].push(listener); else emitter._events[evt] = [emitter._events[evt], listener]; return emitter; } function clearEvent(emitter, evt) { if (--emitter._eventsCount === 0) emitter._events = new Events(); else delete emitter._events[evt]; } function EventEmitter2() { this._events = new Events(); this._eventsCount = 0; } EventEmitter2.prototype.eventNames = function eventNames() { var names = [], events, name; if (this._eventsCount === 0) return names; for (name in events = this._events) { if (has.call(events, name)) names.push(prefix ? name.slice(1) : name); } if (Object.getOwnPropertySymbols) { return names.concat(Object.getOwnPropertySymbols(events)); } return names; }; EventEmitter2.prototype.listeners = function listeners(event2) { var evt = prefix ? prefix + event2 : event2, handlers = this._events[evt]; if (!handlers) return []; if (handlers.fn) return [handlers.fn]; for (var i = 0, l = handlers.length, ee = new Array(l); i < l; i++) { ee[i] = handlers[i].fn; } return ee; }; EventEmitter2.prototype.listenerCount = function listenerCount(event2) { var evt = prefix ? prefix + event2 : event2, listeners = this._events[evt]; if (!listeners) return 0; if (listeners.fn) return 1; return listeners.length; }; EventEmitter2.prototype.emit = function emit(event2, a1, a2, a3, a4, a5) { var evt = prefix ? prefix + event2 : event2; if (!this._events[evt]) return false; var listeners = this._events[evt], len = arguments.length, args, i; if (listeners.fn) { if (listeners.once) this.removeListener(event2, listeners.fn, undefined, true); switch (len) { case 1: return listeners.fn.call(listeners.context), true; case 2: return listeners.fn.call(listeners.context, a1), true; case 3: return listeners.fn.call(listeners.context, a1, a2), true; case 4: return listeners.fn.call(listeners.context, a1, a2, a3), true; case 5: return listeners.fn.call(listeners.context, a1, a2, a3, a4), true; case 6: return listeners.fn.call(listeners.context, a1, a2, a3, a4, a5), true; } for (i = 1, args = new Array(len - 1); i < len; i++) { args[i - 1] = arguments[i]; } listeners.fn.apply(listeners.context, args); } else { var length = listeners.length, j; for (i = 0; i < length; i++) { if (listeners[i].once) this.removeListener(event2, listeners[i].fn, undefined, true); switch (len) { case 1: listeners[i].fn.call(listeners[i].context); break; case 2: listeners[i].fn.call(listeners[i].context, a1); break; case 3: listeners[i].fn.call(listeners[i].context, a1, a2); break; case 4: listeners[i].fn.call(listeners[i].context, a1, a2, a3); break; default: if (!args) for (j = 1, args = new Array(len - 1); j < len; j++) { args[j - 1] = arguments[j]; } listeners[i].fn.apply(listeners[i].context, args); } } } return true; }; EventEmitter2.prototype.on = function on2(event2, fn, context) { return addListener(this, event2, fn, context, false); }; EventEmitter2.prototype.once = function once(event2, fn, context) { return addListener(this, event2, fn, context, true); }; EventEmitter2.prototype.removeListener = function removeListener(event2, fn, context, once) { var evt = prefix ? prefix + event2 : event2; if (!this._events[evt]) return this; if (!fn) { clearEvent(this, evt); return this; } var listeners = this._events[evt]; if (listeners.fn) { if (listeners.fn === fn && (!once || listeners.once) && (!context || listeners.context === context)) { clearEvent(this, evt); } } else { for (var i = 0, events = [], length = listeners.length; i < length; i++) { if (listeners[i].fn !== fn || once && !listeners[i].once || context && listeners[i].context !== context) { events.push(listeners[i]); } } if (events.length) this._events[evt] = events.length === 1 ? events[0] : events; else clearEvent(this, evt); } return this; }; EventEmitter2.prototype.removeAllListeners = function removeAllListeners(event2) { var evt; if (event2) { evt = prefix ? prefix + event2 : event2; if (this._events[evt]) clearEvent(this, evt); } else { this._events = new Events(); this._eventsCount = 0; } return this; }; EventEmitter2.prototype.off = EventEmitter2.prototype.removeListener; EventEmitter2.prototype.addListener = EventEmitter2.prototype.on; EventEmitter2.prefixed = prefix; EventEmitter2.EventEmitter = EventEmitter2; { module.exports = EventEmitter2; } })(eventemitter3); var eventemitter3Exports = eventemitter3.exports; const EventEmitter = /* @__PURE__ */ getDefaultExportFromCjs(eventemitter3Exports); class TimeoutError extends Error { constructor(message2) { super(message2); this.name = "TimeoutError"; } } class AbortError extends Error { constructor(message2) { super(); this.name = "AbortError"; this.message = message2; } } const getDOMException = (errorMessage) => globalThis.DOMException === undefined ? new AbortError(errorMessage) : new DOMException(errorMessage); const getAbortedReason = (signal) => { const reason = signal.reason === undefined ? getDOMException("This operation was aborted.") : signal.reason; return reason instanceof Error ? reason : getDOMException(reason); }; function pTimeout(promise, options) { const { milliseconds, fallback, message: message2, customTimers = { setTimeout, clearTimeout } } = options; let timer; let abortHandler; const wrappedPromise = new Promise((resolve, reject) => { if (typeof milliseconds !== "number" || Math.sign(milliseconds) !== 1) { throw new TypeError(`Expected \`milliseconds\` to be a positive number, got \`${milliseconds}\``); } if (options.signal) { const { signal } = options; if (signal.aborted) { reject(getAbortedReason(signal)); } abortHandler = () => { reject(getAbortedReason(signal)); }; signal.addEventListener("abort", abortHandler, { once: true }); } if (milliseconds === Number.POSITIVE_INFINITY) { promise.then(resolve, reject); return; } const timeoutError = new TimeoutError(); timer = customTimers.setTimeout.call(undefined, () => { if (fallback) { try { resolve(fallback()); } catch (error) { reject(error); } return; } if (typeof promise.cancel === "function") { promise.cancel(); } if (message2 === false) { resolve(); } else if (message2 instanceof Error) { reject(message2); } else { timeoutError.message = message2 ?? `Promise timed out after ${milliseconds} milliseconds`; reject(timeoutError); } }, milliseconds); (async () => { try { resolve(await promise); } catch (error) { reject(error); } })(); }); const cancelablePromise = wrappedPromise.finally(() => { cancelablePromise.clear(); if (abortHandler && options.signal) { options.signal.removeEventListener("abort", abortHandler); } }); cancelablePromise.clear = () => { customTimers.clearTimeout.call(undefined, timer); timer = undefined; }; return cancelablePromise; } function lowerBound(array, value, comparator) { let first = 0; let count = array.length; while (count > 0) { const step = Math.trunc(count / 2); let it = first + step; if (comparator(array[it], value) <= 0) { first = ++it; count -= step + 1; } else { count = step; } } return first; } class PriorityQueue { constructor() { __privateAdd(this, _queue, []); } enqueue(run2, options) { options = { priority: 0, ...options }; const element = { priority: options.priority, run: run2 }; if (this.size && __privateGet(this, _queue)[this.size - 1].priority >= options.priority) { __privateGet(this, _queue).push(element); return; } const index2 = lowerBound(__privateGet(this, _queue), element, (a, b) => b.priority - a.priority); __privateGet(this, _queue).splice(index2, 0, element); } dequeue() { const item = __privateGet(this, _queue).shift(); return item == null ? undefined : item.run; } filter(options) { return __privateGet(this, _queue).filter((element) => element.priority === options.priority).map((element) => element.run); } get size() { return __privateGet(this, _queue).length; } } _queue = new WeakMap(); class PQueue extends EventEmitter { // TODO: The `throwOnTimeout` option should affect the return types of `add()` and `addAll()` constructor(options) { var _a2, _b2; super(); __privateAdd(this, _PQueue_instances); __privateAdd(this, _carryoverConcurrencyCount); __privateAdd(this, _isIntervalIgnored); __privateAdd(this, _intervalCount, 0); __privateAdd(this, _intervalCap); __privateAdd(this, _interval); __privateAdd(this, _intervalEnd, 0); __privateAdd(this, _intervalId); __privateAdd(this, _timeoutId); __privateAdd(this, _queue2); __privateAdd(this, _queueClass); __privateAdd(this, _pending, 0); // The `!` is needed because of https://github.com/microsoft/TypeScript/issues/32194 __privateAdd(this, _concurrency); __privateAdd(this, _isPaused); __privateAdd(this, _throwOnTimeout); /** Per-operation timeout in milliseconds. Operations fulfill once `timeout` elapses if they haven't already. Applies to each future operation. */ __publicField(this, "timeout"); options = { carryoverConcurrencyCount: false, intervalCap: Number.POSITIVE_INFINITY, interval: 0, concurrency: Number.POSITIVE_INFINITY, autoStart: true, queueClass: PriorityQueue, ...options }; if (!(typeof options.intervalCap === "number" && options.intervalCap >= 1)) { throw new TypeError(`Expected \`intervalCap\` to be a number from 1 and up, got \`${((_a2 = options.intervalCap) == null ? undefined : _a2.toString()) ?? ""}\` (${typeof options.intervalCap})`); } if (options.interval === undefined || !(Number.isFinite(options.interval) && options.interval >= 0)) { throw new TypeError(`Expected \`interval\` to be a finite number >= 0, got \`${((_b2 = options.interval) == null ? undefined : _b2.toString()) ?? ""}\` (${typeof options.interval})`); } __privateSet(this, _carryoverConcurrencyCount, options.carryoverConcurrencyCount); __privateSet(this, _isIntervalIgnored, options.intervalCap === Number.POSITIVE_INFINITY || options.interval === 0); __privateSet(this, _intervalCap, options.intervalCap); __privateSet(this, _interval, options.interval); __privateSet(this, _queue2, new options.queueClass()); __privateSet(this, _queueClass, options.queueClass); this.concurrency = options.concurrency; this.timeout = options.timeout; __privateSet(this, _throwOnTimeout, options.throwOnTimeout === true); __privateSet(this, _isPaused, options.autoStart === false); } get concurrency() { return __privateGet(this, _concurrency); } set concurrency(newConcurrency) { if (!(typeof newConcurrency === "number" && newConcurrency >= 1)) { throw new TypeError(`Expected \`concurrency\` to be a number from 1 and up, got \`${newConcurrency}\` (${typeof newConcurrency})`); } __privateSet(this, _concurrency, newConcurrency); __privateMethod(this, _PQueue_instances, processQueue_fn).call(this); } async add(function_, options = {}) { options = { timeout: this.timeout, throwOnTimeout: __privateGet(this, _throwOnTimeout), ...options }; return new Promise((resolve, reject) => { __privateGet(this, _queue2).enqueue(async () => { var _a2; __privateWrapper(this, _pending)._++; __privateWrapper(this, _intervalCount)._++; try { (_a2 = options.signal) == null ? void 0 : _a2.throwIfAborted(); let operation = function_({ signal: options.signal }); if (options.timeout) { operation = pTimeout(Promise.resolve(operation), { milliseconds: options.timeout }); } if (options.signal) { operation = Promise.race([operation, __privateMethod(this, _PQueue_instances, throwOnAbort_fn).call(this, options.signal)]); } const result = await operation; resolve(result); this.emit("completed", result); } catch (error) { if (error instanceof TimeoutError && !options.throwOnTimeout) { resolve(); return; } reject(error); this.emit("error", error); } finally { __privateMethod(this, _PQueue_instances, next_fn).call(this); } }, options); this.emit("add"); __privateMethod(this, _PQueue_instances, tryToStartAnother_fn).call(this); }); } async addAll(functions, options) { return Promise.all(functions.map(async (function_) => this.add(function_, options))); } /** Start (or resume) executing enqueued tasks within concurrency limit. No need to call this if queue is not paused (via `options.autoStart = false` or by `.pause()` method.) */ start() { if (!__privateGet(this, _isPaused)) { return this; } __privateSet(this, _isPaused, false); __privateMethod(this, _PQueue_instances, processQueue_fn).call(this); return this; } /** Put queue execution on hold. */ pause() { __privateSet(this, _isPaused, true); } /** Clear the queue. */ clear() { __privateSet(this, _queue2, new (__privateGet(this, _queueClass))()); } /** Can be called multiple times. Useful if you for example add additional items at a later time. @returns A promise that settles when the queue becomes empty. */ async onEmpty() { if (__privateGet(this, _queue2).size === 0) { return; } await __privateMethod(this, _PQueue_instances, onEvent_fn).call(this, "empty"); } /** @returns A promise that settles when the queue size is less than the given limit: `queue.size < limit`. If you want to avoid having the queue grow beyond a certain size you can `await queue.onSizeLessThan()` before adding a new item. Note that this only limits the number of items waiting to start. There could still be up to `concurrency` jobs already running that this call does not include in its calculation. */ async onSizeLessThan(limit) { if (__privateGet(this, _queue2).size < limit) { return; } await __privateMethod(this, _PQueue_instances, onEvent_fn).call(this, "next", () => __privateGet(this, _queue2).size < limit); } /** The difference with `.onEmpty` is that `.onIdle` guarantees that all work from the queue has finished. `.onEmpty` merely signals that the queue is empty, but it could mean that some promises haven't completed yet. @returns A promise that settles when the queue becomes empty, and all promises have completed; `queue.size === 0 && queue.pending === 0`. */ async onIdle() { if (__privateGet(this, _pending) === 0 && __privateGet(this, _queue2).size === 0) { return; } await __privateMethod(this, _PQueue_instances, onEvent_fn).call(this, "idle"); } /** Size of the queue, the number of queued items waiting to run. */ get size() { return __privateGet(this, _queue2).size; } /** Size of the queue, filtered by the given options. For example, this can be used to find the number of items remaining in the queue with a specific priority level. */ sizeBy(options) { return __privateGet(this, _queue2).filter(options).length; } /** Number of running items (no longer in the queue). */ get pending() { return __privateGet(this, _pending); } /** Whether the queue is currently paused. */ get isPaused() { return __privateGet(this, _isPaused); } } _carryoverConcurrencyCount = new WeakMap(); _isIntervalIgnored = new WeakMap(); _intervalCount = new WeakMap(); _intervalCap = new WeakMap(); _interval = new WeakMap(); _intervalEnd = new WeakMap(); _intervalId = new WeakMap(); _timeoutId = new WeakMap(); _queue2 = new WeakMap(); _queueClass = new WeakMap(); _pending = new WeakMap(); _concurrency = new WeakMap(); _isPaused = new WeakMap(); _throwOnTimeout = new WeakMap(); _PQueue_instances = new WeakSet(); doesIntervalAllowAnother_get = function() { return __privateGet(this, _isIntervalIgnored) || __privateGet(this, _intervalCount) < __privateGet(this, _intervalCap); }; doesConcurrentAllowAnother_get = function() { return __privateGet(this, _pending) < __privateGet(this, _concurrency); }; next_fn = function() { __privateWrapper(this, _pending)._--; __privateMethod(this, _PQueue_instances, tryToStartAnother_fn).call(this); this.emit("next"); }; onResumeInterval_fn = function() { __privateMethod(this, _PQueue_instances, onInterval_fn).call(this); __privateMethod(this, _PQueue_instances, initializeIntervalIfNeeded_fn).call(this); __privateSet(this, _timeoutId, undefined); }; isIntervalPaused_get = function() { const now2 = Date.now(); if (__privateGet(this, _intervalId) === undefined) { const delay = __privateGet(this, _intervalEnd) - now2; if (delay < 0) { __privateSet(this, _intervalCount, __privateGet(this, _carryoverConcurrencyCount) ? __privateGet(this, _pending) : 0); } else { if (__privateGet(this, _timeoutId) === undefined) { __privateSet(this, _timeoutId, setTimeout(() => { __privateMethod(this, _PQueue_instances, onResumeInterval_fn).call(this); }, delay)); } return true; } } return false; }; tryToStartAnother_fn = function() { if (__privateGet(this, _queue2).size === 0) { if (__privateGet(this, _intervalId)) { clearInterval(__privateGet(this, _intervalId)); } __privateSet(this, _intervalId, undefined); this.emit("empty"); if (__privateGet(this, _pending) === 0) { this.emit("idle"); } return false; } if (!__privateGet(this, _isPaused)) { const canInitializeInterval = !__privateGet(this, _PQueue_instances, isIntervalPaused_get); if (__privateGet(this, _PQueue_instances, doesIntervalAllowAnother_get) && __privateGet(this, _PQueue_instances, doesConcurrentAllowAnother_get)) { const job = __privateGet(this, _queue2).dequeue(); if (!job) { return false; } this.emit("active"); job(); if (canInitializeInterval) { __privateMethod(this, _PQueue_instances, initializeIntervalIfNeeded_fn).call(this); } return true; } } return false; }; initializeIntervalIfNeeded_fn = function() { if (__privateGet(this, _isIntervalIgnored) || __privateGet(this, _intervalId) !== undefined) { return; } __privateSet(this, _intervalId, setInterval(() => { __privateMethod(this, _PQueue_instances, onInterval_fn).call(this); }, __privateGet(this, _interval))); __privateSet(this, _intervalEnd, Date.now() + __privateGet(this, _interval)); }; onInterval_fn = function() { if (__privateGet(this, _intervalCount) === 0 && __privateGet(this, _pending) === 0 && __privateGet(this, _intervalId)) { clearInterval(__privateGet(this, _intervalId)); __privateSet(this, _intervalId, undefined); } __privateSet(this, _intervalCount, __privateGet(this, _carryoverConcurrencyCount) ? __privateGet(this, _pending) : 0); __privateMethod(this, _PQueue_instances, processQueue_fn).call(this); }; /** Executes all queued functions until it reaches the limit. */ processQueue_fn = function() { while (__privateMethod(this, _PQueue_instances, tryToStartAnother_fn).call(this)) { } }; throwOnAbort_fn = async function(signal) { return new Promise((_resolve, reject) => { signal.addEventListener("abort", () => { reject(signal.reason); }, { once: true }); }); }; onEvent_fn = async function(event2, filter) { return new Promise((resolve) => { const listener = () => { if (filter && !filter()) { return; } this.off(event2, listener); resolve(); }; this.on(event2, listener); }); }; class Downloader { constructor() { __privateAdd(this, _Downloader_instances); __privateAdd(this, _DOWNLOAD_RETRY, 3); __privateAdd(this, _downloadQueue, new PQueue({ concurrency: 5, interval: 1e3, intervalCap: 4 })); } async updateDirHandle() { return (await selectRootDirHandle()).name; } async download(configs, option = {}) { const { signal, priority } = option; signal == null ? undefined : signal.throwIfAborted(); logger.info("Downloader add:", configs); if (!Array.isArray(configs)) configs = [configs]; if (configs.length < 1) return; const downloads = configs.map((config) => { return __privateMethod(this, _Downloader_instances, dispatchDownload_fn).call(this, config, priority, signal); }); const downloadResult = await Promise.allSettled(downloads); for (const result of downloadResult) { if (result.status === "rejected") throw result.reason; } } } _DOWNLOAD_RETRY = new WeakMap(); _downloadQueue = new WeakMap(); _Downloader_instances = new WeakSet(); xhr_fn = function(config, signal) { if (signal == null ? undefined : signal.aborted) return Promise.resolve([signal.reason, null]); const { src, onProgress, timeout, taskId, headers } = config; return new Promise((resolve) => { const abortObj = _GM_xmlhttpRequest({ url: src, method: "GET", responseType: "blob", timeout, headers, ontimeout() { resolve([new TimoutError(`${taskId} | ${src}`), null]); }, onerror(error) { let err; if (error.status === 429) { err = new RequestError(config.src, error.status); } else { err = new Error(`Download failed. ID: ${taskId}.`); } resolve([err, null]); }, onprogress(res) { if (res.loaded > 0 && res.total > 0) { const progress = Math.floor(res.loaded / res.total * 100); onProgress == null ? undefined : onProgress(progress, config); } }, onload(res) { const { status, statusText, finalUrl } = res; if (status < 200 || status > 299) { resolve([new RequestError(statusText + " " + finalUrl, status), null]); return; } resolve([null, res.response]); } }); signal == null ? undefined : signal.addEventListener( "abort", () => { abortObj.abort(); resolve([signal.reason, null]); }, { once: true } ); }); }; dispatchDownload_fn = async function(config, priority = 0, signal) { var _a2, _b2, _c2, _d2; try { const { src, taskId, path, useFileSystemAccessApi, filenameConflictAction } = config; const result = await __privateGet(this, _downloadQueue).add( async ({ signal: signal2 }) => { signal2 == null ? void 0 : signal2.throwIfAborted(); logger.info("Download start:", src); let result2; let retryCount = 0; do { const xhrResult = await __privateMethod(this, _Downloader_instances, xhr_fn).call(this, config, signal2); signal2 == null ? void 0 : signal2.throwIfAborted(); if (xhrResult[0] === null) { result2 = xhrResult[1]; break; } else { const err = xhrResult[0]; if (!(err instanceof TimoutError) || ++retryCount > __privateGet(this, _DOWNLOAD_RETRY)) { throw err; } else { logger.error(`Download will be retried. Count: ${retryCount}.`, err); } } } while (retryCount <= __privateGet(this, _DOWNLOAD_RETRY)); return result2; }, { signal, priority } ); if (!result) throw new TimoutError(`${taskId} | ${src}`); logger.info("Xhr complete:", src); (_a2 = config.onXhrLoaded) == null ? void 0 : _a2.call(config, config); const saveFile2 = fileSaveAdapters.getAdapter(useFileSystemAccessApi, filenameConflictAction); if (typeof config.beforeFileSave === "function") { const modifiedResult = await config.beforeFileSave(result, config); signal == null ? void 0 : signal.throwIfAborted(); if (!modifiedResult) return; await saveFile2(modifiedResult, path, signal); } else { await saveFile2(result, path, signal); } (_b2 = config.onFileSaved) == null ? void 0 : _b2.call(config, config); logger.info("Download complete:", path); } catch (error) { if (error instanceof CancelError) { (_c2 = config.onAbort) == null ? undefined : _c2.call(config, config); } else { (_d2 = config.onError) == null ? undefined : _d2.call(config, error, config); } throw error; } }; const downloader = new Downloader(); const downloadSetting = createPersistedStore( "pdl-download-setting", { directoryTemplate: legacyConfig.folderPattern ?? "", filenameTemplate: legacyConfig.filenamePattern ?? "{id}", useFileSystemAccessApi: legacyConfig.useFileSystemAccess ?? false, filenameConflictAction: legacyConfig.fileSystemFilenameConflictAction ?? FilenameConflictAction.UNIQUIFY }, { get isSubpathSupported() { return this.useFileSystemAccessApi || env.isSupportSubpath(); }, setDirectoryTemplate(template2) { const newTemplate = template2.split("/").map(replaceInvalidChar).filter(Boolean).join("/"); if (newTemplate !== this.directoryTemplate) { this.directoryTemplate = newTemplate; } return newTemplate; }, setFilenameTemplate(template2) { const newTemplate = replaceInvalidChar(template2); if (newTemplate === "") return this.filenameTemplate; if (newTemplate !== this.filenameTemplate) { this.filenameTemplate = newTemplate; } return newTemplate; } } ); var PixivTagLocale = /* @__PURE__ */ ((PixivTagLocale2) => { PixivTagLocale2["JAPANESE"] = "ja"; PixivTagLocale2["CHINESE"] = "zh"; PixivTagLocale2["TRADITIONAL_CHINESE"] = "zh_tw"; PixivTagLocale2["ENGLISH"] = "en"; return PixivTagLocale2; })(PixivTagLocale || {}); const siteFeature = createPersistedStore("pdl-site-state", { ugoiraFormat: null, mixSeasonalEffect: null, tagLocale: null, compressMultiIllusts: null, compressManga: null, addBookmark: legacyConfig.addBookmark ?? null, bookmarkWithTags: null, privateBookmarkIfR18: null, likeIllustWhenDownloading: null }); async function updatefsaDir() { downloader.updateDirHandle(); } var root_1$6 = /* @__PURE__ */ template(`<input type="text">`); var root_2$1 = /* @__PURE__ */ template(`<input type="text" disabled>`); var root_3$3 = /* @__PURE__ */ template(`<button class="chip variant-soft hover:variant-filled"><span> </span></button>`); var root_5$1 = /* @__PURE__ */ template(`<!> <!> <!>`, 1); var root_4$2 = /* @__PURE__ */ template(`<li><p class="flex-auto"> </p> <span class="text-sm italic"> </span> <button class="btn btn-sm variant-filled"> </button></li> <li><p class="flex-auto"> </p> <!></li>`, 1); var root_9$3 = /* @__PURE__ */ template(`<button class="chip variant-soft hover:variant-filled"><span> </span></button>`); var root_11$1 = /* @__PURE__ */ template(`<!> <!> <!> <!>`, 1); var root_10 = /* @__PURE__ */ template(`<li><div class="flex-auto"><p> </p> <p> </p></div> <!></li>`); var root$6 = /* @__PURE__ */ template(`<div><section><p> </p> <ul><li class=" flex-col gap-3"><div class="input-group input-group-divider grid-cols-[auto_1fr_auto_auto]"><button type="button" class="[&:not([disabled])]:variant-soft-primary"><i class=" w-6 fill-current"><!></i></button> <!> <button type="button" class="variant-soft-surface [&:not([disabled])]:variant-soft-primary"><i class=" w-6 fill-current"><!></i></button></div> <div class=" flex flex-wrap self-start gap-y-1 gap-x-2"></div></li> <li><p class="flex-auto"> </p> <!></li> <!></ul></section> <section><p> </p> <ul><li class=" flex-col gap-3"><div class="input-group input-group-divider grid-cols-[auto_1fr_auto]"><button type="button" class="[&:not([disabled])]:variant-soft-primary"><i class=" w-6 fill-current"><!></i></button> <input type="text" required> <button type="button" class="variant-soft-surface dark:variant-fill-surface [&:not([disabled])]:variant-soft-primary"><i class=" w-6 fill-current"><!></i></button></div> <div class=" flex flex-wrap self-start gap-y-1 gap-x-2"></div></li> <!></ul></section></div>`); function SaveTo($$anchor, $$props) { push($$props, true); const [$$stores, $$cleanup] = setup_stores(); const $directoryHandleStore = () => store_get(directoryHandleStore, "$directoryHandleStore", $$stores); let bg = prop($$props, "bg", 3, "bg-white/30 dark:bg-black/15"), border = prop($$props, "border", 3, "divide-y-[1px] *:border-surface-300-600-token"), padding = prop($$props, "padding", 3, "px-4 *:py-4"), margin = prop($$props, "margin", 3, "mt-2 *:!m-0"), rounded = prop($$props, "rounded", 3, "rounded-container-token *:!rounded-none"), sectionSpace = prop($$props, "sectionSpace", 19, () => `space-y-4`), sectionTitle = prop($$props, "sectionTitle", 3, "font-bold"), UlClass = prop($$props, "class", 3, ""), templates = prop($$props, "templates", 19, () => getContext("supportedTemplate")), descriptionText = prop($$props, "descriptionText", 3, "text-sm text-surface-400"); const ulClasses = /* @__PURE__ */ derived(() => `list *:items-center ${padding()} ${margin()} ${border()} ${bg()} ${rounded()} ${UlClass()}`); let directoryRef; let filenameRef; let directory = state(proxy(downloadSetting.directoryTemplate)); let filename = state(proxy(downloadSetting.filenameTemplate)); const directoryHandleStore = Dexie.liveQuery(async () => { var _a2; return ((_a2 = await historyDb.getDirectoryHandle()) == null ? undefined : _a2.name) ?? ""; }); async function resetFolder() { set(directory, proxy(downloadSetting.directoryTemplate)); await tick(); const pos = get$1(directory).length; directoryRef.focus(); directoryRef.setSelectionRange(pos, pos); } async function resetFilename() { set(filename, proxy(downloadSetting.filenameTemplate)); await tick(); const pos = get$1(filename).length; filenameRef.focus(); filenameRef.setSelectionRange(pos, pos); } function insertDirTemplateAtCursor(template2) { return async () => { const start = directoryRef.selectionStart; const end = directoryRef.selectionEnd; set(directory, get$1(directory).slice(0, start) + template2 + get$1(directory).slice(end)); await tick(); const newStart = start + template2.length; directoryRef.focus(); directoryRef.setSelectionRange(newStart, newStart); }; } function insertFilenameTemplateAtCursor(template2) { return async () => { const start = filenameRef.selectionStart; const end = filenameRef.selectionEnd; set(filename, get$1(filename).slice(0, start) + template2 + get$1(filename).slice(end)); await tick(); const newStart = start + template2.length; filenameRef.focus(); filenameRef.setSelectionRange(newStart, newStart); }; } const folderBtnDisabled = /* @__PURE__ */ derived(() => get$1(directory) === downloadSetting.directoryTemplate); const filenameBtnDisabled = /* @__PURE__ */ derived(() => get$1(filename) === downloadSetting.filenameTemplate); const directoryPlaceholder = /* @__PURE__ */ derived(() => downloadSetting.isSubpathSupported ? t("setting.save_to.placeholder.sub_directory_unused") : env.isViolentmonkey() ? t("setting.save_to.placeholder.vm_not_supported") : t("setting.save_to.placeholder.need_browser_api")); var div = root$6(); var section = child(div); var p = child(section); var text$1 = child(p, true); template_effect(() => set_text(text$1, t("setting.save_to.label.directory"))); reset(p); var ul = sibling(p, 2); var li = child(ul); var div_1 = child(li); var button2 = child(div_1); button2.__click = resetFolder; var i = child(button2); var node = child(i); html(node, () => folderSvg); reset(i); reset(button2); var node_1 = sibling(button2, 2); { var consequent = ($$anchor2) => { var input = root_1$6(); remove_input_defaults(input); bind_this(input, ($$value) => directoryRef = $$value, () => directoryRef); template_effect(() => set_attribute(input, "placeholder", get$1(directoryPlaceholder))); bind_value(input, () => get$1(directory), ($$value) => set(directory, $$value)); append($$anchor2, input); }; var alternate = ($$anchor2) => { var input_1 = root_2$1(); template_effect(() => set_attribute(input_1, "placeholder", get$1(directoryPlaceholder))); append($$anchor2, input_1); }; if_block(node_1, ($$render) => { if (downloadSetting.isSubpathSupported) $$render(consequent); else $$render(alternate, false); }); } var button_1 = sibling(node_1, 2); button_1.__click = () => set(directory, proxy(downloadSetting.setDirectoryTemplate(get$1(directory)))); var i_1 = child(button_1); var node_2 = child(i_1); html(node_2, () => check); reset(i_1); reset(button_1); reset(div_1); var div_2 = sibling(div_1, 2); each(div_2, 21, () => Object.entries(templates()), index, ($$anchor2, $$item) => { let template2 = () => get$1($$item)[0]; let description = () => get$1($$item)[1]; var button_2 = root_3$3(); var event_handler = /* @__PURE__ */ derived(() => insertDirTemplateAtCursor(`{${template2()}}`)); button_2.__click = function(...$$args) { var _a2; (_a2 = get$1(event_handler)) == null ? undefined : _a2.apply(this, $$args); }; var span = child(button_2); var text_1 = child(span, true); reset(span); reset(button_2); template_effect(() => { button_2.disabled = !downloadSetting.isSubpathSupported; set_text(text_1, description()); }); append($$anchor2, button_2); }); reset(div_2); reset(li); var li_1 = sibling(li, 2); var p_1 = child(li_1); var text_2 = child(p_1, true); template_effect(() => set_text(text_2, t("setting.save_to.options.use_fsa"))); reset(p_1); var node_3 = sibling(p_1, 2); var disabled = /* @__PURE__ */ derived(() => !env.isFileSystemAccessAvaliable()); SlideToggle(node_3, { size: "sm", name: "fsa-enable", get disabled() { return get$1(disabled); }, get checked() { return downloadSetting.useFileSystemAccessApi; }, set checked($$value) { downloadSetting.useFileSystemAccessApi = $$value; } }); reset(li_1); var node_4 = sibling(li_1, 2); { var consequent_1 = ($$anchor2) => { var fragment = root_4$2(); var li_2 = first_child(fragment); var p_2 = child(li_2); var text_3 = child(p_2, true); template_effect(() => set_text(text_3, t("setting.save_to.options.fsa_directory"))); reset(p_2); var span_1 = sibling(p_2, 2); var text_4 = child(span_1, true); reset(span_1); var button_3 = sibling(span_1, 2); button_3.__click = [updatefsaDir]; var text_5 = child(button_3, true); template_effect(() => set_text(text_5, t("setting.save_to.button.choose_fsa_directory"))); reset(button_3); reset(li_2); var li_3 = sibling(li_2, 2); var p_3 = child(li_3); var text_6 = child(p_3, true); template_effect(() => set_text(text_6, t("setting.save_to.options.fsa_filename_conflict"))); reset(p_3); var node_5 = sibling(p_3, 2); RadioGroup(node_5, { class: "shrink-0", children: ($$anchor3, $$slotProps) => { var fragment_1 = root_5$1(); var node_6 = first_child(fragment_1); RadioItem(node_6, { name: "FilenameConflictAction", class: "text-sm", get value() { return FilenameConflictAction.UNIQUIFY; }, get group() { return downloadSetting.filenameConflictAction; }, set group($$value) { downloadSetting.filenameConflictAction = $$value; }, children: ($$anchor4, $$slotProps2) => { next(); var text_7 = text(); template_effect(() => set_text(text_7, t("setting.save_to.radio.filename_conflict_option_uniquify"))); append($$anchor4, text_7); }, $$slots: { default: true } }); var node_7 = sibling(node_6, 2); RadioItem(node_7, { name: "FilenameConflictAction", class: "text-sm", get value() { return FilenameConflictAction.OVERWRITE; }, get group() { return downloadSetting.filenameConflictAction; }, set group($$value) { downloadSetting.filenameConflictAction = $$value; }, children: ($$anchor4, $$slotProps2) => { next(); var text_8 = text(); template_effect(() => set_text(text_8, t("setting.save_to.radio.filename_conflict_option_overwrite"))); append($$anchor4, text_8); }, $$slots: { default: true } }); var node_8 = sibling(node_7, 2); RadioItem(node_8, { name: "FilenameConflictAction", class: "text-sm", get value() { return FilenameConflictAction.PROMPT; }, get group() { return downloadSetting.filenameConflictAction; }, set group($$value) { downloadSetting.filenameConflictAction = $$value; }, children: ($$anchor4, $$slotProps2) => { next(); var text_9 = text(); template_effect(() => set_text(text_9, t("setting.save_to.radio.filename_conflict_option_prompt"))); append($$anchor4, text_9); }, $$slots: { default: true } }); append($$anchor3, fragment_1); }, $$slots: { default: true } }); reset(li_3); template_effect(() => set_text(text_4, $directoryHandleStore())); append($$anchor2, fragment); }; if_block(node_4, ($$render) => { if (downloadSetting.useFileSystemAccessApi) $$render(consequent_1); }); } reset(ul); reset(section); var section_1 = sibling(section, 2); var p_4 = child(section_1); var text_10 = child(p_4, true); template_effect(() => set_text(text_10, t("setting.save_to.label.filename"))); reset(p_4); var ul_1 = sibling(p_4, 2); var li_4 = child(ul_1); var div_3 = child(li_4); var button_4 = child(div_3); button_4.__click = resetFilename; var i_2 = child(button_4); var node_9 = child(i_2); html(node_9, () => fileSvg); reset(i_2); reset(button_4); var input_2 = sibling(button_4, 2); remove_input_defaults(input_2); template_effect(() => set_attribute(input_2, "placeholder", t("setting.save_to.placeholder.filename_requried"))); bind_this(input_2, ($$value) => filenameRef = $$value, () => filenameRef); var button_5 = sibling(input_2, 2); button_5.__click = () => set(filename, proxy(downloadSetting.setFilenameTemplate(get$1(filename)))); var i_3 = child(button_5); var node_10 = child(i_3); html(node_10, () => check); reset(i_3); reset(button_5); reset(div_3); var div_4 = sibling(div_3, 2); each(div_4, 21, () => Object.entries(templates()), index, ($$anchor2, $$item) => { let template2 = () => get$1($$item)[0]; let description = () => get$1($$item)[1]; var button_6 = root_9$3(); var event_handler_1 = /* @__PURE__ */ derived(() => insertFilenameTemplateAtCursor(`{${template2()}}`)); button_6.__click = function(...$$args) { var _a2; (_a2 = get$1(event_handler_1)) == null ? undefined : _a2.apply(this, $$args); }; var span_2 = child(button_6); var text_11 = child(span_2, true); reset(span_2); reset(button_6); template_effect(() => set_text(text_11, description())); append($$anchor2, button_6); }); reset(div_4); reset(li_4); var node_11 = sibling(li_4, 2); { var consequent_2 = ($$anchor2) => { var li_5 = root_10(); var div_5 = child(li_5); var p_5 = child(div_5); var text_12 = child(p_5, true); template_effect(() => set_text(text_12, t("setting.save_to.options.tag_language"))); reset(p_5); var p_6 = sibling(p_5, 2); var text_13 = child(p_6, true); template_effect(() => set_text(text_13, t("setting.save_to.options.tag_language_tips"))); reset(p_6); reset(div_5); var node_12 = sibling(div_5, 2); RadioGroup(node_12, { class: " shrink-0", children: ($$anchor3, $$slotProps) => { var fragment_5 = root_11$1(); var node_13 = first_child(fragment_5); RadioItem(node_13, { name: "tagLang", class: "text-sm", get value() { return PixivTagLocale.JAPANESE; }, get group() { return siteFeature.tagLocale; }, set group($$value) { siteFeature.tagLocale = $$value; }, children: ($$anchor4, $$slotProps2) => { next(); var text_14 = text("日本語"); append($$anchor4, text_14); }, $$slots: { default: true } }); var node_14 = sibling(node_13, 2); RadioItem(node_14, { name: "tagLang", class: "text-sm", get value() { return PixivTagLocale.CHINESE; }, get group() { return siteFeature.tagLocale; }, set group($$value) { siteFeature.tagLocale = $$value; }, children: ($$anchor4, $$slotProps2) => { next(); var text_15 = text("简中"); append($$anchor4, text_15); }, $$slots: { default: true } }); var node_15 = sibling(node_14, 2); RadioItem(node_15, { name: "tagLang", class: "text-sm", get value() { return PixivTagLocale.TRADITIONAL_CHINESE; }, get group() { return siteFeature.tagLocale; }, set group($$value) { siteFeature.tagLocale = $$value; }, children: ($$anchor4, $$slotProps2) => { next(); var text_16 = text("繁中"); append($$anchor4, text_16); }, $$slots: { default: true } }); var node_16 = sibling(node_15, 2); RadioItem(node_16, { name: "tagLang", class: "text-sm", get value() { return PixivTagLocale.ENGLISH; }, get group() { return siteFeature.tagLocale; }, set group($$value) { siteFeature.tagLocale = $$value; }, children: ($$anchor4, $$slotProps2) => { next(); var text_17 = text("En"); append($$anchor4, text_17); }, $$slots: { default: true } }); append($$anchor3, fragment_5); }, $$slots: { default: true } }); reset(li_5); template_effect(() => set_class(p_6, clsx(descriptionText()))); append($$anchor2, li_5); }; if_block(node_11, ($$render) => { if (env.isPixiv()) $$render(consequent_2); }); } reset(ul_1); reset(section_1); reset(div); template_effect(() => { set_class(div, clsx(sectionSpace())); set_class(p, clsx(sectionTitle())); set_class(ul, clsx(get$1(ulClasses))); button2.disabled = get$1(folderBtnDisabled); button_1.disabled = get$1(folderBtnDisabled); set_class(p_4, clsx(sectionTitle())); set_class(ul_1, clsx(get$1(ulClasses))); button_4.disabled = get$1(filenameBtnDisabled); button_5.disabled = get$1(filenameBtnDisabled); }); bind_value(input_2, () => get$1(filename), ($$value) => set(filename, $$value)); append($$anchor, div); pop(); $$cleanup(); } delegate(["click"]); const convertSetting = createPersistedStore("pdl-convert-setting", { mp4Bitrate: legacyConfig.mp4Bitrate ?? 20, webmBitrate: legacyConfig.webmBitrate ?? 20, gifQuality: legacyConfig.gifQuality ?? 10, pngColor: legacyConfig.pngColor ?? 256, losslessWebp: legacyConfig.losslessWebp ?? false, webpQuality: legacyConfig.webpQuality ?? 95, webpMehtod: legacyConfig.webpMehtod ?? 4 }); var root_1$5 = /* @__PURE__ */ template(`<!> <!> <!> <!> <!> <!>`, 1); var root_8 = /* @__PURE__ */ template(`<option> </option>`); var root_9$2 = /* @__PURE__ */ template(`<option> </option>`); var root$5 = /* @__PURE__ */ template(`<div><section><p> </p> <ul><li><p class="flex-auto"> </p> <!></li></ul></section> <section><p> </p> <ul><li><div class="flex-auto"><p>Webm</p> <p>Bitrate (Mbps)</p></div> <input type="number" required min="1" max="120" step="1"></li> <li><div class="flex-auto"><p>Mp4</p> <p>Bitrate (Mbps)</p></div> <input type="number" required min="1" max="99" step="1"></li> <li class="flex-col !items-stretch"><p>Webp</p> <ul><li class="items-center"><p class="flex-auto"> </p> <!></li> <li class="items-center"><div class="flex-auto"><p> </p> <p> </p></div> <input type="number" required min="0" max="100" step="1"></li> <li class="items-center"><div class="flex-auto"><p> </p> <p> </p></div> <select></select></li></ul></li> <li><div class="flex-auto"><p>Gif</p> <p> </p></div> <select></select></li> <li><div class="flex-auto"><p>Png</p> <p> </p></div> <input type="number" required min="0" max="256" step="1"></li></ul></section></div>`); function UgoiraConvert($$anchor, $$props) { push($$props, true); let bg = prop($$props, "bg", 3, "bg-white/30 dark:bg-black/15"), border = prop($$props, "border", 3, "divide-y-[1px] *:border-surface-300-600-token"), padding = prop($$props, "padding", 3, "px-4 *:py-4"), margin = prop($$props, "margin", 3, "mt-2 *:!m-0"), rounded = prop($$props, "rounded", 3, "rounded-container-token *:!rounded-none"), sectionSpace = prop($$props, "sectionSpace", 19, () => `space-y-4`), sectionTitle = prop($$props, "sectionTitle", 3, "font-bold"), UlClass = prop($$props, "class", 3, ""), descriptionText = prop($$props, "descriptionText", 3, "text-sm text-surface-400"), inputRounded = prop($$props, "inputRounded", 3, "rounded-full"), inputWidth = prop($$props, "inputWidth", 3, "w-32"); const ulClasses = /* @__PURE__ */ derived(() => `list *:items-center ${padding()} ${margin()} ${border()} ${bg()} ${rounded()} ${UlClass()}`); const inputClasses = /* @__PURE__ */ derived(() => `${inputWidth()} ${inputRounded()} shrink-0`); var div = root$5(); var section = child(div); var p = child(section); var text$1 = child(p, true); template_effect(() => set_text(text$1, t("setting.ugoira.label.format"))); reset(p); var ul = sibling(p, 2); var li = child(ul); var p_1 = child(li); var text_1 = child(p_1, true); template_effect(() => set_text(text_1, t("setting.ugoira.options.select_format"))); reset(p_1); var node = sibling(p_1, 2); RadioGroup(node, { class: "shrink-0", children: ($$anchor2, $$slotProps) => { var fragment = root_1$5(); var node_1 = first_child(fragment); RadioItem(node_1, { name: "ugoiraFormat", class: "text-sm", value: "zip", get group() { return siteFeature.ugoiraFormat; }, set group($$value) { siteFeature.ugoiraFormat = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_2 = text("Zip"); append($$anchor3, text_2); }, $$slots: { default: true } }); var node_2 = sibling(node_1, 2); RadioItem(node_2, { name: "ugoiraFormat", class: "text-sm", get value() { return ConvertFormat.WEBM; }, get group() { return siteFeature.ugoiraFormat; }, set group($$value) { siteFeature.ugoiraFormat = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_3 = text("Webm"); append($$anchor3, text_3); }, $$slots: { default: true } }); var node_3 = sibling(node_2, 2); var disabled = /* @__PURE__ */ derived(() => !env.videoFrameSupported()); RadioItem(node_3, { get disabled() { return get$1(disabled); }, class: "text-sm", name: "ugoiraFormat", get value() { return ConvertFormat.MP4; }, get group() { return siteFeature.ugoiraFormat; }, set group($$value) { siteFeature.ugoiraFormat = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_4 = text("Mp4"); append($$anchor3, text_4); }, $$slots: { default: true } }); var node_4 = sibling(node_3, 2); RadioItem(node_4, { name: "ugoiraFormat", class: "text-sm", get value() { return ConvertFormat.WEBP; }, get group() { return siteFeature.ugoiraFormat; }, set group($$value) { siteFeature.ugoiraFormat = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_5 = text("Webp"); append($$anchor3, text_5); }, $$slots: { default: true } }); var node_5 = sibling(node_4, 2); RadioItem(node_5, { name: "ugoiraFormat", class: "text-sm", get value() { return ConvertFormat.GIF; }, get group() { return siteFeature.ugoiraFormat; }, set group($$value) { siteFeature.ugoiraFormat = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_6 = text("Gif"); append($$anchor3, text_6); }, $$slots: { default: true } }); var node_6 = sibling(node_5, 2); RadioItem(node_6, { name: "ugoiraFormat", class: "text-sm", get value() { return ConvertFormat.PNG; }, get group() { return siteFeature.ugoiraFormat; }, set group($$value) { siteFeature.ugoiraFormat = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_7 = text("Png"); append($$anchor3, text_7); }, $$slots: { default: true } }); append($$anchor2, fragment); }, $$slots: { default: true } }); reset(li); reset(ul); reset(section); var section_1 = sibling(section, 2); var p_2 = child(section_1); var text_8 = child(p_2, true); template_effect(() => set_text(text_8, t("setting.ugoira.label.quality"))); reset(p_2); var ul_1 = sibling(p_2, 2); var li_1 = child(ul_1); var div_1 = child(li_1); var p_3 = sibling(child(div_1), 2); reset(div_1); var input = sibling(div_1, 2); action(input, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return convertSetting.webmBitrate; }, set(value) { convertSetting.webmBitrate = value; } })); reset(li_1); var li_2 = sibling(li_1, 2); var div_2 = child(li_2); var p_4 = sibling(child(div_2), 2); reset(div_2); var input_1 = sibling(div_2, 2); action(input_1, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return convertSetting.mp4Bitrate; }, set(value) { convertSetting.mp4Bitrate = value; } })); reset(li_2); var li_3 = sibling(li_2, 2); var ul_2 = sibling(child(li_3), 2); var li_4 = child(ul_2); var p_5 = child(li_4); var text_9 = child(p_5, true); template_effect(() => set_text(text_9, t("setting.ugoira.options.webp_lossy"))); reset(p_5); var node_7 = sibling(p_5, 2); SlideToggle(node_7, { name: "lossless-webp", size: "sm", get checked() { return convertSetting.losslessWebp; }, set checked($$value) { convertSetting.losslessWebp = $$value; } }); reset(li_4); var li_5 = sibling(li_4, 2); var div_3 = child(li_5); var p_6 = child(div_3); var text_10 = child(p_6, true); template_effect(() => set_text(text_10, t("setting.ugoira.options.webp_quality"))); reset(p_6); var p_7 = sibling(p_6, 2); var text_11 = child(p_7, true); template_effect(() => set_text(text_11, t("setting.ugoira.options.webp_quality_tips"))); reset(p_7); reset(div_3); var input_2 = sibling(div_3, 2); action(input_2, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return convertSetting.webpQuality; }, set(value) { convertSetting.webpQuality = value; } })); reset(li_5); var li_6 = sibling(li_5, 2); var div_4 = child(li_6); var p_8 = child(div_4); var text_12 = child(p_8, true); template_effect(() => set_text(text_12, t("setting.ugoira.options.webp_method"))); reset(p_8); var p_9 = sibling(p_8, 2); var text_13 = child(p_9, true); template_effect(() => set_text(text_13, t("setting.ugoira.options.webp_method_tips"))); reset(p_9); reset(div_4); var select = sibling(div_4, 2); each(select, 20, () => Array.from({ length: 7 }, (_, idx) => idx), index, ($$anchor2, quality) => { var option = root_8(); var option_value = {}; var text_14 = child(option, true); reset(option); template_effect(() => { if (option_value !== (option_value = quality)) { option.value = null == (option.__value = quality) ? "" : quality; } set_text(text_14, quality); }); append($$anchor2, option); }); reset(select); reset(li_6); reset(ul_2); reset(li_3); var li_7 = sibling(li_3, 2); var div_5 = child(li_7); var p_10 = sibling(child(div_5), 2); var text_15 = child(p_10, true); template_effect(() => set_text(text_15, t("setting.ugoira.options.gif_tips"))); reset(p_10); reset(div_5); var select_1 = sibling(div_5, 2); each(select_1, 20, () => Array.from({ length: 20 }, (_, idx) => idx), index, ($$anchor2, quality) => { var option_1 = root_9$2(); var option_1_value = {}; var text_16 = child(option_1, true); reset(option_1); template_effect(() => { if (option_1_value !== (option_1_value = quality + 1)) { option_1.value = null == (option_1.__value = quality + 1) ? "" : quality + 1; } set_text(text_16, quality + 1); }); append($$anchor2, option_1); }); reset(select_1); reset(li_7); var li_8 = sibling(li_7, 2); var div_6 = child(li_8); var p_11 = sibling(child(div_6), 2); var text_17 = child(p_11, true); template_effect(() => set_text(text_17, t("setting.ugoira.options.png_tips"))); reset(p_11); reset(div_6); var input_3 = sibling(div_6, 2); action(input_3, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return convertSetting.pngColor; }, set(value) { convertSetting.pngColor = value; } })); reset(li_8); reset(ul_1); reset(section_1); reset(div); template_effect(() => { set_class(div, clsx(sectionSpace())); set_class(p, clsx(sectionTitle())); set_class(ul, clsx(get$1(ulClasses))); set_class(p_2, clsx(sectionTitle())); set_class(ul_1, clsx(get$1(ulClasses))); set_class(p_3, clsx(descriptionText())); set_class(input, `input ${get$1(inputClasses) ?? ""}`); set_class(p_4, clsx(descriptionText())); set_class(input_1, `input ${get$1(inputClasses) ?? ""}`); set_class(ul_2, `list ${border() ?? ""} ${rounded() ?? ""} [&:not(:last-child)]:*:py-4 [&:last-child]:*:pt-4`); set_class(p_7, clsx(descriptionText())); set_class(input_2, `input ${get$1(inputClasses) ?? ""}`); set_class(p_9, clsx(descriptionText())); set_class(select, `select ${get$1(inputClasses) ?? ""}`); set_class(p_10, clsx(descriptionText())); set_class(select_1, `select ${get$1(inputClasses) ?? ""}`); set_class(p_11, clsx(descriptionText())); set_class(input_3, `input ${get$1(inputClasses) ?? ""}`); }); bind_select_value(select, () => convertSetting.webpMehtod, ($$value) => convertSetting.webpMehtod = $$value); bind_select_value(select_1, () => convertSetting.gifQuality, ($$value) => convertSetting.gifQuality = $$value); append($$anchor, div); pop(); } 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) => { var _a2; const text2 = (_a2 = readEvt.target) == null ? undefined : _a2.result; if (typeof text2 !== "string") throw new Error("Invalid file"); const history2 = JSON.parse(text2); if (!(history2 instanceof Array)) throw new Error("Invalid file"); resolve(history2); }; }); } function importJSON(file) { return readHistoryFile("application/json", file).then((data) => historyDb.import(data)); } function exportAsJSON() { return historyDb.getAll().then((datas) => { const str = JSON.stringify(datas); const blob = new Blob([str], { type: "application/json" }); const filename = `Pixiv Downloader_${location.hostname}_${(/* @__PURE__ */ new Date()).toLocaleString()}.json`; aDownload$1(blob, filename); }); } function exportAsCSV() { return historyDb.generateCsv().then((csv) => { const filename = `Pixiv Downloader_${location.hostname}_${(/* @__PURE__ */ new Date()).toLocaleString()}.csv`; aDownload$1(csv, filename); }); } function useHistoryBackup() { return { importJSON, exportAsJSON, exportAsCSV }; } var BackupInterval = /* @__PURE__ */ ((BackupInterval2) => { BackupInterval2[BackupInterval2["NEVER"] = 0] = "NEVER"; BackupInterval2[BackupInterval2["EVERY_DAY"] = 864e5] = "EVERY_DAY"; BackupInterval2[BackupInterval2["EVERY_7_DAY"] = 6048e5] = "EVERY_7_DAY"; BackupInterval2[BackupInterval2["EVERY_30_DAY"] = 2592e6] = "EVERY_30_DAY"; return BackupInterval2; })(BackupInterval || {}); const backupSetting = createPersistedStore("pdl-backup-setting", { interval: legacyConfig.historyBackupInterval ?? 0, /* NEVER */ lastTimestamp: legacyConfig.lastHistoryBackup ?? 0 }); var root_3$2 = /* @__PURE__ */ template(`<!> <span> </span>`, 1); var root$4 = /* @__PURE__ */ template(`<div><section><p> </p> <ul><li><p class="flex-auto"> </p> <select><option> </option><option> </option><option> </option><option> </option></select></li></ul></section> <section><p> </p> <ul><li><p class="flex-auto"> </p> <button class="btn variant-filled"><!> <span> </span></button></li> <li><p class="flex-auto"> </p> <button class="btn variant-filled"><!> <span> </span></button></li></ul></section> <section><p> </p> <ul><li><p class="flex-auto"> </p> <!></li></ul></section> <section><p> </p> <ul><li><p class="flex-auto"> </p> <button class="btn variant-filled"><!> <span> </span></button></li></ul></section></div>`); function DownloadHistory($$anchor, $$props) { push($$props, true); const [$$stores, $$cleanup] = setup_stores(); const $exportJsonPending = () => store_get(exportJsonPending, "$exportJsonPending", $$stores); const $exportCsvPending = () => store_get(exportCsvPending, "$exportCsvPending", $$stores); const $importPending = () => store_get(importPending, "$importPending", $$stores); const $clearPending = () => store_get(clearPending, "$clearPending", $$stores); let bg = prop($$props, "bg", 3, "bg-white/30 dark:bg-black/15"), border = prop($$props, "border", 3, "divide-y-[1px] *:border-surface-300-600-token"), padding = prop($$props, "padding", 3, "px-4 *:py-4"), margin = prop($$props, "margin", 3, "mt-2 *:!m-0"), rounded = prop($$props, "rounded", 3, "rounded-container-token *:!rounded-none"), sectionSpace = prop($$props, "sectionSpace", 19, () => `space-y-4`), sectionTitle = prop($$props, "sectionTitle", 3, "font-bold"), inputRounded = prop($$props, "inputRounded", 3, "rounded-full"), inputWidth = prop($$props, "inputWidth", 3, "w-32"), UlClass = prop($$props, "class", 3, ""); const ulClasses = /* @__PURE__ */ derived(() => `list *:items-center ${padding()} ${margin()} ${border()} ${bg()} ${rounded()} ${UlClass()}`); const inputClasses = /* @__PURE__ */ derived(() => `${inputWidth()} ${inputRounded()} shrink-0`); function usePendingButton(fn) { const store = writable(false); return { store, async wrapFn(...args) { store.set(true); const result = await fn(...args); store.set(false); return result; } }; } async function importFromJSON(evt) { var _a2; const file = (_a2 = evt.currentTarget.files) == null ? undefined : _a2[0]; if (!file) return; try { await importJSON2(file); } catch (error) { logger.error(error); alert(error); } } function clearDb() { const isConfirm = confirm(t("setting.history.text.confirm_clear_history")); if (!isConfirm) return; return historyDb.clear(); } const { importJSON: importJSON2, exportAsJSON: exportAsJSON2, exportAsCSV: exportAsCSV2 } = useHistoryBackup(); const { store: importPending, wrapFn: wrapImportFromJSON } = usePendingButton(importFromJSON); const { store: clearPending, wrapFn: wrapClearDb } = usePendingButton(clearDb); const { store: exportJsonPending, wrapFn: wrapExportAsJSON } = usePendingButton(exportAsJSON2); const { store: exportCsvPending, wrapFn: wrapExportAsCSV } = usePendingButton(exportAsCSV2); var div = root$4(); var section = child(div); var p = child(section); var text2 = child(p, true); template_effect(() => set_text(text2, t("setting.history.label.scheduled_backups"))); reset(p); var ul = sibling(p, 2); var li = child(ul); var p_1 = child(li); var text_1 = child(p_1, true); template_effect(() => set_text(text_1, t("setting.history.options.scheduled_backups"))); reset(p_1); var select = sibling(p_1, 2); var option = child(select); var option_value = {}; var text_2 = child(option, true); template_effect(() => set_text(text_2, t("setting.history.select.backup_interval_never"))); reset(option); var option_1 = sibling(option); var option_1_value = {}; var text_3 = child(option_1, true); template_effect(() => set_text(text_3, t("setting.history.select.backup_interval_every_day"))); reset(option_1); var option_2 = sibling(option_1); var option_2_value = {}; var text_4 = child(option_2, true); template_effect(() => set_text(text_4, t("setting.history.select.backup_interval_every_7_day"))); reset(option_2); var option_3 = sibling(option_2); var option_3_value = {}; var text_5 = child(option_3, true); template_effect(() => set_text(text_5, t("setting.history.select.backup_interval_every_30_day"))); reset(option_3); reset(select); reset(li); reset(ul); reset(section); var section_1 = sibling(section, 2); var p_2 = child(section_1); var text_6 = child(p_2, true); template_effect(() => set_text(text_6, t("setting.history.label.export"))); reset(p_2); var ul_1 = sibling(p_2, 2); var li_1 = child(ul_1); var p_3 = child(li_1); var text_7 = child(p_3, true); template_effect(() => set_text(text_7, t("setting.history.options.export_as_json"))); reset(p_3); var button2 = sibling(p_3, 2); button2.__click = async () => { await wrapExportAsJSON(); backupSetting.lastTimestamp = (/* @__PURE__ */ new Date()).getTime(); }; var node = child(button2); { var consequent = ($$anchor2) => { ProgressRadial($$anchor2, { stroke: 80, width: "w-5", meter: "stroke-primary-500", track: "stroke-primary-500/30" }); }; if_block(node, ($$render) => { if ($exportJsonPending()) $$render(consequent); }); } var span = sibling(node, 2); var text_8 = child(span, true); template_effect(() => set_text(text_8, t("setting.history.button.export"))); reset(span); reset(button2); reset(li_1); var li_2 = sibling(li_1, 2); var p_4 = child(li_2); var text_9 = child(p_4, true); template_effect(() => set_text(text_9, t("setting.history.options.export_as_csv"))); reset(p_4); var button_1 = sibling(p_4, 2); button_1.__click = wrapExportAsCSV; var node_1 = child(button_1); { var consequent_1 = ($$anchor2) => { ProgressRadial($$anchor2, { stroke: 80, width: "w-5", meter: "stroke-primary-500", track: "stroke-primary-500/30" }); }; if_block(node_1, ($$render) => { if ($exportCsvPending()) $$render(consequent_1); }); } var span_1 = sibling(node_1, 2); var text_10 = child(span_1, true); template_effect(() => set_text(text_10, t("setting.history.button.export"))); reset(span_1); reset(button_1); reset(li_2); reset(ul_1); reset(section_1); var section_2 = sibling(section_1, 2); var p_5 = child(section_2); var text_11 = child(p_5, true); template_effect(() => set_text(text_11, t("setting.history.label.import"))); reset(p_5); var ul_2 = sibling(p_5, 2); var li_3 = child(ul_2); var p_6 = child(li_3); var text_12 = child(p_6, true); template_effect(() => set_text(text_12, t("setting.history.options.import_json"))); reset(p_6); var node_2 = sibling(p_6, 2); FileButton(node_2, { name: "import-file", accept: ".json", get disabled() { mark_store_binding(); return $importPending(); }, set disabled($$value) { store_set(importPending, proxy($$value)); }, $$events: { change: wrapImportFromJSON }, children: ($$anchor2, $$slotProps) => { var fragment_2 = root_3$2(); var node_3 = first_child(fragment_2); { var consequent_2 = ($$anchor3) => { ProgressRadial($$anchor3, { stroke: 80, width: "w-5", meter: "stroke-primary-500", track: "stroke-primary-500/30" }); }; if_block(node_3, ($$render) => { if ($importPending()) $$render(consequent_2); }); } var span_2 = sibling(node_3, 2); var text_13 = child(span_2, true); template_effect(() => set_text(text_13, t("setting.history.button.import"))); reset(span_2); append($$anchor2, fragment_2); }, $$slots: { default: true } }); reset(li_3); reset(ul_2); reset(section_2); var section_3 = sibling(section_2, 2); var p_7 = child(section_3); var text_14 = child(p_7, true); template_effect(() => set_text(text_14, t("setting.history.label.clear"))); reset(p_7); var ul_3 = sibling(p_7, 2); var li_4 = child(ul_3); var p_8 = child(li_4); var text_15 = child(p_8, true); template_effect(() => set_text(text_15, t("setting.history.options.clear_history"))); reset(p_8); var button_2 = sibling(p_8, 2); button_2.__click = wrapClearDb; var node_4 = child(button_2); { var consequent_3 = ($$anchor2) => { ProgressRadial($$anchor2, { stroke: 80, width: "w-5", meter: "stroke-primary-500", track: "stroke-primary-500/30" }); }; if_block(node_4, ($$render) => { if ($clearPending()) $$render(consequent_3); }); } var span_3 = sibling(node_4, 2); var text_16 = child(span_3, true); template_effect(() => set_text(text_16, t("setting.history.button.clear"))); reset(span_3); reset(button_2); reset(li_4); reset(ul_3); reset(section_3); reset(div); template_effect(() => { set_class(div, clsx(sectionSpace())); set_class(p, clsx(sectionTitle())); set_class(ul, clsx(get$1(ulClasses))); set_class(select, `select ${get$1(inputClasses) ?? ""}`); if (option_value !== (option_value = BackupInterval.NEVER)) { option.value = null == (option.__value = BackupInterval.NEVER) ? "" : BackupInterval.NEVER; } if (option_1_value !== (option_1_value = BackupInterval.EVERY_DAY)) { option_1.value = null == (option_1.__value = BackupInterval.EVERY_DAY) ? "" : BackupInterval.EVERY_DAY; } if (option_2_value !== (option_2_value = BackupInterval.EVERY_7_DAY)) { option_2.value = null == (option_2.__value = BackupInterval.EVERY_7_DAY) ? "" : BackupInterval.EVERY_7_DAY; } if (option_3_value !== (option_3_value = BackupInterval.EVERY_30_DAY)) { option_3.value = null == (option_3.__value = BackupInterval.EVERY_30_DAY) ? "" : BackupInterval.EVERY_30_DAY; } set_class(p_2, clsx(sectionTitle())); set_class(ul_1, clsx(get$1(ulClasses))); button2.disabled = $exportJsonPending(); button_1.disabled = $exportCsvPending(); set_class(p_5, clsx(sectionTitle())); set_class(ul_2, clsx(get$1(ulClasses))); set_class(p_7, clsx(sectionTitle())); set_class(ul_3, clsx(get$1(ulClasses))); button_2.disabled = $clearPending(); }); bind_select_value(select, () => backupSetting.interval, ($$value) => backupSetting.interval = $$value); append($$anchor, div); pop(); $$cleanup(); } delegate(["click"]); const alignLeft = `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-align-left-icon lucide-align-left"><path d="M15 12H3"/><path d="M17 18H3"/><path d="M21 6H3"/></svg>`; const alignRight = `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-align-right-icon lucide-align-right"><path d="M21 12H9"/><path d="M21 18H7"/><path d="M21 6H3"/></svg>`; const alignTop = `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-align-start-horizontal-icon lucide-align-start-horizontal"><rect width="6" height="16" x="4" y="6" rx="2"/><rect width="6" height="9" x="14" y="6" rx="2"/><path d="M22 2H2"/></svg>`; const alignBottom = `<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-align-end-horizontal-icon lucide-align-end-horizontal"><rect width="6" height="16" x="4" y="2" rx="2"/><rect width="6" height="9" x="14" y="9" rx="2"/><path d="M22 22H2"/></svg>`; var root_1$4 = /* @__PURE__ */ template(`<!> <!>`, 1); var root_4$1 = /* @__PURE__ */ template(`<button class="btn variant-filled rounded-r-none text-xl flex-[1]">+</button> <input type="number" class="input variant-form-material text-sm flex-[3]" min="0"> <button class=" btn variant-filled rounded-l-none text-xl flex-[1]">-</button>`, 1); var root_6$2 = /* @__PURE__ */ template(`<!> <!>`, 1); var root_9$1 = /* @__PURE__ */ template(`<button class="btn variant-filled rounded-r-none text-xl flex-[1]">+</button> <input type="number" class="input variant-form-material text-sm flex-[3]" min="0"> <button class=" btn variant-filled rounded-l-none text-xl flex-[1]">-</button>`, 1); var root_12 = /* @__PURE__ */ template(`<div class="flex justify-between items-center"><p> </p> <div class="text-xs"> </div></div>`); var root_13 = /* @__PURE__ */ template(`<div class="flex justify-between items-center"><p> </p> <div class="text-xs"> </div></div>`); var root_11 = /* @__PURE__ */ template(`<section><p> </p> <ul><li class="flex-col !items-stretch md:flex-row md:!items-baseline gap-4 *:!m-0"><!> <!></li></ul></section>`); var root_15 = /* @__PURE__ */ template(`<i><!></i>`); var root_16 = /* @__PURE__ */ template(`<i><!></i>`); var root_14 = /* @__PURE__ */ template(`<!> <!>`, 1); var root_18 = /* @__PURE__ */ template(`<i><!></i>`); var root_19 = /* @__PURE__ */ template(`<i><!></i>`); var root_17 = /* @__PURE__ */ template(`<!> <!>`, 1); var root$3 = /* @__PURE__ */ template(`<div><div class="flex items-center justify-center"><div></div></div> <section><p> </p> <ul><li class="flex-col !items-stretch md:flex-row md:!items-baseline gap-4 *:!m-0"><div class=" flex-1"><div class="flex justify-between items-center"><p> </p> <!></div> <div class="flex flex-row mt-2 gap-1 h-8"><!></div></div> <div class=" flex-1"><div class="flex justify-between items-center"><p> </p> <!></div> <div class="flex flex-row mt-2 gap-1 h-8"><!></div></div></li></ul></section> <!> <section><p> </p> <ul><li class="flex-col !items-stretch md:flex-row md:!items-center gap-4 *:!m-0"><div class="flex flex-1 justify-between items-center"><p> </p> <!></div> <div class="flex flex-1 justify-between items-center"><p> </p> <!></div></li></ul></section></div>`); function BtnPosition($$anchor, $$props) { push($$props, true); let bg = prop($$props, "bg", 3, "bg-white/30 dark:bg-black/15"), border = prop($$props, "border", 3, "divide-y-[1px] *:border-surface-300-600-token"), padding = prop($$props, "padding", 3, "px-4 *:py-4"), margin = prop($$props, "margin", 3, "mt-2 *:!m-0"), rounded = prop($$props, "rounded", 3, "rounded-container-token *:!rounded-none"), sectionSpace = prop($$props, "sectionSpace", 19, () => `space-y-4`), sectionTitle = prop($$props, "sectionTitle", 3, "font-bold"), UlClass = prop($$props, "class", 3, ""); const ulClasses = /* @__PURE__ */ derived(() => `list *:items-center ${padding()} ${margin()} ${border()} ${bg()} ${rounded()} ${UlClass()}`); const max = 100; const step = 4; let buttonContainer; onMount(() => { const sampleBtn = new ThumbnailButton({ id: "0", shouldObserveDb: false, onClick: () => undefined }); sampleBtn.setAttribute("disabled", ""); buttonContainer.appendChild(sampleBtn); if (!env.isPixiv()) return; const sampleBookmarkBtn = new ThumbnailButton({ id: "0", type: ThumbnailBtnType.PixivMyBookmark, shouldObserveDb: false, onClick: () => undefined }); sampleBookmarkBtn.setAttribute("disabled", ""); sampleBookmarkBtn.setStatus(ThumbnailBtnStatus.Complete); buttonContainer.appendChild(sampleBookmarkBtn); }); var div = root$3(); var div_1 = child(div); var div_2 = child(div_1); bind_this(div_2, ($$value) => buttonContainer = $$value, () => buttonContainer); reset(div_1); var section = sibling(div_1, 2); var p = child(section); var text$1 = child(p, true); template_effect(() => set_text(text$1, t("setting.button_position.label.common"))); reset(p); var ul = sibling(p, 2); var li = child(ul); var div_3 = child(li); var div_4 = child(div_3); var p_1 = child(div_4); var text_1 = child(p_1, true); template_effect(() => set_text(text_1, t("setting.button_position.options.horizontal_position"))); reset(p_1); var node = sibling(p_1, 2); RadioGroup(node, { children: ($$anchor2, $$slotProps) => { var fragment = root_1$4(); var node_1 = first_child(fragment); RadioItem(node_1, { name: "thumbnail-btn-unit-x", get value() { return BtnLengthUnit.PERCENT; }, class: " text-xs", get group() { return buttonPosition.thumbnailBtnUnitX; }, set group($$value) { buttonPosition.thumbnailBtnUnitX = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_2 = text("%"); append($$anchor3, text_2); }, $$slots: { default: true } }); var node_2 = sibling(node_1, 2); RadioItem(node_2, { name: "thumbnail-btn-unit-x", get value() { return BtnLengthUnit.PX; }, class: " text-xs", get group() { return buttonPosition.thumbnailBtnUnitX; }, set group($$value) { buttonPosition.thumbnailBtnUnitX = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_3 = text("px"); append($$anchor3, text_3); }, $$slots: { default: true } }); append($$anchor2, fragment); }, $$slots: { default: true } }); reset(div_4); var div_5 = sibling(div_4, 2); var node_3 = child(div_5); { var consequent = ($$anchor2) => { var fragment_1 = root_4$1(); var button2 = first_child(fragment_1); button2.__click = () => buttonPosition["--pdl-btn-left-px"] += 1; var input = sibling(button2, 2); action(input, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return buttonPosition["--pdl-btn-left-px"]; }, set(val) { buttonPosition["--pdl-btn-left-px"] = val; } })); var button_1 = sibling(input, 2); button_1.__click = () => buttonPosition["--pdl-btn-left-px"] -= 1; template_effect(() => button_1.disabled = buttonPosition["--pdl-btn-left-px"] < 1); append($$anchor2, fragment_1); }; var alternate = ($$anchor2) => { RangeSlider($$anchor2, { name: "thumbnail-btn-left", step, max, ticked: true, class: " flex-grow self-center", get value() { return buttonPosition[ButtonStyle.LEFT_PERCENT]; }, set value($$value) { buttonPosition[ButtonStyle.LEFT_PERCENT] = $$value; } }); }; if_block(node_3, ($$render) => { if (buttonPosition.thumbnailBtnUnitX === BtnLengthUnit.PX) $$render(consequent); else $$render(alternate, false); }); } reset(div_5); reset(div_3); var div_6 = sibling(div_3, 2); var div_7 = child(div_6); var p_2 = child(div_7); var text_4 = child(p_2, true); template_effect(() => set_text(text_4, t("setting.button_position.options.vertical_position"))); reset(p_2); var node_4 = sibling(p_2, 2); RadioGroup(node_4, { children: ($$anchor2, $$slotProps) => { var fragment_3 = root_6$2(); var node_5 = first_child(fragment_3); RadioItem(node_5, { name: "thumbnail-btn-unit-y", get value() { return BtnLengthUnit.PERCENT; }, class: " text-xs", get group() { return buttonPosition.thumbnailBtnUnitY; }, set group($$value) { buttonPosition.thumbnailBtnUnitY = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_5 = text("%"); append($$anchor3, text_5); }, $$slots: { default: true } }); var node_6 = sibling(node_5, 2); RadioItem(node_6, { name: "thumbnail-btn-unit-y", get value() { return BtnLengthUnit.PX; }, class: " text-xs", get group() { return buttonPosition.thumbnailBtnUnitY; }, set group($$value) { buttonPosition.thumbnailBtnUnitY = $$value; }, children: ($$anchor3, $$slotProps2) => { next(); var text_6 = text("px"); append($$anchor3, text_6); }, $$slots: { default: true } }); append($$anchor2, fragment_3); }, $$slots: { default: true } }); reset(div_7); var div_8 = sibling(div_7, 2); var node_7 = child(div_8); { var consequent_1 = ($$anchor2) => { var fragment_4 = root_9$1(); var button_2 = first_child(fragment_4); button_2.__click = () => buttonPosition["--pdl-btn-top-px"] += 1; var input_1 = sibling(button_2, 2); action(input_1, ($$node, $$action_arg) => inputValidation == null ? undefined : inputValidation($$node, $$action_arg), () => ({ get() { return buttonPosition["--pdl-btn-top-px"]; }, set(val) { buttonPosition["--pdl-btn-top-px"] = val; } })); var button_3 = sibling(input_1, 2); button_3.__click = () => buttonPosition["--pdl-btn-top-px"] -= 1; template_effect(() => button_3.disabled = buttonPosition["--pdl-btn-top-px"] < 1); append($$anchor2, fragment_4); }; var alternate_1 = ($$anchor2) => { RangeSlider($$anchor2, { name: "thumbnail-btn-top", step, max, ticked: true, class: " flex-grow self-center", get value() { return buttonPosition[ButtonStyle.TOP_PERCENT]; }, set value($$value) { buttonPosition[ButtonStyle.TOP_PERCENT] = $$value; } }); }; if_block(node_7, ($$render) => { if (buttonPosition.thumbnailBtnUnitY === BtnLengthUnit.PX) $$render(consequent_1); else $$render(alternate_1, false); }); } reset(div_8); reset(div_6); reset(li); reset(ul); reset(section); var node_8 = sibling(section, 2); { var consequent_2 = ($$anchor2) => { var section_1 = root_11(); var p_3 = child(section_1); var text_7 = child(p_3, true); template_effect(() => set_text(text_7, t("setting.button_position.label.my_bookmark"))); reset(p_3); var ul_1 = sibling(p_3, 2); var li_1 = child(ul_1); var node_9 = child(li_1); RangeSlider(node_9, { name: "pdl-bookmark-btn-left", step, max, ticked: true, class: "flex-grow", get value() { return buttonPosition[ButtonStyle.PIXIV_BOOKMARK_LEFT_PERCENT]; }, set value($$value) { buttonPosition[ButtonStyle.PIXIV_BOOKMARK_LEFT_PERCENT] = $$value; }, children: ($$anchor3, $$slotProps) => { var div_9 = root_12(); var p_4 = child(div_9); var text_8 = child(p_4, true); template_effect(() => set_text(text_8, t("setting.button_position.options.horizontal_position"))); reset(p_4); var div_10 = sibling(p_4, 2); var text_9 = child(div_10); reset(div_10); reset(div_9); template_effect(() => set_text(text_9, `${buttonPosition[ButtonStyle.PIXIV_BOOKMARK_LEFT_PERCENT] ?? ""} / ${max}`)); append($$anchor3, div_9); }, $$slots: { default: true } }); var node_10 = sibling(node_9, 2); RangeSlider(node_10, { name: "pdl-bookmark-btn-top", step, max, ticked: true, class: "flex-grow", get value() { return buttonPosition[ButtonStyle.PIXIV_BOOKMARK_TOP_PERCENT]; }, set value($$value) { buttonPosition[ButtonStyle.PIXIV_BOOKMARK_TOP_PERCENT] = $$value; }, children: ($$anchor3, $$slotProps) => { var div_11 = root_13(); var p_5 = child(div_11); var text_10 = child(p_5, true); template_effect(() => set_text(text_10, t("setting.button_position.options.vertical_position"))); reset(p_5); var div_12 = sibling(p_5, 2); var text_11 = child(div_12); reset(div_12); reset(div_11); template_effect(() => set_text(text_11, `${buttonPosition[ButtonStyle.PIXIV_BOOKMARK_TOP_PERCENT] ?? ""} / ${max}`)); append($$anchor3, div_11); }, $$slots: { default: true } }); reset(li_1); reset(ul_1); reset(section_1); template_effect(() => { set_class(p_3, clsx(sectionTitle())); set_class(ul_1, clsx(get$1(ulClasses))); }); append($$anchor2, section_1); }; if_block(node_8, ($$render) => { if (env.isPixiv()) $$render(consequent_2); }); } var section_2 = sibling(node_8, 2); var p_6 = child(section_2); var text_12 = child(p_6, true); template_effect(() => set_text(text_12, t("setting.button_position.label.gallery"))); reset(p_6); var ul_2 = sibling(p_6, 2); var li_2 = child(ul_2); var div_13 = child(li_2); var p_7 = child(div_13); var text_13 = child(p_7, true); template_effect(() => set_text(text_13, t("setting.button_position.options.horizontal_alignment"))); reset(p_7); var node_11 = sibling(p_7, 2); RadioGroup(node_11, { children: ($$anchor2, $$slotProps) => { var fragment_6 = root_14(); var node_12 = first_child(fragment_6); RadioItem(node_12, { name: "artwork-align-x", get value() { return BtnAlignX.LEFT; }, class: " text-xs", get group() { return buttonPosition.artworkBtnAlignX; }, set group($$value) { buttonPosition.artworkBtnAlignX = $$value; }, children: ($$anchor3, $$slotProps2) => { var i = root_15(); var node_13 = child(i); html(node_13, () => alignLeft); reset(i); append($$anchor3, i); }, $$slots: { default: true } }); var node_14 = sibling(node_12, 2); RadioItem(node_14, { name: "artwork-align-x", get value() { return BtnAlignX.RIGHT; }, class: " text-xs", get group() { return buttonPosition.artworkBtnAlignX; }, set group($$value) { buttonPosition.artworkBtnAlignX = $$value; }, children: ($$anchor3, $$slotProps2) => { var i_1 = root_16(); var node_15 = child(i_1); html(node_15, () => alignRight); reset(i_1); append($$anchor3, i_1); }, $$slots: { default: true } }); append($$anchor2, fragment_6); }, $$slots: { default: true } }); reset(div_13); var div_14 = sibling(div_13, 2); var p_8 = child(div_14); var text_14 = child(p_8, true); template_effect(() => set_text(text_14, t("setting.button_position.options.vertical_alignment"))); reset(p_8); var node_16 = sibling(p_8, 2); RadioGroup(node_16, { children: ($$anchor2, $$slotProps) => { var fragment_7 = root_17(); var node_17 = first_child(fragment_7); RadioItem(node_17, { name: "artwork-align-y", get value() { return BtnAlignY.TOP; }, class: " text-xs", get group() { return buttonPosition.artworkBtnAlignY; }, set group($$value) { buttonPosition.artworkBtnAlignY = $$value; }, children: ($$anchor3, $$slotProps2) => { var i_2 = root_18(); var node_18 = child(i_2); html(node_18, () => alignTop); reset(i_2); append($$anchor3, i_2); }, $$slots: { default: true } }); var node_19 = sibling(node_17, 2); RadioItem(node_19, { name: "artwork-align-y", get value() { return BtnAlignY.BOTTOM; }, class: " text-xs", get group() { return buttonPosition.artworkBtnAlignY; }, set group($$value) { buttonPosition.artworkBtnAlignY = $$value; }, children: ($$anchor3, $$slotProps2) => { var i_3 = root_19(); var node_20 = child(i_3); html(node_20, () => alignBottom); reset(i_3); append($$anchor3, i_3); }, $$slots: { default: true } }); append($$anchor2, fragment_7); }, $$slots: { default: true } }); reset(div_14); reset(li_2); reset(ul_2); reset(section_2); reset(div); template_effect(() => { set_class(div, clsx(sectionSpace())); set_class(div_2, `w-48 h-48 backdrop-blur-sm rounded-lg relative overflow-hidden ${bg() ?? ""}`); set_class(p, clsx(sectionTitle())); set_class(ul, clsx(get$1(ulClasses))); set_class(p_6, clsx(sectionTitle())); set_class(ul_2, clsx(get$1(ulClasses))); }); append($$anchor, div); pop(); } delegate(["click"]); let themeWatcher = new MediaQuery("(prefers-color-scheme: dark)"); let themeVersion = state(0); const clientSetting = createPersistedStore( "pdl-client-state", { theme: "auto", locale: locale.current, version: null, showPopupButton: legacyConfig.showPopupButton ?? true }, { get autoTheme() { get$1(themeVersion); return themeWatcher.current ? "dark" : "light"; }, setThemeWatcher(watcher) { set(themeVersion, get$1(themeVersion) + 1); themeWatcher = watcher; } } ); effect_root(() => { user_effect(() => setlocale(clientSetting.locale)); }); var root_1$3 = /* @__PURE__ */ template(`<li><p class="flex-auto"> </p> <!></li>`); var root_2 = /* @__PURE__ */ template(`<li><p class="flex-auto"> </p> <!></li> <li><div class="flex-auto"><p> </p> <p> </p></div> <!></li>`, 1); var root_5 = /* @__PURE__ */ template(`<li><label class="label flex flex-grow items-center justify-center"><p class="flex-auto"> </p> <!></label></li>`); var root_4 = /* @__PURE__ */ template(`<ul><li><label class="label flex flex-grow items-center justify-center"><p class="flex-auto"> </p> <!></label></li> <!></ul>`); var root_3$1 = /* @__PURE__ */ template(`<li class="flex-col !items-stretch"><div class="flex items-center"><div class="flex-auto"><p> </p> <p> </p></div> <!></div> <!></li>`); var root_6$1 = /* @__PURE__ */ template(`<section><p>实验性功能</p> <ul><li><div class="flex-auto"><p>为单页插图增加 #pixivGlow2024 效果</p> <p>* 转换至动图格式。如果插图尺寸过大,可能占用大量内存 / 转换失败</p></div> <!></li></ul></section>`); var root$2 = /* @__PURE__ */ template(`<div><ul><li><p class="flex-auto"> </p> <!></li> <!> <!> <!></ul> <!></div>`); function Others($$anchor, $$props) { push($$props, true); let bg = prop($$props, "bg", 3, "bg-white/30 dark:bg-black/15"), border = prop($$props, "border", 3, "divide-y-[1px] *:border-surface-300-600-token"), padding = prop($$props, "padding", 3, "px-4 *:py-4"), margin = prop($$props, "margin", 3, "mt-2 *:!m-0"), rounded = prop($$props, "rounded", 3, "rounded-container-token *:!rounded-none"), sectionSpace = prop($$props, "sectionSpace", 19, () => `space-y-4`), sectionTitle = prop($$props, "sectionTitle", 3, "font-bold"), descriptionText = prop($$props, "descriptionText", 3, "text-sm text-surface-400"), UlClass = prop($$props, "class", 3, ""); const ulClasses = /* @__PURE__ */ derived(() => `list *:items-center ${padding()} ${margin()} ${border()} ${bg()} ${rounded()} ${UlClass()}`); var div = root$2(); var ul = child(div); var li = child(ul); var p = child(li); var text2 = child(p, true); template_effect(() => set_text(text2, t("setting.others.options.show_setting_button"))); reset(p); var node = sibling(p, 2); SlideToggle(node, { name: "show-popup-button", size: "sm", get checked() { return clientSetting.showPopupButton; }, set checked($$value) { clientSetting.showPopupButton = $$value; } }); reset(li); var node_1 = sibling(li, 2); { var consequent = ($$anchor2) => { var li_1 = root_1$3(); var p_1 = child(li_1); var text_1 = child(p_1, true); template_effect(() => set_text(text_1, t("setting.others.options.bundle_multipage_illust"))); reset(p_1); var node_2 = sibling(p_1, 2); SlideToggle(node_2, { name: "bundle-illusts", size: "sm", get checked() { return siteFeature.compressMultiIllusts; }, set checked($$value) { siteFeature.compressMultiIllusts = $$value; } }); reset(li_1); append($$anchor2, li_1); }; if_block(node_1, ($$render) => { if (env.isPixiv() || env.isNijie()) $$render(consequent); }); } var node_3 = sibling(node_1, 2); { var consequent_1 = ($$anchor2) => { var fragment = root_2(); var li_2 = first_child(fragment); var p_2 = child(li_2); var text_2 = child(p_2, true); template_effect(() => set_text(text_2, t("setting.others.options.bundle_manga"))); reset(p_2); var node_4 = sibling(p_2, 2); SlideToggle(node_4, { name: "bundle-manga", size: "sm", get checked() { return siteFeature.compressManga; }, set checked($$value) { siteFeature.compressManga = $$value; } }); reset(li_2); var li_3 = sibling(li_2, 2); var div_1 = child(li_3); var p_3 = child(div_1); var text_3 = child(p_3, true); template_effect(() => set_text(text_3, t("setting.others.options.like_illust_when_downloading"))); reset(p_3); var p_4 = sibling(p_3, 2); var text_4 = child(p_4, true); template_effect(() => set_text(text_4, t("setting.others.options.option_does_not_apply_to_batch_download"))); reset(p_4); reset(div_1); var node_5 = sibling(div_1, 2); SlideToggle(node_5, { name: "like-illust", size: "sm", get checked() { return siteFeature.likeIllustWhenDownloading; }, set checked($$value) { siteFeature.likeIllustWhenDownloading = $$value; } }); reset(li_3); template_effect(() => set_class(p_4, clsx(descriptionText()))); append($$anchor2, fragment); }; if_block(node_3, ($$render) => { if (env.isPixiv()) $$render(consequent_1); }); } var node_6 = sibling(node_3, 2); { var consequent_4 = ($$anchor2) => { var li_4 = root_3$1(); var div_2 = child(li_4); var div_3 = child(div_2); var p_5 = child(div_3); var text_5 = child(p_5, true); template_effect(() => set_text(text_5, t("setting.others.options.add_bookmark_when_downloading"))); reset(p_5); var p_6 = sibling(p_5, 2); var text_6 = child(p_6, true); template_effect(() => set_text(text_6, t("setting.others.options.option_does_not_apply_to_batch_download"))); reset(p_6); reset(div_3); var node_7 = sibling(div_3, 2); SlideToggle(node_7, { name: "add-bookmark", size: "sm", get checked() { return siteFeature.addBookmark; }, set checked($$value) { siteFeature.addBookmark = $$value; } }); reset(div_2); var node_8 = sibling(div_2, 2); { var consequent_3 = ($$anchor3) => { var ul_1 = root_4(); var li_5 = child(ul_1); var label = child(li_5); var p_7 = child(label); var text_7 = child(p_7, true); template_effect(() => set_text(text_7, t("setting.others.options.add_bookmark_with_tags"))); reset(p_7); var node_9 = sibling(p_7, 2); SlideToggle(node_9, { name: "bookmark-with-tags", size: "sm", get checked() { return siteFeature.bookmarkWithTags; }, set checked($$value) { siteFeature.bookmarkWithTags = $$value; } }); reset(label); reset(li_5); var node_10 = sibling(li_5, 2); { var consequent_2 = ($$anchor4) => { var li_6 = root_5(); var label_1 = child(li_6); var p_8 = child(label_1); var text_8 = child(p_8, true); template_effect(() => set_text(text_8, t("setting.others.options.add_bookmark_private_r18"))); reset(p_8); var node_11 = sibling(p_8, 2); SlideToggle(node_11, { name: "private-bookmark-if-r18", size: "sm", get checked() { return siteFeature.privateBookmarkIfR18; }, set checked($$value) { siteFeature.privateBookmarkIfR18 = $$value; } }); reset(label_1); reset(li_6); append($$anchor4, li_6); }; if_block(node_10, ($$render) => { if (env.isPixiv()) $$render(consequent_2); }); } reset(ul_1); template_effect(() => set_class(ul_1, `list ${border() ?? ""} ${rounded() ?? ""} [&:not(:last-child)]:*:py-4 [&:last-child]:*:pt-4`)); append($$anchor3, ul_1); }; if_block(node_8, ($$render) => { if (siteFeature.addBookmark && (env.isPixiv() || env.isNijie())) $$render(consequent_3); }); } reset(li_4); template_effect(() => set_class(p_6, clsx(descriptionText()))); append($$anchor2, li_4); }; if_block(node_6, ($$render) => { if (siteFeature.addBookmark !== null) $$render(consequent_4); }); } reset(ul); var node_12 = sibling(ul, 2); { var consequent_5 = ($$anchor2) => { var section = root_6$1(); var p_9 = child(section); var ul_2 = sibling(p_9, 2); var li_7 = child(ul_2); var div_4 = child(li_7); var p_10 = sibling(child(div_4), 2); reset(div_4); var node_13 = sibling(div_4, 2); SlideToggle(node_13, { name: "mix-effect", size: "sm", get checked() { return siteFeature.mixSeasonalEffect; }, set checked($$value) { siteFeature.mixSeasonalEffect = $$value; } }); reset(li_7); reset(ul_2); reset(section); template_effect(() => { set_class(p_9, clsx(sectionTitle())); set_class(ul_2, clsx(get$1(ulClasses))); set_class(p_10, `${descriptionText() ?? ""} !text-error-500`); }); append($$anchor2, section); }; if_block(node_12, ($$render) => { if (env.isPixiv()) $$render(consequent_5); }); } reset(div); template_effect(() => { set_class(div, clsx(sectionSpace())); set_class(ul, clsx(get$1(ulClasses))); }); append($$anchor, div); pop(); } var root$1 = /* @__PURE__ */ template(`<div><section><p> </p> <ul><li><span><!></span></li></ul></section> <section><p> </p> <ul><li><p><!></p></li> <li class=" justify-center"><figure><img alt="credit" class=" rounded-full m-auto"> <figcaption class="mt-4"> </figcaption></figure></li></ul></section></div>`); function FeedBack($$anchor, $$props) { push($$props, true); let bg = prop($$props, "bg", 3, "bg-white/30 dark:bg-black/15"), border = prop($$props, "border", 3, "divide-y-[1px] *:border-surface-300-600-token"), padding = prop($$props, "padding", 3, "px-4 *:py-4"), margin = prop($$props, "margin", 3, "mt-2 *:!m-0"), rounded = prop($$props, "rounded", 3, "rounded-container-token *:!rounded-none"), sectionSpace = prop($$props, "sectionSpace", 19, () => `space-y-4`), sectionTitle = prop($$props, "sectionTitle", 3, "font-bold"), UlClass = prop($$props, "class", 3, ""); const ulClasses = /* @__PURE__ */ derived(() => `list *:items-center ${padding()} ${margin()} ${border()} ${bg()} ${rounded()} ${UlClass()}`); var div = root$1(); var section = child(div); var p = child(section); var text2 = child(p, true); template_effect(() => set_text(text2, t("setting.feedback.label.feedback"))); reset(p); var ul = sibling(p, 2); var li = child(ul); var span = child(li); var node = child(span); html(node, () => t("setting.feedback.text.feedback_desc")); reset(span); reset(li); reset(ul); reset(section); var section_1 = sibling(section, 2); var p_1 = child(section_1); var text_1 = child(p_1, true); template_effect(() => set_text(text_1, t("setting.feedback.label.donate"))); reset(p_1); var ul_1 = sibling(p_1, 2); var li_1 = child(ul_1); var p_2 = child(li_1); var node_1 = child(p_2); html(node_1, () => t("setting.feedback.text.give_me_a_star")); reset(p_2); reset(li_1); var li_2 = sibling(li_1, 2); var figure = child(li_2); var img = child(figure); set_attribute(img, "src", creditCode); var figcaption = sibling(img, 2); var text_2 = child(figcaption, true); template_effect(() => set_text(text_2, t("setting.feedback.text.donate_desc"))); reset(figcaption); reset(figure); reset(li_2); reset(ul_1); reset(section_1); reset(div); template_effect(() => { set_class(div, clsx(sectionSpace())); set_class(p, clsx(sectionTitle())); set_class(ul, clsx(get$1(ulClasses))); set_class(p_1, clsx(sectionTitle())); set_class(ul_1, clsx(get$1(ulClasses))); }); append($$anchor, div); pop(); } const userAuthentication = createPersistedStore("pdl-auth-state", { cf_clearance: ((_b = legacyConfig.auth) == null ? undefined : _b.cf_clearance) ?? null, username: ((_c = legacyConfig.auth) == null ? undefined : _c.username) ?? null, apiKey: ((_d = legacyConfig.auth) == null ? undefined : _d.apiKey) ?? null }); var root_1$2 = /* @__PURE__ */ template(`<section><p> </p> <div><div><input type="text" class="input"></div></div></section>`); var root = /* @__PURE__ */ template(`<div></div>`); function Auth($$anchor, $$props) { push($$props, true); let bg = prop($$props, "bg", 3, "bg-white/30 dark:bg-black/15"), border = prop($$props, "border", 3, "divide-y-[1px] *:border-surface-300-600-token"), padding = prop($$props, "padding", 3, "px-4 *:py-4"), margin = prop($$props, "margin", 3, "mt-2 *:!m-0"), rounded = prop($$props, "rounded", 3, "rounded-container-token *:!rounded-none"), sectionSpace = prop($$props, "sectionSpace", 19, () => `space-y-4`), sectionTitle = prop($$props, "sectionTitle", 3, "font-bold"), UlClass = prop($$props, "class", 3, ""); const blockClasses = /* @__PURE__ */ derived(() => `${padding()} ${margin()} ${border()} ${bg()} ${rounded()} ${UlClass()}`); const enabledAuth = Object.entries(userAuthentication).filter(([_, val]) => val !== null).map(([key, _]) => key); var div = root(); each(div, 20, () => enabledAuth, (key) => key, ($$anchor2, key) => { var section = root_1$2(); var p = child(section); var text2 = child(p, true); template_effect(() => set_text(text2, key.toUpperCase())); reset(p); var div_1 = sibling(p, 2); var div_2 = child(div_1); var input = child(div_2); remove_input_defaults(input); reset(div_2); reset(div_1); reset(section); template_effect(() => { set_class(p, clsx(sectionTitle())); set_class(div_1, clsx(get$1(blockClasses))); set_attribute(input, "name", key); }); bind_value(input, () => userAuthentication[key], ($$value) => userAuthentication[key] = $$value); append($$anchor2, section); }); reset(div); template_effect(() => set_class(div, clsx(sectionSpace()))); append($$anchor, div); pop(); } const menuOpen = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M21,15.61L19.59,17L14.58,12L19.59,7L21,8.39L17.44,12L21,15.61M3,6H16V8H3V6M3,13V11H13V13H3M3,18V16H16V18H3Z" /></svg>`; const menuClose = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M3 6H13V8H3V6M3 16H13V18H3V16M3 11H15V13H3V11M16 7L14.58 8.39L18.14 12L14.58 15.61L16 17L21 12L16 7Z" /></svg>`; var on_click = (_, showListbox) => set(showListbox, !get$1(showListbox)); var root_6 = /* @__PURE__ */ template(`<button type="button" class="btn-icon hover:variant-soft-surface ml-1"><i class="w-8 fill-current"><!></i></button>`); var root_9 = /* @__PURE__ */ template(`<h3 class="h3"> </h3>`); var root_1$1 = /* @__PURE__ */ template(`<div><!> <!> <div class="mt-4 pr-4 scrollbar-track-transparent scrollbar-thumb-slate-400/50 scrollbar-corner-transparent scrollbar-thin overflow-y-auto" style="scrollbar-gutter: stable"><!></div></div>`); function Config($$anchor, $$props) { push($$props, true); let slected = state(0); let showListbox = state(true); const optionList = [ { name: () => t("setting.save_to.title"), component: SaveTo }, { name: () => t("setting.ugoira.title"), component: UgoiraConvert, show: env.isPixiv() }, { name: () => t("setting.history.title"), component: DownloadHistory }, { name: () => t("setting.button_position.title"), component: BtnPosition }, { name: () => t("setting.others.title"), component: Others }, { name: () => t("setting.authorization.title"), component: Auth, show: Object.values(userAuthentication).some((val) => val !== null) }, { name: () => t("setting.feedback.title"), component: FeedBack } ]; const OptionComponent = /* @__PURE__ */ derived(() => optionList[get$1(slected)].component); const gridCol = /* @__PURE__ */ derived(() => get$1(showListbox) ? "grid-cols-[140px_1fr]" : "grid-cols-[0px_1fr]"); const transform = /* @__PURE__ */ derived(() => get$1(showListbox) ? "translate-x-0" : "-translate-x-full"); const sidebarWidth = "w-[140px]"; ModalWrapper($$anchor, { get parent() { return $$props.parent; }, height: "h-screen md:h-[600px]", width: "w-screen md:max-w-screen-md xl:max-w-screen-lg", padding: "", children: ($$anchor2, $$slotProps) => { var div = root_1$1(); var node = child(div); ListBox(node, { get class() { return `pt-4 pr-6 row-start-1 row-span-2 ${sidebarWidth} transition-[transform] ${get$1(transform) ?? ""}`; }, children: ($$anchor3, $$slotProps2) => { var fragment_1 = comment(); var node_1 = first_child(fragment_1); each(node_1, 17, () => optionList, index, ($$anchor4, option, idx) => { var fragment_2 = comment(); var node_2 = first_child(fragment_2); { var consequent = ($$anchor5) => { ListBoxItem($$anchor5, { name: "option", value: idx, class: "rounded-token", get group() { return get$1(slected); }, set group($$value) { set(slected, proxy($$value)); }, children: ($$anchor6, $$slotProps3) => { next(); var text$1 = text(); template_effect(() => set_text(text$1, get$1(option).name())); append($$anchor6, text$1); }, $$slots: { default: true } }); }; if_block(node_2, ($$render) => { if (!("show" in get$1(option)) || get$1(option).show) $$render(consequent); }); } append($$anchor4, fragment_2); }); append($$anchor3, fragment_1); }, $$slots: { default: true } }); var node_3 = sibling(node, 2); { const lead = ($$anchor3) => { var button2 = root_6(); button2.__click = [on_click, showListbox]; var i = child(button2); var node_4 = child(i); { var consequent_1 = ($$anchor4) => { var fragment_5 = comment(); var node_5 = first_child(fragment_5); html(node_5, () => menuClose); append($$anchor4, fragment_5); }; var alternate = ($$anchor4) => { var fragment_6 = comment(); var node_6 = first_child(fragment_6); html(node_6, () => menuOpen); append($$anchor4, fragment_6); }; if_block(node_4, ($$render) => { if (get$1(showListbox)) $$render(consequent_1); else $$render(alternate, false); }); } reset(i); reset(button2); append($$anchor3, button2); }; AppBar(node_3, { padding: "py-2", background: "bg-transparent", class: "mr-6 border-b border-surface-800-100-token", lead, children: ($$anchor3, $$slotProps2) => { var h3 = root_9(); var text_1 = child(h3, true); template_effect(() => set_text(text_1, optionList[get$1(slected)].name())); reset(h3); append($$anchor3, h3); }, $$slots: { lead: true, default: true } }); } var div_1 = sibling(node_3, 2); var node_7 = child(div_1); component(node_7, () => get$1(OptionComponent), ($$anchor3, $$component) => { $$component($$anchor3, { bg: "bg-white/30 dark:bg-surface-500/20 backdrop-blur-sm" }); }); reset(div_1); reset(div); template_effect(() => set_class(div, `h-full pt-4 pb-6 pl-6 grid grid-rows-[auto_1fr] transition-[grid-template-columns] ${get$1(gridCol) ?? ""}`)); append($$anchor2, div); }, $$slots: { default: true } }); pop(); } delegate(["click"]); var on_keydown = (e) => e.stopImmediatePropagation(); var root_3 = /* @__PURE__ */ template(`<button type="button" class="btn btn-sm variant-filled fixed bottom-24 rounded-none rounded-s-full opacity-40 hover:opacity-100 right-0 translate-x-[calc(100%-44px)] hover:translate-x-0 delay-100"><i class=" text-sm w-6 fill-current"><!></i> <span class="text-sm"> </span></button>`); var root_1 = /* @__PURE__ */ template(`<div data-theme="skeleton"><!> <!> <!> <!></div>`); function App($$anchor, $$props) { push($$props, true); let supportedTemplate = prop($$props, "supportedTemplate", 19, () => ({})); setContext("supportedTemplate", supportedTemplate()); initializeStores(); const modalStore = getModalStore(); const toastStore = getToastStore(); const dark = /* @__PURE__ */ derived(() => { if (clientSetting.theme === "auto") { return clientSetting.autoTheme === "dark"; } return clientSetting.theme === "dark"; }); let root2; const components = { changelog: { ref: Changelog }, setting: { ref: Config } }; function showChangelog() { modalStore.trigger({ type: "component", component: "changelog" }); } function showSetting() { modalStore.trigger({ type: "component", component: "setting" }); } function toast2(settings) { let background; switch (settings.type || "success") { case "success": background = "variant-filled-primary"; break; case "warning": background = "variant-filled-warning"; break; case "error": background = "variant-filled-error"; break; default: throw new Error("Unknown toast type."); } if (settings.type) delete settings.type; return toastStore.trigger({ ...settings, background }); } function preventBackDropClick(event2) { if (!(event2.target instanceof Element)) return; const classList = event2.target.classList; if (classList.contains("modal-backdrop") || classList.contains("modal-transition")) { event2.stopPropagation(); } } function modalExist() { return !!(root2 == null ? undefined : root2.querySelector(".modal-backdrop")); } function handleKeydown(event2) { if (!modalExist()) return; if (event2.code === "Escape") { modalStore.close(); return; } else if (event2.ctrlKey || event2.shiftKey) { return; } if (!event2.composedPath().includes(root2)) { event2.stopImmediatePropagation(); event2.preventDefault(); } } onMount(() => { const shadow = root2.getRootNode(); addStyleToShadow(shadow); shadow.host.setAttribute("style", "position:fixed; z-index:99999"); if (clientSetting.version !== "1.10.0") { clientSetting.version = "1.10.0"; showChangelog(); } globalThis.addEventListener(EVENT_DIR_HANDLE_NOT_FOUND, (evt) => { const customEvent = evt; const { getFileHandle: getFileHandle2, abort } = customEvent.detail; let actionIsActive = false; toast2({ message: t("toast.message.pick_directory_handle"), type: "warning", autohide: false, action: { label: t("toast.actionLabel.browse"), response: () => { actionIsActive = true; getFileHandle2(); } }, callback({ status }) { if (status === "closed" && !actionIsActive) { abort(); } } }); }); globalThis.addEventListener(EVENT_FILE_HANDLE_NOT_FOUND, (evt) => { const customEvent = evt; if ($$props.useBatchDownload) { const { downloading, hasTask } = $$props.useBatchDownload(); const { signal } = customEvent.detail; if (signal && downloading.current && hasTask(signal)) { return; } } const { getFileHandle: getFileHandle2, abort } = customEvent.detail; let actionIsActive = false; toast2({ message: t("toast.message.file_handle_not_found"), type: "warning", autohide: false, action: { label: t("toast.actionLabel.save"), response: () => { actionIsActive = true; getFileHandle2(); } }, callback({ status }) { if (status === "closed" && !actionIsActive) { abort(); } } }); }); globalThis.addEventListener(EVENT_REQUEST_USER_ACTIVATION, (evt) => { const customEvent = evt; const { onAction, onClosed } = customEvent.detail; let actionIsActive = false; toast2({ message: t("toast.message.request_directory_handle_permission"), type: "warning", autohide: false, action: { label: t("toast.actionLabel.allow_permission"), response: () => { actionIsActive = true; onAction(); } }, callback({ status }) { if (status === "closed" && !actionIsActive) { onClosed(); } } }); }); }); var div = root_1(); event("keydown", $window, handleKeydown, true); div.__keydown = [on_keydown]; var node = child(div); Modal(node, { components, class: "!p-0" }); var node_1 = sibling(node, 2); { var consequent = ($$anchor2) => { Downloader$1($$anchor2, { get downloaderConfig() { return $$props.downloaderConfig; }, get useBatchDownload() { return $$props.useBatchDownload; } }); }; if_block(node_1, ($$render) => { if ($$props.downloaderConfig && $$props.useBatchDownload) $$render(consequent); }); } var node_2 = sibling(node_1, 2); { var consequent_1 = ($$anchor2) => { var button2 = root_3(); button2.__click = showSetting; var i = child(button2); var node_3 = child(i); html(node_3, () => cog); reset(i); var span = sibling(i, 2); var text2 = child(span, true); template_effect(() => set_text(text2, t("button.setting"))); reset(span); reset(button2); append($$anchor2, button2); }; if_block(node_2, ($$render) => { if (clientSetting.showPopupButton) $$render(consequent_1); }); } var node_4 = sibling(node_2, 2); Toast(node_4, {}); reset(div); bind_this(div, ($$value) => root2 = $$value, () => root2); template_effect(() => set_class(div, clsx({ contents: true, dark: get$1(dark) }))); event("mousedown", div, preventBackDropClick, true); event("mouseup", div, preventBackDropClick, true); append($$anchor, div); return pop({ showChangelog, showSetting, toast: toast2 }); } delegate(["keydown", "click"]); function useChannel() { const TAB_ID = String(Math.random()); const queue = []; let downloading = false; let pending2 = false; let onFullfilled; let onRejected; channelEvent.on( "batchdownload.query", () => { downloading && channelEvent.emit( "batchdownload.set-pending" /* SET_PENDING */ ); } ); channelEvent.on( "batchdownload.set-pending", () => { !pending2 && (pending2 = true); } ); channelEvent.on( "batchdownload.set-idle", () => { pending2 && (pending2 = false); } ); channelEvent.on( "batchdownload.add-queue", (tabId) => { downloading && queue.push(tabId); } ); channelEvent.on( "batchdownload.remove-queue", (tabId) => { if (!downloading) return; const idx = queue.findIndex((id) => id === tabId); idx !== -1 && queue.splice(idx, 1); } ); channelEvent.on( "batchdownload.process-next", (tabIds) => { if (tabIds[0] !== TAB_ID) return; queue.push(...tabIds.slice(1)); pending2 = false; downloading = true; onFullfilled(); } ); window.addEventListener("unload", () => { if (pending2) { channelEvent.emit( "batchdownload.remove-queue", TAB_ID ); return; } if (downloading) { queue.length ? channelEvent.emit( "batchdownload.process-next", queue ) : channelEvent.emit( "batchdownload.set-idle" /* SET_IDLE */ ); } }); channelEvent.emit( "batchdownload.query" /* QUERY */ ); return { async requestDownload() { if (!pending2) { downloading = true; channelEvent.emit( "batchdownload.set-pending" /* SET_PENDING */ ); logger.info("channel post: SET_PENDING"); return; } const waitUntilIdle = new Promise((resolve, reject) => { onFullfilled = resolve; onRejected = reject; }); channelEvent.emit( "batchdownload.add-queue", TAB_ID ); logger.info("channel post: ADD_QUEUE", TAB_ID); return waitUntilIdle; }, cancelDownloadRequest(reason) { if (!pending2) return; channelEvent.emit( "batchdownload.remove-queue", TAB_ID ); logger.info("channel post: REMOVE_QUEUE", TAB_ID); onRejected(reason); }, processNextDownload() { if (!downloading) return; downloading = false; if (queue.length) { pending2 = true; channelEvent.emit( "batchdownload.process-next", queue ); queue.length = 0; logger.info("channel post: PROCESS_NEXT"); } else { channelEvent.emit( "batchdownload.set-idle" /* SET_IDLE */ ); logger.info("channel post: SET_IDLE"); } } }; } const STATUS_CODES_TOO_MANY_REQUEST = 429; const DOWNLOAD_RESUME_TIMEOUT = 3e4; const DEFAULT_CONCURRENCY = 5; function defineBatchDownload(downloaderConfig) { const { requestDownload, cancelDownloadRequest, processNextDownload } = useChannel(); let batchDownloaderState = null; let artworkCount = state(0); let successd = state(proxy([])); let failed = state(proxy([])); let excluded = state(proxy([])); let downloading = state(false); let log = state(undefined); const failedIdTasks = []; const failedMetaTasks = []; const unavaliableIdTasks = []; const unavaliableMetaTasks = []; const includeFilters = []; const excludeFilters = []; let controller; const taskControllers = /* @__PURE__ */ new Map(); const downloadQueue = new PQueue({ concurrency: DEFAULT_CONCURRENCY, interval: 1100, intervalCap: 1 }); const batchDownloadStore = { artworkCount: { get current() { return get$1(artworkCount); } }, successd: { get current() { return get$1(successd); } }, failed: { get current() { return get$1(failed); } }, excluded: { get current() { return get$1(excluded); } }, downloading: { get current() { return get$1(downloading); } }, log: { get current() { return get$1(log); } }, batchDownload, abort() { controller && controller.abort(new CancelError()); }, hasTask(signal) { return taskControllers.has(signal); }, getConcurrency() { return downloadQueue.concurrency; }, setConcurrency(num) { if (typeof num === "undefined") { downloadQueue.concurrency = DEFAULT_CONCURRENCY; return; } downloadQueue.concurrency = num; } }; function isStringArray(arr) { return typeof arr[0] === "string"; } function reset2() { set(artworkCount, 0); set(successd, proxy([])); set(failed, proxy([])); set(excluded, proxy([])); failedIdTasks.length = 0; unavaliableIdTasks.length = 0; failedMetaTasks.length = 0; unavaliableMetaTasks.length = 0; writeLog("Info", "Reset store."); } function setDownloading(state2) { if (get$1(downloading) && state2) { throw new Error("Already downloading."); } set(downloading, proxy(state2)); } function addSuccessd(idOrIdArr) { if (Array.isArray(idOrIdArr)) { get$1(successd).push(...idOrIdArr); writeLog("Complete", idOrIdArr[idOrIdArr.length - 1]); } else { get$1(successd).push(idOrIdArr); writeLog("Complete", idOrIdArr); } } function addFailed(item) { let failedItem; if (Array.isArray(item)) { get$1(failed).push(...item); failedItem = item[item.length - 1]; } else { get$1(failed).push(item); failedItem = item; } const { id, reason } = failedItem; if (reason instanceof Error || typeof reason === "string") { writeLog("Fail", id, reason); } } function addExcluded(idOrMeta) { if (Array.isArray(idOrMeta)) { isStringArray(idOrMeta) ? get$1(excluded).push(...idOrMeta) : get$1(excluded).push(...idOrMeta.map((meta) => meta.id)); writeLog("Info", `${idOrMeta.length + " task" + (idOrMeta.length > 1 ? "s were" : " was")} excluded...`); } else { const id = typeof idOrMeta === "string" ? idOrMeta : idOrMeta.id; get$1(excluded).push(id); writeLog("Info", `${id} was excluded...`); } } function writeLog(type, arg, error) { const item = { type, message: "" }; switch (type) { case "Error": if (!(arg instanceof Error)) throw new TypeError("error` is expected to be error, but got " + typeof arg); item.message = `[${arg.name}] ${arg.message}`; break; case "Fail": if (typeof arg !== "string") throw new TypeError("`id` is expected to be string, but got " + typeof arg); typeof error === "string" ? item.message = `[Fail] ${arg}...${error}` : item.message = `[Fail] ${arg}...${error ? error.name + ":" + error.message : ""}`; break; default: item.message = `[${type}] ${arg}`; break; } set(log, proxy(item)); } function filterTag(partialMeta, customTagFilter) { if (!("tags" in partialMeta) || !Array.isArray(partialMeta.tags)) return true; const { whitelistTag, blacklistTag } = batchDownloaderState; const defaultTagFilter = (userTags, metaTags) => userTags.some((tag) => metaTags.includes(tag)); customTagFilter ?? (customTagFilter = defaultTagFilter); if (whitelistTag.length) { return customTagFilter(whitelistTag, partialMeta.tags); } if (blacklistTag.length) { return !customTagFilter(blacklistTag, partialMeta.tags); } return true; } async function checkValidity(partialMeta) { try { const { enableTagFilter } = downloaderConfig.filterOption; if (enableTagFilter === true) { if (!filterTag(partialMeta)) return false; } else if (enableTagFilter) { if (!filterTag(partialMeta, enableTagFilter)) return false; } if (!includeFilters.length) return false; for (let i = 0; i < excludeFilters.length; i++) { const fn = excludeFilters[i]; const isExcluded = await fn(partialMeta); if (isExcluded) return false; } for (let i = 0; i < includeFilters.length; i++) { const fn = includeFilters[i]; const isValid = await fn(partialMeta); if (isValid) return true; } } catch (error) { console.error(error); } return false; } async function batchDownload(fnId, ...restArgs) { setDownloading(true); writeLog("Info", "Download start..."); downloadQueue.start(); batchDownloaderState = snapshot(batchDownloaderStore.$state); const selectedFilters = batchDownloaderState.selectedFilters ?? []; selectedFilters.forEach((id) => { const filter = downloaderConfig.filterOption.filters.find((filter2) => filter2.id === id); if (filter) { if (filter.type === "include") { includeFilters.push(filter.fn); } else { excludeFilters.push(filter.fn); } } }); reset2(); const { beforeDownload, afterDownload } = downloaderConfig; let generatorAfterDownloadCb = undefined; let generator; controller = new AbortController(); const signal = controller.signal; signal.addEventListener( "abort", () => { var _a2; cancelDownloadRequest(signal.reason); (_a2 = downloaderConfig.onDownloadAbort) == null ? undefined : _a2.call(downloaderConfig); }, { once: true } ); try { const pageIdItem = getGenPageIdItem(fnId); if (!pageIdItem || !("fn" in pageIdItem)) throw new Error("Invalid generator id: " + fnId); const { filterInGenerator, beforeDownload: beforeDownloadInGenerator, afterDownload: afterDownload2 } = pageIdItem; generatorAfterDownloadCb = afterDownload2; typeof beforeDownload === "function" && await beforeDownload(); typeof beforeDownloadInGenerator === "function" && await beforeDownloadInGenerator(); generator = getGenerator(pageIdItem, ...restArgs); writeLog("Info", "Waiting for other downloads to finish..."); await requestDownload(); writeLog("Info", "Starting..."); await dispatchDownload(generator, filterInGenerator, signal); if (batchDownloaderState.retryFailed && (failedIdTasks.length || failedMetaTasks.length)) { if (failedIdTasks.length) { generator = getIdRetryGenerator(get$1(artworkCount), failedIdTasks.slice(), unavaliableIdTasks.slice()); failedIdTasks.length = 0; unavaliableIdTasks.length = 0; } else if (failedMetaTasks.length) { generator = getMetaRetryGenerator(get$1(artworkCount), failedMetaTasks.slice(), unavaliableMetaTasks.slice()); failedMetaTasks.length = 0; unavaliableMetaTasks.length = 0; } set(failed, proxy([])); writeLog("Info", "Retry..."); await dispatchDownload(generator, filterInGenerator, signal); } writeLog("Info", "Download complete."); } catch (error) { generator == null ? undefined : generator.return(); if (!signal.aborted) { controller.abort(error); } if (error instanceof Error) { writeLog("Error", error); } throw error; } finally { typeof generatorAfterDownloadCb === "function" && generatorAfterDownloadCb(); typeof afterDownload === "function" && afterDownload(); controller = null; batchDownloaderState = null; includeFilters.length = 0; excludeFilters.length = 0; downloadQueue.pause(); setDownloading(false); processNextDownload(); } } function getGenPageIdItem(fnId) { const { pageOption } = downloaderConfig; for (const key in pageOption) { if (key === fnId) { return pageOption[key]; } } } function getGenerator(item, ...restArgs) { let generator; const { downloadAllPages, pageEnd, pageStart } = batchDownloaderState; if (!downloadAllPages && pageEnd < pageStart) throw new Error("End page must not be less than the start page."); const pageRange = downloadAllPages ? null : [pageStart, pageEnd]; if ("filterInGenerator" in item && !item.filterInGenerator) { generator = item.fn(pageRange, ...restArgs); } else { generator = item.fn(pageRange, checkValidity, ...restArgs); } return generator; } function* getIdRetryGenerator(total, failedArtworks, unavaliableTasks) { yield { total, page: 0, avaliable: failedArtworks, invalid: [], unavaliable: unavaliableTasks }; } function* getMetaRetryGenerator(total, failedArtworks, unavaliableTasks) { yield { total, page: 0, avaliable: failedArtworks, invalid: [], unavaliable: unavaliableTasks }; } async function dispatchDownload(generator, filterInGenerator, batchDownloadSignal) { batchDownloadSignal.throwIfAborted(); const waitUntilDownloadReject = new Promise((_, reject) => { batchDownloadSignal.addEventListener( "abort", () => { reject(batchDownloadSignal.reason); downloadQueue.size !== 0 && downloadQueue.clear(); taskControllers.forEach((controller2) => controller2.abort(batchDownloadSignal.reason)); taskControllers.clear(); }, { once: true } ); }); const { parseMetaByArtworkId, downloadArtworkByMeta } = downloaderConfig; let result; while ((result = await Promise.race([generator.next(), waitUntilDownloadReject])) && !result.done) { batchDownloadSignal.throwIfAborted(); const { total, avaliable, invalid, unavaliable } = result.value; logger.info(total, avaliable, invalid, unavaliable); if (total !== get$1(artworkCount)) { set(artworkCount, proxy(total)); } if (invalid.length) { addExcluded(invalid); } if (unavaliable.length) { if (isStringArray(unavaliable)) { addFailed(unavaliable.map((id) => ({ id, reason: new InvalidPostError(id) }))); unavaliableIdTasks.push(...unavaliable); } else { addFailed(unavaliable.map((meta) => ({ id: meta.id, reason: new InvalidPostError(meta.id) }))); unavaliableMetaTasks.push(...unavaliable); } } if (!avaliable.length) { await Promise.race([sleep(1500), waitUntilDownloadReject]); continue; } for (const idOrMeta of avaliable) { const taskController = new AbortController(); const taskSingal = taskController.signal; taskControllers.set(taskSingal, taskController); downloadQueue.add( async ({ signal: taskSingal2 }) => { if (!taskSingal2) throw new Error("Expect `QueueAddOptions.signal` to be a AbortSignal but got undefined."); const isId = typeof idOrMeta === "string"; const metaId = isId ? idOrMeta : idOrMeta.id; try { taskSingal2.throwIfAborted(); let artworkMeta; if (!isId) { artworkMeta = idOrMeta; } else { artworkMeta = await parseMetaByArtworkId(metaId); taskSingal2.throwIfAborted(); if (!filterInGenerator && !await checkValidity(artworkMeta)) { addExcluded(metaId); return; } } writeLog("Add", metaId); await downloadArtworkByMeta(artworkMeta, taskSingal2); !taskSingal2.aborted && addSuccessd(metaId); } catch (error) { if (taskSingal2.aborted) return; addFailed({ id: metaId, reason: error }); if (error instanceof PermissionError) { controller == null ? undefined : controller.abort(error); return; } const isInvalid = error instanceof InvalidPostError; if (isId) { (isInvalid ? unavaliableIdTasks : failedIdTasks).push(idOrMeta); } else { (isInvalid ? unavaliableMetaTasks : failedMetaTasks).push(idOrMeta); } if (error instanceof RequestError && error.status === STATUS_CODES_TOO_MANY_REQUEST && !downloadQueue.isPaused) { downloadQueue.pause(); setTimeout( () => { !batchDownloadSignal.aborted && downloadQueue.start(); }, DOWNLOAD_RESUME_TIMEOUT ); writeLog("Error", new Error("Http status: 429, wait for 30 seconds.")); } logger.error(error); } }, { signal: taskSingal } ).catch(logger.warn).finally(() => { taskControllers.delete(taskSingal); }); } await Promise.race([ waitUntilDownloadReject, downloadQueue.onEmpty() ]); } return Promise.race([ waitUntilDownloadReject, downloadQueue.onIdle() ]); } return function batchDownloadDefinition() { return batchDownloadStore; }; } const PdlApp = create_custom_element( App, { supportedTemplate: {}, downloaderConfig: {}, useBatchDownload: {} }, [], ["showChangelog", "showSetting", "toast"], true, (customElementConstructor) => { return class extends customElementConstructor { constructor(props) { super(); //@ts-expect-error no_unsed_var __publicField(this, "supportedTemplate"); //@ts-expect-error no_unsed_var __publicField(this, "downloaderConfig"); //@ts-expect-error no_unsed_var __publicField(this, "useBatchDownload"); this.supportedTemplate = props.supportedTemplate; } initBatchDownloader(config) { this.downloaderConfig = config; return this.useBatchDownload = defineBatchDownload(config); } }; } ); customElements.define("pdl-app", PdlApp); class SiteInject { constructor() { __publicField(this, "app"); __publicField(this, "useBatchDownload"); this.app = this.createApp(); this.backupIfNeeded(); } static get hostname() { throw new Error("`hostname` should be overwritten by a subclass."); } createApp() { return new PdlApp({ supportedTemplate: this.getSupportedTemplate() }); } backupIfNeeded() { const { interval, lastTimestamp } = backupSetting; if (interval === BackupInterval.NEVER) return; const timestamp = (/* @__PURE__ */ new Date()).getTime(); if (lastTimestamp + interval > timestamp) return; useHistoryBackup().exportAsJSON(); backupSetting.lastTimestamp = timestamp; } toast(settings) { return this.app.toast(settings); } inject() { _GM_registerMenuCommand( t("button.setting"), () => { var _a2; if ((_a2 = this.app.shadowRoot) == null ? undefined : _a2.querySelector(".modal")) { return; } this.app.showSetting(); }, "s" ); document.body.append(this.app); } } var PostValidState = /* @__PURE__ */ ((PostValidState2) => { PostValidState2[PostValidState2["VALID"] = 0] = "VALID"; PostValidState2[PostValidState2["INVALID"] = 1] = "INVALID"; PostValidState2[PostValidState2["UNAVAILABLE"] = 2] = "UNAVAILABLE"; return PostValidState2; })(PostValidState || {}); class ParserBase { constructor() { __publicField(this, "UNKNOWN_ARTIST", "UnknownArtist"); __publicField(this, "UNKNOWN_CHARACTER", "UnknownCharacter"); } async *paginationGenerator(pageRange, getPostData, buildMeta, isValid) { const [pageStart = 1, pageEnd = 0] = pageRange ?? []; let page = pageStart; let { lastPage, data: postDatas } = await getPostData(page); if (!postDatas || !postDatas.length) throw new Error(`There is no post in page ${page}.`); let total = postDatas.length; let fetchError = null; do { let nextPageData = undefined; let nextPageIsLast = true; if (page !== pageEnd && !lastPage) { try { const { lastPage: lastPage2, data } = await getPostData(page + 1); const dataLen = (data == null ? void 0 : data.length) ?? 0; if (dataLen) { total += dataLen; nextPageData = data; nextPageIsLast = lastPage2; } } catch (error) { fetchError = error; } } const avaliable = []; const invalid = []; const unavaliable = []; if (typeof isValid === "function") { for (const data of postDatas) { const isPostValid = await isValid(data); const idOrMeta = buildMeta(data); if (isPostValid === 0) { avaliable.push(idOrMeta); } else if (isPostValid === 1) { invalid.push(idOrMeta); } else { unavaliable.push(idOrMeta); } } } else { for (const data of postDatas) { avaliable.push(buildMeta(data)); } } yield { total, page, avaliable, invalid, unavaliable }; page++; postDatas = nextPageData; lastPage = nextPageIsLast; } while (postDatas); if (fetchError) throw fetchError; } } class GelbooruParserV020 extends ParserBase { parseArtworkSrc(doc) { return doc.querySelector('meta[property="og:image"]').getAttribute("content"); } parseArtworkNameBySrc(src) { const imageNameMatch = new RegExp("(?<=\\/)\\w+\\.\\w+(?=\\?|$)").exec(src); if (!imageNameMatch) throw new Error("Can not parse image name from src."); const imageName = imageNameMatch[0]; return imageName.split("."); } parseTags(doc) { const artist = []; const character = []; const tags = []; const tagEls = doc.querySelectorAll('li[class*="tag-type"]'); tagEls.forEach((tagEl) => { var _a2; const tagTypeMatch = new RegExp("(?<=tag-type-)\\w+").exec(tagEl.className); if (!tagTypeMatch) throw new Error("Unknown tag: " + tagEl.className); const tagType = tagTypeMatch[0]; const tag = (((_a2 = tagEl.querySelector('a[href*="page=post"]')) == null ? undefined : _a2.textContent) || "").replaceAll(" ", "_"); if (tagType === "artist") { artist.push(tag); } else if (tagType === "character") { character.push(tag); } tags.push(tagType + ":" + tag); }); return { artist, character, tags }; } parseStatistics(doc) { var _a2, _b2, _c2, _d2, _e, _f; const uploaderEl = doc.querySelector('a[href*="page=account&s=profile"]'); const postDateStr = (_b2 = (_a2 = uploaderEl == null ? undefined : uploaderEl.parentElement) == null ? undefined : _a2.firstChild) == null ? undefined : _b2.nodeValue; const postDate = postDateStr ? postDateStr.split(": ")[1] : ""; let source2 = ""; const sourceEl = (_d2 = (_c2 = uploaderEl == null ? undefined : uploaderEl.parentElement) == null ? undefined : _c2.nextElementSibling) == null ? undefined : _d2.nextElementSibling; if (sourceEl && /^source:/i.test(sourceEl.textContent ?? "")) { const sourceLink = sourceEl.querySelector("a"); if (sourceLink) { source2 = sourceLink.href; } else { source2 = ((_e = sourceEl.textContent) == null ? undefined : _e.replace(/^source: ?/i, "")) ?? ""; } } const rating = /Rating: ?(General|Explicit|Questionable|Safe|Sensitive)/.exec(doc.documentElement.innerHTML)[1].toLowerCase(); const score = (_f = doc.querySelector("span[id^=psc]")) == null ? undefined : _f.textContent; if (!score) throw new Error("Cannot parse score"); return { postDate, score: +score, source: source2, rating }; } buildMeta(id, doc) { const src = this.parseArtworkSrc(doc); const [title, extendName] = this.parseArtworkNameBySrc(src); const { artist, character, tags } = this.parseTags(doc); const { postDate, source: source2, rating, score } = this.parseStatistics(doc); return { id, src, extendName, artist: artist.join(",") || this.UNKNOWN_ARTIST, character: character.join(",") || this.UNKNOWN_CHARACTER, title, tags, createDate: postDate, score, source: source2, rating }; } parseBlacklistTags() { const [tagsStr] = new RegExp("(?<=tag_blacklist=).*?(?=;|$)").exec(document.cookie) ?? []; if (!tagsStr) return []; const tags = decodeURIComponent(decodeURIComponent(tagsStr)); return tags.split(" "); } parseFavoriteByDoc(doc) { const favDataScripts = doc.querySelectorAll("span + script"); const favData = Array.from(favDataScripts).map((el) => { const content = el.textContent; const [id] = new RegExp("(?<=posts\\[)[0-9]+?(?=\\])").exec(content); const [dataStr] = content.match(/{.+}/); const { tags, rating, score, user } = JSON.parse( dataStr.replace(".split(/ /g)", "").replaceAll("'", '"') ); return { id, tags: decodeURIComponent(tags).split(/\s/g), rating: rating.toLowerCase(), score: +score, user }; }); return favData; } parsePostsByDoc(doc) { const imageItems = Array.from( doc.querySelectorAll( "span[id]:has(a > img), .thumbnail-preview > a[id]:has(img)" ) ); const postData = imageItems.map((el) => { const image = el.querySelector("img"); const fullTags = image.title.trim().replaceAll(/ +/g, " ").split(" "); const id = el.id.slice(1); const tags = []; let rating = ""; let score = 0; let user = ""; for (let i = 0; i < fullTags.length; i++) { const tag = fullTags[i]; if (tag.startsWith("rating:")) { rating = tag.slice(7); } else if (tag.startsWith("score:")) { score = +tag.slice(6); } else if (tag.startsWith("user:")) { user = tag.slice(5); } else { tags.push(tag); } } return { id, tags, rating, score: +score, user }; }); logger.info(`Parse posts in ${doc.URL}:`, postData); return postData; } } function createCompressor() { const zip = new JSZip(); return { add(id, name, data) { var _a2; (_a2 = zip.folder(id)) == null ? undefined : _a2.file(name, data); }, bundle(id, comment2) { const folder = zip.folder(id); if (!folder) throw new TypeError("no such folder:" + id); return folder.generateAsync({ type: "blob", comment: comment2 }); }, remove(ids) { zip.remove(ids); }, fileCount(id) { var _a2; let count = 0; (_a2 = zip.folder(id)) == null ? undefined : _a2.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(); var SupportedTemplate = /* @__PURE__ */ ((SupportedTemplate2) => { SupportedTemplate2["ID"] = "id"; SupportedTemplate2["ARTIST"] = "artist"; SupportedTemplate2["ARTISTID"] = "artistID"; SupportedTemplate2["CHARACTER"] = "character"; SupportedTemplate2["DATE"] = "date"; SupportedTemplate2["MD5"] = "md5"; SupportedTemplate2["SCORE"] = "score"; SupportedTemplate2["TAGS"] = "tags"; SupportedTemplate2["TITLE"] = "title"; SupportedTemplate2["PAGE"] = "page"; return SupportedTemplate2; })(SupportedTemplate || {}); class MediaDownloadConfig { constructor(mediaMeta) { __privateAdd(this, _MediaDownloadConfig_instances); __publicField(this, "id"); __publicField(this, "src"); __publicField(this, "ext"); __publicField(this, "artist"); __publicField(this, "title"); __publicField(this, "tags"); __publicField(this, "createDate"); __publicField(this, "imageTimeout", 6e4); __publicField(this, "videoTimeout"); __publicField(this, "taskId"); __publicField(this, "downloaded"); __publicField(this, "total"); __publicField(this, "onDownloadCompleted"); const { id, src, extendName, artist, title, tags, createDate } = mediaMeta; this.id = id; this.src = src; this.ext = extendName; this.artist = artist; this.title = title; this.tags = tags; this.createDate = createDate; this.total = Array.isArray(src) ? src.length : 1; this.downloaded = 0; } static get supportedTemplate() { throw new Error("Should be overwritten by a subclass"); } normalizeString(str) { return replaceInvalidChar(unescapeHtml(str)); } isStringArray(val) { return Array.isArray(val); } isImageExt(ext) { return regexp.imageExt.test(ext); } getTaskId() { return this.taskId ?? (this.taskId = this.id + "_" + Math.random().toString(36).slice(2)); } getSrc(idx = 0) { return Array.isArray(this.src) ? this.src[idx] : this.src; } getExt(idx = 0) { return Array.isArray(this.ext) ? this.ext[idx] : this.ext; } getDownloadTimeout(idx = 0) { return this.isImageExt(Array.isArray(this.ext) ? this.ext[idx] : this.ext) ? this.imageTimeout : this.videoTimeout; } getMultipleMediaDownloadCB(setProgress) { return this.onDownloadCompleted ?? (this.onDownloadCompleted = () => { setProgress(++this.downloaded / this.total * 100); }); } getPathTemplate(folderTemplate, filenameTemplate) { return folderTemplate ? `${folderTemplate}/${filenameTemplate}` : filenameTemplate; } getSavePath(folderTemplate, filenameTemplate, ext, templateData) { const path = __privateMethod(this, _MediaDownloadConfig_instances, replaceTemplate_fn).call(this, this.getPathTemplate(folderTemplate, filenameTemplate), templateData); return `${path}.${ext}`; } } _MediaDownloadConfig_instances = new WeakSet(); replaceTemplate_fn = function(template2, data) { const re = new RegExp( `{(${"artist"}|${"artistID"}|${"character"}|${"id"}|${"md5"}|${"page"}|${"score"}|${"tags"}|${"title"})}`, "g" ); const path = template2.replace( re, (match, templateName) => { if (!(templateName in data)) return match; const val = data[templateName]; return !val ? match : this.normalizeString(val); } ); const dateRe = new RegExp(`{(${"date"})(\\((.+?)\\))?}`, "g"); return path.replace( dateRe, (match, _templateName, _formatMatch, formatValue) => { if (!data.date) return match; const format = formatValue || "YYYY-MM-DD"; const date = data.date; return dayjs(date).format(format); } ); }; class MayBeMultiIllustsConfig extends MediaDownloadConfig { constructor() { super(...arguments); __publicField(this, "handleBeforeSaveCb"); __publicField(this, "handleErrorCb"); __publicField(this, "handleAbortCb"); __publicField(this, "filenames"); } handleBundleFactory(filenames) { this.filenames ?? (this.filenames = filenames); return this.handleBeforeSaveCb ?? (this.handleBeforeSaveCb = async (imgBlob, config, signal) => { signal == null ? undefined : signal.throwIfAborted(); const { taskId, src } = config; const index2 = this.src.indexOf(src); if (index2 === -1) throw new Error("No src matches."); compressor.add(taskId, this.filenames[index2], imgBlob); if (compressor.fileCount(taskId) !== filenames.length) return; const zipData = await compressor.bundle(taskId, this.getZipComment()); signal == null ? undefined : signal.throwIfAborted(); compressor.remove(taskId); return zipData; }); } handleBundleErrorFactory() { return this.handleErrorCb ?? (this.handleErrorCb = () => { compressor.remove(this.getTaskId()); }); } handleBundleAbortFactory() { return this.handleAbortCb ?? (this.handleAbortCb = () => { compressor.remove(this.getTaskId()); }); } } class BooruDownloadConfig extends MediaDownloadConfig { constructor(meta) { super(meta); __publicField(this, "character"); __publicField(this, "score"); this.character = meta.character; this.score = meta.score; } static get supportedTemplate() { return { [ "id" /* ID */ ]: "{id}", [ "artist" /* ARTIST */ ]: "{artist}", [ "character" /* CHARACTER */ ]: "{character}", [ "date" /* DATE */ ]: "{date}, {date(YYYY-MM-DD)}", [ "md5" /* MD5 */ ]: "{md5}", [ "score" /* SCORE */ ]: "{score}" }; } getHeaders(cfClearance) { return { cookie: `cf_clearance=${cfClearance}` }; } getTemplateData() { return { id: this.id, artist: this.artist, character: this.character, date: this.createDate, md5: this.title, title: this.title, score: String(this.score) }; } create(option) { const { filenameTemplate, filenameConflictAction, directoryTemplate, useFileSystemAccessApi, setProgress, cfClearance } = option; return { headers: cfClearance ? this.getHeaders(cfClearance) : undefined, taskId: this.getTaskId(), src: this.getSrc(), path: this.getSavePath( directoryTemplate, filenameTemplate, this.getExt(), this.getTemplateData() ), timeout: this.getDownloadTimeout(), onProgress: setProgress, useFileSystemAccessApi, filenameConflictAction }; } } class GelbooruV020 extends SiteInject { constructor() { if (clientSetting.version === null) { siteFeature.addBookmark ?? (siteFeature.addBookmark = false); } super(); __privateAdd(this, _GelbooruV020_instances); __publicField(this, "searchParams", new URLSearchParams(location.search)); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: this.getAvatar.bind(this), parseMetaByArtworkId: async (id) => { const doc = await this.api.getPostDoc(id); return this.parser.buildMeta(id, doc); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfigs = new BooruDownloadConfig(meta).create({ ...downloadSetting, cfClearance: userAuthentication.cf_clearance || undefined }); await downloader.download(downloadConfigs, { signal }); const { id, tags, artist, title, source: source2, rating } = meta; historyDb.add({ pid: Number(id), user: artist, title, tags, source: source2, rating }); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn(meta) { return !!meta.tags && !meta.tags.includes("video"); } }, // Safebooru doesn't really have videos, although it has about 73 GIFs tagged with 'video' { id: "allow_video", type: "include", name: () => t("downloader.category.filter.video"), checked: true, fn(meta) { return !!meta.tags && meta.tags.includes("video"); } } ], enableTagFilter: true }, pageOption: { favorites: { name: "Favorites", match: /(?=.*page=favorites)(?=.*s=view)(?=.*id=[0-9]+)/, filterInGenerator: true, fn: (pageRange, checkValidity, userId) => { userId ?? (userId = new RegExp("(?<=id=)[0-9]+").exec(location.search)[0]); const getFavoriteByPage = async (page) => { const THUMBS_PER_PAGE = 50; const pid = (page - 1) * THUMBS_PER_PAGE; const doc = await this.api.getFavoriteDoc(userId, pid); const data = this.parser.parseFavoriteByDoc(doc); return { lastPage: data.length < THUMBS_PER_PAGE, data }; }; return this.parser.paginationGenerator( pageRange, getFavoriteByPage, (post) => post.id, __privateMethod(this, _GelbooruV020_instances, validityCheckFactory_fn).call(this, checkValidity) ); } }, pools: { name: "Pools", match: /(?=.*page=pool)(?=.*s=show)(?=.*id=[0-9]+)/, filterInGenerator: true, fn: (_, checkValidity, poolId) => { poolId ?? (poolId = new RegExp("(?<=id=)[0-9]+").exec(location.search)[0]); const getPoolData = async () => { const doc = await this.api.getPoolDoc(poolId); return { lastPage: true, data: this.parser.parsePostsByDoc(doc) }; }; return this.parser.paginationGenerator( [1, 1], getPoolData, (post) => post.id, __privateMethod(this, _GelbooruV020_instances, validityCheckFactory_fn).call(this, checkValidity) ); } }, posts: { name: "Posts", match: /(?=.*page=post)(?=.*s=list)/, filterInGenerator: true, fn: (pageRange, checkValidity, tags) => { tags ?? (tags = new URLSearchParams(location.search).get("tags") ?? "all"); const getPostsByPage = async (page) => { const THUMBS_PER_PAGE = 42; const pid = (page - 1) * THUMBS_PER_PAGE; const doc = await this.api.getPostsDoc(pid, tags); const data = this.parser.parsePostsByDoc(doc); return { lastPage: data.length < THUMBS_PER_PAGE, data }; }; return this.parser.paginationGenerator( pageRange, getPostsByPage, (post) => post.id, __privateMethod(this, _GelbooruV020_instances, validityCheckFactory_fn).call(this, checkValidity) ); } } } })); } getSupportedTemplate() { return BooruDownloadConfig.supportedTemplate; } getAvatar() { return "/favicon.ico"; } async downloadArtwork(btn2) { const id = btn2.dataset.id; const doc = await this.api.getPostDoc(id); const mediaMeta = this.parser.buildMeta(id, doc); const downloadConfig = new BooruDownloadConfig(mediaMeta).create({ ...downloadSetting, cfClearance: userAuthentication.cf_clearance || undefined, setProgress: (progress) => { btn2.setProgress(progress); } }); siteFeature.addBookmark && __privateMethod(this, _GelbooruV020_instances, addBookmark_fn).call(this, id); await downloader.download(downloadConfig, { priority: 1 }); const { tags, artist, title, source: source2, rating } = mediaMeta; historyDb.add({ pid: Number(id), user: artist, title, tags, source: source2, rating }); } setThumbnailStyle(btnContainer) { btnContainer.style.position = "relative"; } createThumbnailBtn() { const btnContainers = document.querySelectorAll(this.getThumbnailSelector()); if (!btnContainers.length) return; btnContainers.forEach((el) => { this.setThumbnailStyle(el); const idMathch = new RegExp("(?<=&id=)\\d+").exec(el.href); if (!idMathch) return; const id = idMathch[0]; el.appendChild( new ThumbnailButton({ id, onClick: this.downloadArtwork.bind(this) }) ); }); } createArtworkBtn(id) { const btnContainer = document.querySelector("div.flexi > div"); btnContainer.style.position = "relative"; btnContainer.appendChild( new ArtworkButton({ id, site: "gelbooru", onClick: this.downloadArtwork.bind(this) }) ); } isPostsList() { return this.searchParams.get("page") === "post" && this.searchParams.get("s") === "list"; } isPostView() { return this.searchParams.get("page") === "post" && this.searchParams.get("s") === "view"; } isPool() { return this.searchParams.get("page") === "pool" && this.searchParams.get("s") === "show"; } isMyfavorites() { return this.searchParams.get("page") === "favorites" && this.searchParams.get("s") === "view"; } isAccountProfile() { return this.searchParams.get("page") === "account" && this.searchParams.get("s") === "profile"; } inject() { super.inject(); if (this.searchParams.size === 0) return; if (this.isPostView()) { if (!document.querySelector("#image, #gelcomVideoPlayer")) return; const id = this.searchParams.get("id"); this.createArtworkBtn(id); } else { this.createThumbnailBtn(); } } } _GelbooruV020_instances = new WeakSet(); validityCheckFactory_fn = function(checkValidity) { return async (post) => { const { id, tags } = post; return await checkValidity({ id, tags }) ? PostValidState.VALID : PostValidState.INVALID; }; }; addBookmark_fn = function(id) { _unsafeWindow.addFav(id); }; class ApiBase { constructor(option = { rateLimit: 4 }) { __publicField(this, "queue"); __publicField(this, "fetch"); const { rateLimit } = option; this.queue = new PQueue({ interval: 1e3, intervalCap: rateLimit }); this.fetch = (input, init2) => { return this.queue.add(() => fetch(input, init2), { throwOnTimeout: true }); }; } async getHtml(url) { logger.info("Fetch url:", url); const res = await this.fetch(url); if (!res.ok) throw new RequestError(res.url, res.status); return await res.text(); } async getDoc(url) { const html2 = await this.getHtml(url); return new DOMParser().parseFromString(html2, "text/html"); } async getJSON(url, init2) { logger.info("Fetch url:", url); const res = await this.fetch(url, init2); if (!res.ok) throw new RequestError(res.url, res.status); return await res.json(); } } class GelbooruApiV020 extends ApiBase { async getPosts(params) { let url = "/index.php?page=dapi&s=post&q=index&json=1"; Object.entries(params).forEach(([key, val]) => { if (typeof val === "number") { val = String(val); } else if (Array.isArray(val)) { val = val.join("+"); } url += `&${key}=${val}`; }); const res = await this.fetch(url); if (!res.ok) throw new RequestError(url, res.status); try { return await res.json(); } catch (error) { if (error instanceof SyntaxError) { return []; } else { throw error; } } } getPostDoc(id) { return this.getDoc("index.php?page=post&s=view&id=" + id); } // Does not include blacklisted images(unless you search the blacklisted tag), making it // more suitable for batch downloads that need to handle page ranges. getPostsDoc(pid = 0, tags = "all") { if (Array.isArray(tags)) tags = tags.join("+"); return this.getDoc(`/index.php?page=post&s=list&tags=${tags}&pid=${pid}`); } getFavoriteDoc(userId, pid = 0) { return this.getDoc(`/index.php?page=favorites&s=view&id=${userId}&pid=${pid}`); } getPoolDoc(poolId) { return this.getDoc(`/index.php?page=pool&s=show&id=${poolId}`); } } class Rule34Parser extends GelbooruParserV020 { parseArtworkSrc(doc) { var _a2; return ((_a2 = doc.querySelector("#gelcomVideoPlayer > source")) == null ? undefined : _a2.src) || doc.querySelector('meta[property="og:image"]').getAttribute("content"); } parseFavoriteByDoc(doc) { const favDataScripts = doc.querySelectorAll(".image-list > span + script"); const favData = Array.from(favDataScripts).map((el) => { const content = el.textContent; const [id] = new RegExp("(?<=posts\\[)[0-9]+?(?=\\])").exec(content); const [tags] = new RegExp(`(?<=tags: ["']).*?(?=["']\\.)`).exec(content); const [rating] = new RegExp(`(?<=rating: ["']).*?(?=["'],)`).exec(content); const [score] = new RegExp(`(?<=score: ["'])[0-9]+(?=["'],)`).exec(content); const [user] = new RegExp(`(?<=user: ["']).*?(?=["']\\s+})`).exec(content); return { id, tags: tags.split(/\s/g), rating: rating.toLowerCase(), score: +score, user }; }); return favData; } } class Rule34 extends GelbooruV020 { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "rule34/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); userAuthentication.cf_clearance ?? (userAuthentication.cf_clearance = ""); } clientSetting.setThemeWatcher({ get current() { return !!document.head.querySelector('link[href*="dark.css"]'); } }); super(); __publicField(this, "api", new GelbooruApiV020()); __publicField(this, "parser", new Rule34Parser()); } static get hostname() { return "rule34.xxx"; } getAvatar() { return "/images/r34chibi.png"; } getThumbnailSelector() { return ".thumb > a:first-child:not(:has(.blacklist-img))"; } setThumbnailStyle(btnContainer) { btnContainer.setAttribute( "style", "position: relative; align-self: center; width: auto; height: auto;" ); const imgEl = btnContainer.querySelector("img"); const setContainerHeight = () => { const aspectRatio = imgEl.naturalHeight / imgEl.naturalWidth; aspectRatio > 1 && (btnContainer.style.height = "inherit"); }; setContainerHeight(); imgEl.onload = setContainerHeight; } createArtworkBtn(id) { var _a2; let isImage = false; let btnContainer = document.querySelector("#gelcomVideoContainer"); if (!btnContainer) { isImage = true; const image = document.querySelector("#image"); btnContainer = document.createElement("div"); (_a2 = image.parentElement) == null ? undefined : _a2.insertBefore(btnContainer, image); btnContainer.append(image); } else { btnContainer.style.fontSize = "0px"; } btnContainer.style.position = "relative"; btnContainer.appendChild( new ArtworkButton({ id, site: isImage ? undefined : "fluid_video", onClick: this.downloadArtwork.bind(this) }) ); } } class DanbooruParser extends ParserBase { constructor() { super(...arguments); __privateAdd(this, _DanbooruParser_instances); } getBlacklistValidationTags(data) { const { tag_string, rating, uploader_id, is_deleted, is_flagged, is_pending } = data; const matchTags = tag_string.match(/\S+/g) ?? []; matchTags.push("rating:" + rating); matchTags.push("uploaderid:" + uploader_id); is_deleted && matchTags.push("status:deleted"); is_flagged && matchTags.push("status:flagged"); is_pending && matchTags.push("status:pending"); return matchTags; } parseCsrfToken() { var _a2; const token = (_a2 = document.head.querySelector('meta[name="csrf-token"]')) == null ? undefined : _a2.content; return token || null; } getBlacklistByHtml() { var _a2; return ((_a2 = document.querySelector('meta[name="blacklisted-tags"]')) == null ? undefined : _a2.content) ?? ""; } getBlacklistItem(blacklistTags, source2 = "api") { let separator; if (source2 === "html") { separator = /,/; } else { separator = /\n+/; } const tags = blacklistTags.replace(/(rating:\w)\w+/gi, "$1").toLowerCase().split(separator).filter((tag) => tag.trim() !== ""); return tags.map(__privateMethod(this, _DanbooruParser_instances, parseBlacklistItem_fn)); } isBlacklisted(matchTags, blacklist) { const scoreRe = /score:(-?[0-9]+)/; const scoreMatch = (matchTags.find((tag) => scoreRe.test(tag)) ?? "").match(scoreRe); const score = scoreMatch ? +scoreMatch[1] : scoreMatch; return blacklist.some((blacklistItem) => { const { require: require2, exclude, optional, min_score } = blacklistItem; const hasTag = (tag) => matchTags.includes(tag); const scoreTest = min_score === null || score === null || score < min_score; return require2.every(hasTag) && scoreTest && (!optional.length || intersect(matchTags, optional).length) && !exclude.some(hasTag); }); } buildMetaByDoc(doc) { var _a2, _b2; const src = (_a2 = doc.querySelector("a[download]")) == null ? undefined : _a2.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 = ((_b2 = doc.querySelector("time")) == null ? undefined : _b2.getAttribute("datetime")) ?? ""; let comment2 = ""; const commentEl = doc.querySelector("#original-artist-commentary"); commentEl && (comment2 = getElementText(commentEl)); const imageContainer = doc.querySelector("section.image-container"); const { score = "0", source: source2 = "", rating = "", id } = imageContainer.dataset; if (!id) throw new Error("Can not parse post id."); return { id, src, extendName, artist: artists.join(",") || this.UNKNOWN_ARTIST, character: characters.join(",") || this.UNKNOWN_CHARACTER, title, comment: comment2, tags, createDate: postDate, score: +score, source: source2, rating }; } buildMetaByApi(post, artistCommentary) { const { id, created_at, file_ext, file_url, md5, tag_string_artist, tag_string_character, tag_string_copyright, tag_string_general, tag_string_meta, score, source: source2, rating } = post; const { original_title = "", original_description = "" } = artistCommentary ?? {}; const addTypeToTag = (type, tag) => tag.split(" ").map((tag2) => type + ":" + tag2); const tags = [ ...addTypeToTag("artist", tag_string_artist), ...addTypeToTag("character", tag_string_character), ...addTypeToTag("copyright", tag_string_copyright), ...addTypeToTag("general", tag_string_general), ...addTypeToTag("meta", tag_string_meta) ]; const comment2 = original_title && original_description ? original_title + "\n" + original_description : original_title || original_description; return { id: String(id), src: file_url ?? "", extendName: file_ext, artist: tag_string_artist.replaceAll(" ", ",") || this.UNKNOWN_ARTIST, character: tag_string_character.replaceAll(" ", ",") || this.UNKNOWN_CHARACTER, title: md5, comment: comment2, tags, createDate: created_at, rating: rating ?? "", score, source: source2 }; } } _DanbooruParser_instances = new WeakSet(); /** * https://github.com/danbooru/danbooru/blob/master/app/javascript/src/javascripts/blacklists.js */ parseBlacklistItem_fn = function(tags) { const tagsArr = tags.split(" "); const require2 = []; const exclude = []; const optional = []; let min_score = null; tagsArr.forEach((tag) => { if (tag.charAt(0) === "-") { exclude.push(tag.slice(1)); } else if (tag.charAt(0) === "~") { optional.push(tag.slice(1)); } else if (tag.match(/^score:<.+/)) { const score = tag.match(/^score:<(.+)/)[1]; min_score = Number.parseInt(score); } else { require2.push(tag); } }); return { tags, require: require2, exclude, optional, min_score }; }; class DanbooruPoolButton extends ThumbnailButton { constructor(props) { super({ ...props, shouldObserveDb: false }); __publicField(this, "downloadingStore"); const { downloading } = props; this.downloadingStore = toStore(() => downloading.current); } static get tagNameLowerCase() { return "pdl-danbooru-pool-button"; } connectedCallback() { super.connectedCallback(); this.unsubscriber = this.downloadingStore.subscribe((val) => { if (val) { this.setAttribute("disabled", ""); } else { this.removeAttribute("disabled"); } }); } } customElements.define(DanbooruPoolButton.tagNameLowerCase, DanbooruPoolButton); class DanbooruApi extends ApiBase { // Danbooru uses some custom status codes in the 4xx and 5xx range: // 200 OK: Request was successful // 204 No Content: Request was successful (returned by create actions) // 400 Bad Request: The given parameters could not be parsed // 401 Unauthorized: Authentication failed // 403 Forbidden: Access denied (see help:users for permissions information) // 404 Not Found: Not found // 410 Gone: Pagination limit (see help:users for pagination limits) // 420 Invalid Record: Record could not be saved // 422 Locked: The resource is locked and cannot be modified // 423 Already Exists: Resource already exists // 424 Invalid Parameters: The given parameters were invalid // 429 User Throttled: User is throttled, try again later (see help:users for API limits) // 500 Internal Server Error: A database timeout, or some unknown error occurred on the server // 502 Bad Gateway: Server cannot currently handle the request, try again later (returned during heavy load) // 503 Service Unavailable: Server cannot currently handle the request, try again later (returned during downbooru) async getJSON(url, init2) { logger.info("Fetch url:", url); const res = await this.fetch(url, init2); if (res.status >= 500) throw new RequestError(res.url, res.status); const data = await res.json(); if ("success" in data && !data.success) { const { error, message: message2 } = data; throw new JsonDataError(error + ", " + message2); } return data; } getPool(id) { return this.getJSON(`/pools/${id}.json`); } getPost(id) { return this.getJSON(`/posts/${id}.json`); } getPostDoc(id) { return this.getDoc("/posts/" + id); } getPostList(param) { const { tags = [], limit = 0, page = 0 } = param ?? {}; const searchParam = new URLSearchParams(); (tags == null ? undefined : tags.length) && searchParam.append("tags", tags.join(" ")); limit && searchParam.append("limit", String(limit)); page && searchParam.append("page", String(page)); return this.getJSON(`/posts.json?${searchParam.toString()}`); } /** * In aftbooru, account needs to be over a week old to get commentary. */ getArtistCommentary(id) { return this.getJSON(`/posts/${id}/artist_commentary.json`); } getFavoriteGroups(id) { return this.getJSON(`/favorite_groups/${id}.json`); } getProfile() { return this.getJSON(`/profile.json`); } async addFavorite(id, token) { const res = await this.fetch(`/favorites?post_id=${id}`, { method: "POST", headers: { "X-Csrf-Token": token } }); if (!res.ok) throw new RequestError(res.url, res.status); return await res.text(); } } class AbstractDanbooru extends SiteInject { constructor() { if (clientSetting.version === null) { siteFeature.addBookmark ?? (siteFeature.addBookmark = false); } const userTheme = document.body.getAttribute("data-current-user-theme"); if (userTheme !== "auto") { clientSetting.setThemeWatcher({ get current() { return userTheme === "dark"; } }); } super(); __privateAdd(this, _AbstractDanbooru_instances); __publicField(this, "profile", null); __publicField(this, "blacklist", null); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: () => this.getAvatar(), beforeDownload: async () => { this.profile = await this.api.getProfile(); const blacklistTags = this.profile.blacklisted_tags; this.blacklist = blacklistTags ? this.parser.getBlacklistItem(blacklistTags) : null; }, afterDownload: () => { this.profile = null; this.blacklist = null; }, parseMetaByArtworkId: async (id) => { return this.getMetaByPostId(id); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfig = new BooruDownloadConfig(meta).create({ ...downloadSetting }); await downloader.download(downloadConfig, { signal }); const { id, tags, artist, title, comment: comment2, source: source2, rating } = meta; historyDb.add({ pid: Number(id), user: artist, title, comment: comment2, tags, source: source2, rating }); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "exclude_blacklist", type: "exclude", name: () => t("downloader.category.filter.exclude_blacklist"), checked: true, fn: async (meta) => { return !!meta.blacklistValidationTags && this.parser.isBlacklisted(meta.blacklistValidationTags, this.blacklist); } }, { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn(meta) { return !!meta.extendName && /bmp|jp(e)?g|png|tif|gif|exif|svg|webp/i.test(meta.extendName); } }, { id: "allow_video", type: "include", name: () => t("downloader.category.filter.video"), checked: true, fn(meta) { return !!meta.extendName && /mp4|avi|mov|mkv|flv|wmv|webm|mpeg|mpg|m4v/i.test(meta.extendName); } } ], enableTagFilter: true }, pageOption: { pool: { name: "Pool", match: new RegExp("(?<=\\/pools\\/)[0-9]+"), filterInGenerator: true, fn: (pageRange, checkValidity) => { var _a2; const poolId = (_a2 = new RegExp("(?<=\\/pools\\/)[0-9]+").exec(location.pathname)) == null ? undefined : _a2[0]; if (!poolId) throw new Error("Invalid pool id"); const perPage = this.profile.per_page; const getPostDataByPage = async (page) => { const data = await this.api.getPostList({ tags: [`ordpool:${poolId}`], limit: perPage, page }); return { lastPage: data.length < perPage, data }; }; return this.parser.paginationGenerator( pageRange, getPostDataByPage, (post) => String(post.id), __privateMethod(this, _AbstractDanbooru_instances, validityCheckFactory_fn2).call(this, checkValidity) ); } }, favorite_groups: { name: "FavoriteGroups", match: new RegExp("(?<=\\/favorite_groups\\/)[0-9]+"), filterInGenerator: true, fn: (pageRange, checkValidity) => { var _a2; const groupId = (_a2 = new RegExp("(?<=\\/favorite_groups\\/)[0-9]+").exec(location.pathname)) == null ? undefined : _a2[0]; if (!groupId) throw new Error("Invalid pool id"); const perPage = this.profile.per_page; const getPostDataByPage = async (page) => { const data = await this.api.getPostList({ tags: [`ordfavgroup:${groupId}`], limit: perPage, page }); return { lastPage: data.length < perPage, data }; }; return this.parser.paginationGenerator( pageRange, getPostDataByPage, (post) => String(post.id), __privateMethod(this, _AbstractDanbooru_instances, validityCheckFactory_fn2).call(this, checkValidity) ); } }, post_list: { name: "Post", match: () => location.pathname === "/" || location.pathname === "/posts", filterInGenerator: true, fn: (pageRange, checkValidity) => { var _a2; const perPage = this.profile.per_page; const searchParam = new URLSearchParams(location.search); const tags = ((_a2 = searchParam.get("tags")) == null ? undefined : _a2.split(" ")) ?? []; const limit = searchParam.get("limit"); const limitParam = limit ? Number(limit) : perPage; const getPostDataByPage = async (page) => { const data = await this.api.getPostList({ tags, limit: limitParam, page }); return { lastPage: data.length < limitParam, data }; }; const showDeletedPosts = (tags == null ? undefined : tags.includes("status:deleted")) || this.profile.show_deleted_posts; return this.parser.paginationGenerator( pageRange, getPostDataByPage, (post) => String(post.id), __privateMethod(this, _AbstractDanbooru_instances, validityCheckFactory_fn2).call(this, checkValidity, showDeletedPosts) ); } }, pool_gallery_button: { name: "pool_gallery_button", match: () => false, filterInGenerator: true, fn: (pageRange, checkValidity, poolId) => { if (!poolId) throw new Error("Invalid pool id"); const perPage = this.profile.per_page; const getPostDataByPage = async (page) => { const data = await this.api.getPostList({ tags: [`ordpool:${poolId}`], limit: perPage, page }); return { lastPage: data.length < perPage, data }; }; return this.parser.paginationGenerator( pageRange, getPostDataByPage, (post) => String(post.id), __privateMethod(this, _AbstractDanbooru_instances, validityCheckFactory_fn2).call(this, checkValidity) ); } }, show_downloader_in_pool_gallery: { name: "pool_gallery", match: /\/pools\/gallery/ } } })); } getSupportedTemplate() { return BooruDownloadConfig.supportedTemplate; } async getPostAndComment(id) { const [postResult, commentResult] = await Promise.allSettled([ this.api.getPost(id), this.api.getArtistCommentary(id) ]); if (postResult.status === "rejected") throw postResult.reason; let comment2 = undefined; if (commentResult.status === "rejected") { if (!(commentResult.reason instanceof JsonDataError)) { throw commentResult.reason; } } else { comment2 = commentResult.value; } return { post: postResult.value, comment: comment2 }; } async getMetaByPostId(id) { const { post, comment: commentary } = await this.getPostAndComment(id); return this.parser.buildMetaByApi(post, commentary); } async addBookmark(id) { try { const token = this.parser.parseCsrfToken(); if (!token) throw new Error("Can not get csrf-token"); const script = await this.api.addFavorite(id, token); const galleryMatch = new RegExp("(?<=^\\/posts\\/)\\d+").exec(location.pathname); if (galleryMatch && id === galleryMatch[0]) { evalScript(script); } this.toast({ message: "You have favorited this post", timeout: 2e3 }); } catch (error) { logger.error(error); this.toast({ message: error.message, type: "error" }); } } async downloadArtwork(btn2) { const id = btn2.dataset.id; const mediaMeta = await this.getMetaByPostId(id); const downloadConfig = new BooruDownloadConfig(mediaMeta).create({ ...downloadSetting, setProgress: (progress) => { btn2.setProgress(progress); } }); siteFeature.addBookmark && this.addBookmark(id); await downloader.download(downloadConfig, { priority: 1 }); const { tags, artist, title, comment: comment2, source: source2, rating } = mediaMeta; historyDb.add({ pid: Number(id), user: artist, title, comment: comment2, tags, source: source2, rating }); } createThumbnailBtn() { const btnContainers = document.querySelectorAll( "article a.post-preview-link" ); if (!btnContainers.length) return; btnContainers.forEach((el) => { var _a2; const id = (_a2 = new RegExp("(?<=\\/posts\\/)\\d+").exec(el.href)) == null ? undefined : _a2[0]; if (!id) return; const btn2 = new ThumbnailButton({ id, onClick: this.downloadArtwork }); el.appendChild(btn2); }); } createArtworkBtn() { const btnContainer = document.querySelector( "section.image-container:has(:is(picture, video))" ); if (!btnContainer) return; const id = btnContainer.getAttribute("data-id"); btnContainer.appendChild( new ArtworkButton({ id, site: btnContainer.querySelector("video") ? "native_video" : undefined, onClick: this.downloadArtwork }) ); } createPoolThumbnailBtn() { const btnContainers = document.querySelectorAll( "article a.post-preview-link" ); if (!btnContainers.length) return; const { downloading, batchDownload } = this.useBatchDownload(); const onClick = (btn2) => { const poolId = btn2.dataset.id; return batchDownload("pool_gallery_button", poolId); }; btnContainers.forEach((el) => { var _a2; const poolId = (_a2 = new RegExp("(?<=\\/pools\\/)\\d+").exec(el.href)) == null ? undefined : _a2[0]; if (!poolId) return; const btn2 = new DanbooruPoolButton({ id: poolId, downloading, onClick }); el.appendChild(btn2); }); } inject() { super.inject(); this.downloadArtwork = this.downloadArtwork.bind(this); const path = location.pathname; if (/^\/posts\/\d+/.test(path)) { this.createArtworkBtn(); this.createThumbnailBtn(); } else if (/^\/pools\/gallery/.test(path)) { this.createPoolThumbnailBtn(); } else { this.createThumbnailBtn(); } } } _AbstractDanbooru_instances = new WeakSet(); validityCheckFactory_fn2 = function(checkValidity, showDeletedPosts = true) { return async (post) => { const { id, is_deleted, file_ext, file_url, tag_string } = post; if (!file_url) return PostValidState.UNAVAILABLE; if (is_deleted && !showDeletedPosts) return PostValidState.INVALID; const blacklistValidationTags = this.parser.getBlacklistValidationTags(post); return await checkValidity({ id: String(id), extendName: file_ext, tags: tag_string.split(" "), blacklistValidationTags }) ? PostValidState.VALID : PostValidState.INVALID; }; }; class Danbooru extends AbstractDanbooru { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "danbooru/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); } super(); __publicField(this, "api", new DanbooruApi()); __publicField(this, "parser", new DanbooruParser()); } static get hostname() { return "danbooru.donmai.us"; } getAvatar() { return "/packs/static/danbooru-logo-128x128-ea111b6658173e847734.png"; } } var IllustType = /* @__PURE__ */ ((IllustType2) => { IllustType2[IllustType2["illusts"] = 0] = "illusts"; IllustType2[IllustType2["manga"] = 1] = "manga"; IllustType2[IllustType2["ugoira"] = 2] = "ugoira"; return IllustType2; })(IllustType || {}); var BookmarkRestrict = /* @__PURE__ */ ((BookmarkRestrict2) => { BookmarkRestrict2[BookmarkRestrict2["public"] = 0] = "public"; BookmarkRestrict2[BookmarkRestrict2["private"] = 1] = "private"; return BookmarkRestrict2; })(BookmarkRestrict || {}); class PixivApi extends ApiBase { async getJSON(url, init2) { const data = await super.getJSON(url, init2); if (data.error) throw new JsonDataError(data.message); return data.body; } async getArtworkHtml(illustId, lang) { const res = await this.fetch(`/artworks/${illustId}?lang=${lang}`); if (!res.ok) throw new RequestError(res.url, res.status); return await res.text(); } getArtworkDoc(illustId, lang) { return this.getDoc(`/artworks/${illustId}?lang=${lang}`); } getArtworkDetail(illustId, lang) { return this.getJSON(`/ajax/illust/${illustId}?lang=${lang}`); } getUnlistedArtworkDetail(unlistedId, lang) { return this.getJSON(`/ajax/illust/unlisted/${unlistedId}?lang=${lang}`); } addBookmark(illustId, token, tags = [], restrict = BookmarkRestrict.public) { return this.getJSON("/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 }) }); } likeIllust(illustId, token) { return this.getJSON("/ajax/illusts/like", { method: "POST", headers: { accept: "application/json", "content-type": "application/json; charset=utf-8", "x-csrf-token": token }, body: JSON.stringify({ illust_id: illustId }) }); } getFollowLatestWorks(page, mode = "all") { return this.getJSON(`/ajax/follow_latest/illust?p=${page}&mode=${mode}&lang=jp`); } getUserAllProfile(userId) { return this.getJSON("/ajax/user/" + userId + "/profile/all"); } getUgoiraMeta(illustId) { return this.getJSON("/ajax/illust/" + illustId + "/ugoira_meta"); } getUserData(userId) { return this.getJSON("/ajax/user/" + userId); } getSeriesData(seriesId, page) { return this.getJSON(`/ajax/series/${seriesId}?p=${page}`); } } const pixivApi = new PixivApi(); const pixivParser = { async parse(illustId, param) { let illustData; let token; const { tagLang, type } = param; if (type === "api") { illustData = await pixivApi.getArtworkDetail(illustId, tagLang); token = ""; } else if (type === "unlisted") { illustData = await pixivApi.getUnlistedArtworkDetail(illustId, tagLang); token = ""; } else { const doc = await pixivApi.getArtworkDoc(illustId, tagLang); const preloadDataEl = doc.querySelector('meta[name="preload-data"]'); const globalDataEl = doc.querySelector('meta[name="global-data"]'); if (preloadDataEl && globalDataEl) { illustData = JSON.parse(preloadDataEl.content).illust[illustId]; token = JSON.parse(globalDataEl.content).token; } else { const nextDataEL = doc.querySelector("script#__NEXT_DATA__"); if (!nextDataEL) throw new Error("Cannot get csrf token."); const nextData = JSON.parse(nextDataEL.textContent); const preloadState = JSON.parse( nextData.props.pageProps.serverSerializedPreloadedState ); token = preloadState.api.token; illustData = await pixivApi.getArtworkDetail(illustId, tagLang); } } const { id, illustType, userName, userId, illustTitle, illustComment, tags, pageCount, createDate, urls, bookmarkData, likeData, bookmarkCount } = illustData; const tagsArr = []; const tagsTranslatedArr = []; tags.tags.forEach((tagData) => { var _a2; tagsArr.push(tagData.tag); tagsTranslatedArr.push(((_a2 = tagData.translation) == null ? undefined : _a2.en) || tagData.tag); }); const unescapeComment = illustComment.replaceAll(/<|&lt;/g, "<").replaceAll(/>|&gt;/g, ">"); const p = document.createElement("p"); p.innerHTML = unescapeComment; const comment2 = getElementText(p); const meta = { id, src: urls.original, extendName: urls.original.slice(-3), artist: userName, title: illustTitle, tags: tagsArr, tagsTranslated: tagsTranslatedArr, userId, comment: comment2, bookmarkData, createDate, likeData, token, bookmarkCount }; if (illustType === IllustType.ugoira) { const ugoiraMeta = await pixivApi.getUgoiraMeta(illustId); const pageCount2 = ugoiraMeta.frames.length; const src = Array.from( { length: pageCount2 }, (_, i) => meta.src.replace("ugoira0", "ugoira" + i) ); const extendName = Array.from({ length: pageCount2 }).fill(meta.extendName); return { ...meta, src, extendName, illustType, ugoiraMeta }; } else if (pageCount > 1) { const src = Array.from({ length: pageCount }, (_, i) => meta.src.replace("_p0", "_p" + i)); const extendName = Array.from({ length: pageCount }).fill(meta.extendName); return { ...meta, src, extendName, illustType }; } else { return { ...meta, illustType }; } }, async *illustMangaGenerator(pageRange, checkValidity, userId) { const ARTWORKS_PER_PAGE = 48; const profile = await pixivApi.getUserAllProfile(userId); let ids = []; typeof profile.illusts === "object" && ids.push(...Object.keys(profile.illusts)); typeof profile.manga === "object" && ids.push(...Object.keys(profile.manga)); if (!ids.length) throw new Error(`User ${userId} has no illusts or mangas.`); ids = ids.sort((a, b) => Number(b) - Number(a)); let sliceStart; let sliceEnd; const [startPage = null, endPage = null] = pageRange ?? []; let page = startPage ?? 1; startPage === null ? sliceStart = 0 : sliceStart = (startPage - 1) * ARTWORKS_PER_PAGE; endPage === null ? sliceEnd = ids.length : sliceEnd = endPage * ARTWORKS_PER_PAGE; const selectedIds = ids.slice(sliceStart, sliceEnd); if (!selectedIds.length) throw new RangeError(`Page ${page} exceeds the limit.`); const baseUrl = `https://www.pixiv.net/ajax/user/${userId}/profile/illusts?`; const total = selectedIds.length; do { const chunk = selectedIds.splice(0, ARTWORKS_PER_PAGE); const queryStr = chunk.map((id) => "ids[]=" + id).join("&") + `&work_category=illustManga&is_first_page=0&lang=ja`; const data = await pixivApi.getJSON(baseUrl + queryStr); const workDatas = Object.values(data.works).sort((a, b) => Number(b.id) - Number(a.id)); const avaliable = []; const invalid = []; const unavaliable = []; for (let i = 0; i < workDatas.length; i++) { const work = workDatas[i]; const { id, isMasked } = work; if (isMasked) { unavaliable.push(String(id)); continue; } const isValid = await checkValidity(work); isValid ? avaliable.push(id) : invalid.push(id); } yield { total, page, avaliable, invalid, unavaliable }; page++; } while (selectedIds.length > 0); }, async *chunkGenerator(pageRange, checkValidity, userId, category, tag, bookmarkRest = "show") { const ARTWORKS_PER_PAGE = 48; const [startPage = null, endPage = null] = pageRange ?? []; if (!userId) throw new Error('Require argument "userId".'); let offset; let offsetEnd; let total; let page = startPage ?? 1; startPage === null ? offset = 0 : offset = (startPage - 1) * ARTWORKS_PER_PAGE; do { let requestUrl; if (category === "bookmarks") { requestUrl = `/ajax/user/${userId}/illusts/bookmarks?tag=${tag}&offset=${offset}&limit=${ARTWORKS_PER_PAGE}&rest=${bookmarkRest}&lang=ja`; } else { requestUrl = `/ajax/user/${userId}/${category}/tag?tag=${tag}&offset=${offset}&limit=${ARTWORKS_PER_PAGE}&lang=ja`; } const userPageData = await pixivApi.getJSON(requestUrl); const { works, total: totalArtwork } = userPageData; if (totalArtwork === 0) throw new Error(`User ${userId} has no ${category} tagged with ${tag}.`); if (!offsetEnd) { endPage === null ? offsetEnd = totalArtwork : offsetEnd = endPage * ARTWORKS_PER_PAGE > totalArtwork ? totalArtwork : endPage * ARTWORKS_PER_PAGE; if (offsetEnd <= offset) throw new RangeError(`Page ${page} exceeds the limit.`); total = offsetEnd - offset; } const avaliable = []; const invalid = []; const unavaliable = []; for (let i = 0; i < works.length; i++) { const work = works[i]; const { id, isMasked } = work; if (isMasked) { unavaliable.push(String(id)); continue; } const isValid = await checkValidity(work); isValid ? avaliable.push(id) : invalid.push(id); } yield { total, page, avaliable, invalid, unavaliable }; page++; } while ((offset += ARTWORKS_PER_PAGE) < offsetEnd); }, async *bookmarkGenerator(pageRange, checkValidity, userId, bookmarkRest = "show", tag = "") { yield* this.chunkGenerator(pageRange, checkValidity, userId, "bookmarks", tag, bookmarkRest); }, async *taggedArtworkGenerator(pageRange, checkValidity, userId, category, tag, bookmarkRest = "show") { if (category === "bookmarks") { yield* this.bookmarkGenerator(pageRange, checkValidity, userId, bookmarkRest, tag); } else { yield* this.chunkGenerator(pageRange, checkValidity, userId, category, tag); } }, async *followLatestGenerator(pageRange, checkValidity, mode = "all") { const PAGE_LIMIT = 34; const ARTWORKS_PER_PAGE = 60; let [startPage = null, endPage = null] = pageRange ?? []; startPage === null && (startPage = 1); (endPage === null || endPage > PAGE_LIMIT) && (endPage = PAGE_LIMIT); if (startPage > PAGE_LIMIT) throw new RangeError(`Page ${startPage} exceeds the limit.`); let earliestId; let total; let cache; let page = startPage; function findEarliestId(ids2) { return Math.min(...ids2); } async function* yieldData(data2, page2) { const avaliable = []; const invalid = []; const unavaliable = []; const { illust } = data2.thumbnails; for (let i = 0; i < illust.length; i++) { const work = illust[i]; const { id, isMasked } = work; if (isMasked) { unavaliable.push(String(id)); continue; } const isValid = await checkValidity(work); isValid ? avaliable.push(id) : invalid.push(id); } const { ids: ids2 } = data2.page; if (ids2.length !== illust.length) { const idDiff = ids2.filter((id) => !illust.some((item) => +item.id === id)); unavaliable.push(...idDiff.map((id) => String(id))); } yield { total, page: page2, avaliable, invalid, unavaliable }; } const data = await pixivApi.getFollowLatestWorks(page, mode); const ids = data.page.ids; total = ids.length; earliestId = findEarliestId(ids); if (endPage === startPage) { yield* yieldData(data, startPage); return; } if (total === ARTWORKS_PER_PAGE) { const secondPageData = await pixivApi.getFollowLatestWorks(++page, mode); const secondIds = secondPageData.page.ids; const secondPageEarliestId = findEarliestId(secondIds); if (secondPageEarliestId < earliestId) { earliestId = secondPageEarliestId; cache = secondPageData; total += secondIds.length; } } yield* yieldData(data, startPage); if (total === ARTWORKS_PER_PAGE) return; if (total < ARTWORKS_PER_PAGE * 2 || endPage - startPage === 1) { yield* yieldData(cache, page); return; } while (++page <= endPage) { const data2 = await pixivApi.getFollowLatestWorks(page, mode); const ids2 = data2.page.ids; const pageEarliestId = findEarliestId(ids2); if (pageEarliestId >= earliestId) { logger.info("getFollowLatestGenerator: got duplicate works"); yield* yieldData(cache, page - 1); break; } earliestId = pageEarliestId; total += ids2.length; yield* yieldData(cache, page - 1); cache = data2; } yield* yieldData(cache, page - 1); }, async *seriesGenerator(pageRange, checkValidity, seriesId) { const [startPage = 1, endPage = 0] = pageRange ?? []; let yieldedId = 0; let total = 0; let currentPage = startPage; do { const seriesData = await pixivApi.getSeriesData(seriesId, currentPage); const { series } = seriesData.page; if (!series.length) throw new Error(`Invalid page: ${currentPage}`); const { illust } = seriesData.thumbnails; if (!total) { const isLastPage = series.some(({ order }) => order === 1); const totalWorkCount = seriesData.page.total; if (isLastPage) { total = series.length; } else if (endPage === 0) { total = totalWorkCount; } else { const artworksPerPage = series.length; const lastPage = Math.ceil(totalWorkCount / artworksPerPage); if (endPage >= lastPage) { const lastPageWorkCount = totalWorkCount % artworksPerPage || artworksPerPage; total = (lastPage - startPage) * artworksPerPage + lastPageWorkCount; } else { total = (endPage - startPage + 1) * artworksPerPage; } } } const avaliable = []; const invalid = []; const unavaliable = []; for (let i = 0; i < series.length; i++) { const { workId } = series[i]; const thumbnail = illust.find((thumbnail2) => thumbnail2.id === workId); if (!thumbnail || thumbnail.isMasked) { unavaliable.push(workId); continue; } const isValid = await checkValidity(thumbnail); isValid ? avaliable.push(workId) : invalid.push(workId); } yield { total, page: currentPage, avaliable, invalid, unavaliable }; yieldedId += series.length; currentPage++; } while (yieldedId < total); } }; function getSelfId() { var _a2, _b2; return ((_b2 = (_a2 = _unsafeWindow.dataLayer) == null ? undefined : _a2[0]) == null ? undefined : _b2.user_id) ?? ""; } 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 instanceof HTMLSpanElement && node.className.includes("_history-item")) { const img = node.querySelector("img"); if (!img) return ""; const matchPid = regexp.historyThumbnailsId.exec(img.src); if (matchPid) return matchPid[0]; } else { const isActivityThumb = regexp.activityHref.exec(node.getAttribute("href") || ""); if (isActivityThumb && node.classList.contains("work")) { return isActivityThumb[1]; } } return ""; } function createThumbnailBtn(nodes, downloadArtwork) { 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(ThumbnailButton.tagNameLowerCase) && (illustId = getIllustId(e))) { if (isSelfBookmark) { type = ThumbnailBtnType.PixivMyBookmark; } else if (e.className.includes("_history-related-item")) { e.style.position = "relative"; type = ThumbnailBtnType.PixivHistory; } else if (e.className.includes("_history-item")) { type = ThumbnailBtnType.PixivHistory; } const btn2 = new ThumbnailButton({ id: illustId, type, onClick: downloadArtwork }); e.appendChild(btn2); } }); } function fixPixivPreviewer(nodes) { if (!regexp.searchPage.test(location.pathname)) return; nodes.forEach((node) => { var _a2; (_a2 = node.querySelector(ThumbnailButton.tagNameLowerCase)) == null ? undefined : _a2.remove(); }); } function createToolbarBtn(id, downloadArtwork) { const toolbar = document.querySelector("main section section"); if (!toolbar || toolbar.querySelector(ThumbnailButton.tagNameLowerCase)) return; const btn2 = new ThumbnailButton({ id, type: ThumbnailBtnType.PixivToolbar, onClick: downloadArtwork }); const pdlBtnWrap = toolbar.lastElementChild.cloneNode(); pdlBtnWrap.appendChild(btn2); toolbar.appendChild(pdlBtnWrap); } function createWorkExpanedViewBtn(id, downloadArtwork, unlistedId) { const works = document.querySelectorAll( "figure a.gtm-expand-full-size-illust" ); if (works.length < 2) return; works.forEach((work, idx) => { var _a2; const container = (_a2 = work.parentElement) == null ? undefined : _a2.parentElement; if (!container || container.querySelector(ArtworkButton.tagNameLowerCase)) return; container.appendChild( new ArtworkButton({ id, page: idx, site: "pixiv", extraData: unlistedId ? { unlistedId } : undefined, onClick: downloadArtwork }) ); }); } let observer; let btn; function createPresentationBtn(id, downloadArtwork, unlistedId) { const containers = document.querySelector("body > [role='presentation'] div[class]"); if (!containers) { if (observer) { observer.disconnect(); observer = null; btn = null; } return; } if (containers.querySelector(ThumbnailButton.tagNameLowerCase)) 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: ThumbnailBtnType.PixivPresentation, page: Number(pageNum), extraData: unlistedId ? { unlistedId } : undefined, onClick: downloadArtwork }); containers.appendChild(btn); observer = new MutationObserver((mutationList) => { const newImg = mutationList[1]["addedNodes"][0]; const [pageNum2] = regexp.originSrcPageNum.exec(newImg.src) ?? []; if (!pageNum2) return logger.throw("Invalid image element."); btn == null ? undefined : btn.remove(); btn = new ThumbnailButton({ id, type: ThumbnailBtnType.PixivPresentation, page: Number(pageNum2), extraData: unlistedId ? { unlistedId } : undefined, onClick: downloadArtwork }); containers.appendChild(btn); }); observer.observe(img.parentElement, { childList: true, subtree: true }); } function createPreviewModalBtn(id, downloadArtwork, unlistedId) { var _a2; 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 = (_a2 = document.querySelectorAll( ".gtm-manga-viewer-close-icon:not(.pdl-listened)" )) == null ? undefined : _a2[1]; if (!illustModalBtn && !mangaModalBtn && !mangaViewerModalBtn) return; [illustModalBtn, mangaModalBtn, mangaViewerModalBtn].forEach((node) => { if (node) { node.classList.add("pdl-listened"); node.addEventListener("click", () => { handleModalClick(id, downloadArtwork, unlistedId); }); } }); } function handleModalClick(id, downloadArtwork, unlistedId) { 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); previewList.childNodes.forEach((node, idx) => { node.style.position = "relative"; node.appendChild( new ThumbnailButton({ id, page: idx, extraData: unlistedId ? { unlistedId } : undefined, onClick: downloadArtwork }) ); }); }, 300); } function createMangaViewerBtn(id, downloadArtwork, unlistedId) { const mangaViewerBackBtn = document.querySelector(".gtm-manga-viewer-close-icon"); if (!mangaViewerBackBtn) return; const container = mangaViewerBackBtn.parentElement; if (!container || container.querySelector(ThumbnailButton.tagNameLowerCase)) return; container.appendChild( new ThumbnailButton({ id, type: ThumbnailBtnType.PixivMangaViewer, extraData: unlistedId ? { unlistedId } : undefined, onClick: downloadArtwork }) ); } const toolbarStyle = ".button-wrapper{display:flex;justify-content:flex-end;align-items:center;height:32px;padding:8px 12px}"; class UnlistedArtworkToolbar extends HTMLElement { constructor(props) { super(); __publicField(this, "props"); this.props = props; } static get tagNameLowerCase() { return "pdl-unlisted-artwork-toolbar"; } render() { if (this.shadowRoot) return; const shadowRoot = this.attachShadow({ mode: "open" }); shadowRoot.innerHTML = `<style>${toolbarStyle}</style><div class="button-wrapper"></div>`; const thumbnailButton = new ThumbnailButton({ ...this.props, type: ThumbnailBtnType.PixivToolbar }); const wrapper = shadowRoot.querySelector(".button-wrapper"); wrapper.appendChild(thumbnailButton); } connectedCallback() { this.render(); } } customElements.define(UnlistedArtworkToolbar.tagNameLowerCase, UnlistedArtworkToolbar); function createUnlistedToolbar(id, downloadArtwork, unlistedId) { const toolbar = document.querySelector(UnlistedArtworkToolbar.tagNameLowerCase); if (toolbar) return; const container = document.querySelector('div[style^="transform: translateY"]'); if (!container) return; const el = new UnlistedArtworkToolbar({ id, onClick: downloadArtwork, extraData: { unlistedId } }); container.appendChild(el); const showAllBtn = container.querySelector( 'button[type="button"]:not([style])' ); showAllBtn && (showAllBtn.style.bottom = "48px"); } class TagListButton extends HTMLElement { constructor(tagUrl, downloading, handleDownload) { super(); __publicField(this, "btn"); __publicField(this, "downloadingStore"); __publicField(this, "unsubscriber"); this.tagUrl = tagUrl; this.handleDownload = handleDownload; this.downloadingStore = toStore(() => downloading.current); this.dispatchDownload = this.dispatchDownload.bind(this); } static get tagNameLowerCase() { return "pdl-tag-list-button"; } async render() { if (this.shadowRoot) return; const shadowRoot = this.attachShadow({ mode: "open" }); addStyleToShadow(shadowRoot); shadowRoot.innerHTML = ` <div class=" flex items-center"> <hr class="!border-t-0 border-l h-6 ml-4 mr-2" /> <button class=" h-[38px] w-[38px] btn-icon [&:not([disabled])]:hover:bg-slate-400/30 disabled:cursor-wait disabled:opacity-70"> <i class="text-sm w-6 fill-current mx-2"> ${downloadSvg} </i> </button> </div> `; } getTagProps() { const url = new URL(this.tagUrl); const { searchParams, pathname } = url; const extractUrlMatch = regexp.userPageTags.exec(pathname); if (!extractUrlMatch) throw new Error(`Could not extract tag props from: ${pathname}`); const [, userId, urlCategory, tag] = extractUrlMatch; if (!tag) throw new Error(`Could not extract tag from: ${pathname}`); let category; if (urlCategory === "illustrations" || urlCategory === "artworks") { category = "illusts"; } else { category = urlCategory; } return { userId, category, tag, rest: searchParams.get("rest") === "hide" ? "hide" : "show" }; } dispatchDownload(evt) { evt == null ? undefined : evt.preventDefault(); this.handleDownload(this.getTagProps()).catch(logger.error); } connectedCallback() { this.render(); this.btn ?? (this.btn = this.shadowRoot.querySelector("button")); this.btn.addEventListener("click", this.dispatchDownload); this.unsubscriber = this.downloadingStore.subscribe((val) => { if (val) { this.setAttribute("disabled", ""); } else { this.removeAttribute("disabled"); } }); } disconnectedCallback() { var _a2, _b2; (_a2 = this.unsubscriber) == null ? undefined : _a2.call(this); (_b2 = this.btn) == null ? undefined : _b2.removeEventListener("click", this.dispatchDownload); } static get observedAttributes() { return ["disabled"]; } attributeChangedCallback(name, oldValue, newValue) { var _a2, _b2; if (typeof newValue === "string") { (_a2 = this.btn) == null ? undefined : _a2.setAttribute("disabled", ""); } else { (_b2 = this.btn) == null ? undefined : _b2.removeAttribute("disabled"); } } } customElements.define(TagListButton.tagNameLowerCase, TagListButton); function createTagListBtn(downloading, handleDownload) { var _a2; const listContainer = document.querySelector('div[style*="position: relative"]'); if (!listContainer) return; const modalRoot = listContainer == null ? undefined : listContainer.closest('div[role="presentation"], div[class="charcoal-token"]'); const closeBtn = (_a2 = modalRoot == null ? undefined : modalRoot.querySelector("svg")) == null ? undefined : _a2.parentElement; const tagElements = listContainer.querySelectorAll( 'div[style*="position: absolute"] a' ); tagElements.forEach((ele) => { if (ele.querySelector(TagListButton.tagNameLowerCase)) return; const btn2 = new TagListButton(ele.href, downloading, (props) => { closeBtn == null ? undefined : closeBtn.click(); return handleDownload(props); }); ele.appendChild(btn2); }); } class ArtworkTagButton extends HTMLElement { constructor(tagElement, downloading, handleDownload) { super(); __publicField(this, "btn"); __publicField(this, "ob"); __publicField(this, "downloadingStore"); __publicField(this, "unsubscriber"); this.tagElement = tagElement; this.handleDownload = handleDownload; this.downloadingStore = toStore(() => downloading.current); this.dispatchDownload = this.dispatchDownload.bind(this); this.ob = new MutationObserver(() => { this.changeBtnColor(); }); } static get tagNameLowerCase() { return "pdl-artwork-tag"; } // 为了美观 resetTagStyle() { this.tagElement.style.borderTopRightRadius = "0px"; this.tagElement.style.borderBottomRightRadius = "0px"; } changeBtnColor() { if (!this.btn) return; const { color, backgroundColor } = getComputedStyle(this.tagElement); this.btn.style.color = color; this.btn.style.backgroundColor = backgroundColor; } async render() { if (this.shadowRoot) return; const shadowRoot = this.attachShadow({ mode: "open" }); addStyleToShadow(shadowRoot); shadowRoot.innerHTML = ` <button class="flex h-full items-center pr-2 rounded-e-[4px] disabled:cursor-wait disabled:opacity-70"> <hr class="!border-t-0 border-l h-6 pr-2" /> <i class="text-sm w-6 fill-current"> ${downloadSvg} </i> </button>`; this.resetTagStyle(); } getTagProps() { const tagTitles = this.tagElement.querySelectorAll("div[title]"); const tagStr = tagTitles[tagTitles.length - 1].getAttribute("title"); const tag = tagStr.startsWith("#") ? tagStr.slice(1) : "未分類"; const url = new URL(this.tagElement.href); const { searchParams, pathname } = url; const extractUrlMatch = regexp.userPageTags.exec(pathname); if (!extractUrlMatch) throw new Error(`Could not extract tag props from: ${pathname}`); const [, userId, urlCategory] = extractUrlMatch; let category; if (urlCategory === "illustrations" || urlCategory === "artworks") { category = "illusts"; } else { category = urlCategory; } return { userId, category, tag, rest: searchParams.get("rest") === "hide" ? "hide" : "show" }; } dispatchDownload() { this.handleDownload(this.getTagProps()).catch(logger.error); } connectedCallback() { this.render(); this.btn ?? (this.btn = this.shadowRoot.querySelector("button")); this.changeBtnColor(); this.btn.addEventListener("click", this.dispatchDownload); this.unsubscriber = this.downloadingStore.subscribe((val) => { if (val) { this.setAttribute("disabled", ""); } else { this.removeAttribute("disabled"); } }); this.ob.observe(this.tagElement, { attributes: true, attributeFilter: ["class"] }); } disconnectedCallback() { var _a2, _b2; (_a2 = this.unsubscriber) == null ? undefined : _a2.call(this); (_b2 = this.btn) == null ? undefined : _b2.removeEventListener("click", this.dispatchDownload); this.ob.disconnect(); } static get observedAttributes() { return ["disabled"]; } attributeChangedCallback(name, oldValue, newValue) { var _a2, _b2; if (typeof newValue === "string") { (_a2 = this.btn) == null ? undefined : _a2.setAttribute("disabled", ""); } else { (_b2 = this.btn) == null ? undefined : _b2.removeAttribute("disabled"); } } } customElements.define(ArtworkTagButton.tagNameLowerCase, ArtworkTagButton); function createFrequentTagBtn(downloading, handleDownload) { const tagsEles = Array.from(document.querySelectorAll("a[status]")); if (!tagsEles.length) return; tagsEles.forEach((ele) => { var _a2; if (((_a2 = ele.nextElementSibling) == null ? undefined : _a2.tagName.toLowerCase()) === ArtworkTagButton.tagNameLowerCase) return; const artworkTagBtn = new ArtworkTagButton(ele, downloading, handleDownload); ele.parentElement.appendChild(artworkTagBtn); }); } class Converter { constructor() { __privateAdd(this, _Converter_instances); __privateAdd(this, _ugoiraFramesData, {}); __privateAdd(this, _queue3, new PQueue({ concurrency: 2 })); } addFrame(addFrameOption) { var _a2; const { id, frame, delay, order } = addFrameOption; (_a2 = __privateGet(this, _ugoiraFramesData))[id] ?? (_a2[id] = { ugoiraFrames: [], delays: [] }); const { ugoiraFrames, delays } = __privateGet(this, _ugoiraFramesData)[id]; if (order === undefined) { const length = frames.length; ugoiraFrames[length] = frame; delays[length] = delay; } else { ugoiraFrames[order] = frame; delays[order] = delay; } } clearFrames(taskId) { taskId in __privateGet(this, _ugoiraFramesData) && delete __privateGet(this, _ugoiraFramesData)[taskId]; } framesCount(taskId) { return taskId in __privateGet(this, _ugoiraFramesData) ? __privateGet(this, _ugoiraFramesData)[taskId]["ugoiraFrames"].filter(Boolean).length : 0; } async convert(convertOption) { const { id, signal, onProgress } = convertOption; signal == null ? undefined : signal.throwIfAborted(); const result = await __privateGet(this, _queue3).add( ({ signal: signal2 }) => { signal2 == null ? undefined : signal2.throwIfAborted(); if (!(id in __privateGet(this, _ugoiraFramesData))) { throw new Error("No frame data matched with taskId: " + id); } const { ugoiraFrames, delays } = __privateGet(this, _ugoiraFramesData)[id]; if (!ugoiraFrames.length || !delays.length) { throw new Error("No frame data found in taskId: " + id); } this.clearFrames(id); onProgress == null ? undefined : onProgress(0); return __privateMethod(this, _Converter_instances, processConvert_fn).call(this, { ...convertOption, frames: ugoiraFrames, delays }); }, { signal } ); if (!result) throw new Error(`Task ${id} has no result returned.`); return result; } async appendPixivEffect(option) { const { id, signal, illust, seasonalEffect, onProgress } = option; signal == null ? undefined : signal.throwIfAborted(); const result = await __privateGet(this, _queue3).add( async ({ signal: signal2 }) => { signal2 == null ? undefined : signal2.throwIfAborted(); logger.info("Append Effect:", id); onProgress == null ? undefined : onProgress(0); const mixEffect = convertAdapter.getMixEffectFn(); const t0 = performance.now(); const { frames: frames2, delays } = await mixEffect(illust, seasonalEffect, signal2); const t1 = performance.now(); logger.info(`Effect appended: ${id} ${t1 - t0}ms.`); return __privateMethod(this, _Converter_instances, processConvert_fn).call(this, { ...option, frames: frames2, delays, signal: signal2 }); }, { signal } ); if (!result) throw new Error(`Task ${id} has no result returned.`); return result; } } _ugoiraFramesData = new WeakMap(); _queue3 = new WeakMap(); _Converter_instances = new WeakSet(); processConvert_fn = async function(processConvertOption) { const { id, qualityOption, frames: frames2, delays, signal, onProgress } = processConvertOption; logger.info("Start convert:", id); const adapter = convertAdapter.getAdapter(qualityOption); const t0 = performance.now(); const result = await adapter(frames2, delays, signal, onProgress); const t1 = performance.now(); logger.info(`Convert finished: ${id} ${t1 - t0}ms.`); return result; }; const converter = new Converter(); class PixivDownloadConfig extends MayBeMultiIllustsConfig { constructor(mediaMeta) { super(mediaMeta); __publicField(this, "ugoiraMeta"); __publicField(this, "illustType"); __publicField(this, "userId"); __publicField(this, "comment"); __publicField(this, "translatedTags"); __publicField(this, "bookmarkCount"); this.ugoiraMeta = "ugoiraMeta" in mediaMeta ? mediaMeta.ugoiraMeta : undefined; this.illustType = mediaMeta.illustType; this.userId = mediaMeta.userId; this.comment = mediaMeta.comment; this.translatedTags = mediaMeta.tagsTranslated; this.bookmarkCount = mediaMeta.bookmarkCount; } static get supportedTemplate() { return { [SupportedTemplate.ID]: "{id}", [SupportedTemplate.ARTIST]: "{artist}", [SupportedTemplate.ARTISTID]: "{artistID}", [SupportedTemplate.DATE]: "{date} {date(YYYY-MM-DD)}", [SupportedTemplate.PAGE]: "{page}", [SupportedTemplate.SCORE]: "{score}: bookmarkCount", [SupportedTemplate.TAGS]: "{tags}", [SupportedTemplate.TITLE]: "{title}" }; } getHeaders() { return { referer: "https://www.pixiv.net" }; } getZipComment() { if (!this.ugoiraMeta) return this.comment; const delays = this.ugoiraMeta.frames.map(({ delay }) => delay); return this.comment + "\n" + JSON.stringify(delays); } getTemplateData(data) { return { id: this.id, artist: this.normalizeString(this.artist) || this.userId, artistID: this.userId, date: this.createDate, score: String(this.bookmarkCount), title: this.normalizeString(this.title) || this.id, tags: this.tags.join("_"), ...data }; } handleConvertFactory(qualityOption, setProgress) { return this.handleBeforeSaveCb ?? (this.handleBeforeSaveCb = async (imgBlob, config, signal) => { signal == null ? undefined : signal.throwIfAborted(); const { taskId, src } = config; const index2 = this.src.indexOf(src); if (index2 === -1) throw new Error("No src matches."); if (!this.ugoiraMeta) return; this.handleAbortCb ?? (this.handleAbortCb = () => { converter.clearFrames(taskId); }); signal == null ? undefined : signal.addEventListener("abort", this.handleAbortCb, { once: true }); converter.addFrame({ id: taskId, frame: imgBlob, delay: this.ugoiraMeta.frames[index2]["delay"], order: index2 }); if (converter.framesCount(taskId) !== this.ugoiraMeta.frames.length) return; return await converter.convert({ id: taskId, qualityOption, onProgress: setProgress, signal }); }); } handleSeasonalEffectFactory(qualityOption, onProgress) { return this.handleBeforeSaveCb ?? (this.handleBeforeSaveCb = async (imgBlob, config, signal) => { signal == null ? undefined : signal.throwIfAborted(); const effectId = "pixivGlow2024"; const url = "https://source.pixiv.net/special/seasonal-effect-tag/pixiv-glow-2024/effect.png"; const { taskId } = config; const effectData = await historyDb.getImageEffect(effectId); if (effectData && !("width" in effectData)) { const { data } = effectData; const blob = await converter.appendPixivEffect({ id: taskId, qualityOption, illust: imgBlob, seasonalEffect: data, onProgress, signal }); return blob; } else { const effctBlob = await new Promise((resolve, reject) => { _GM_xmlhttpRequest({ url, headers: { referer: "https://www.pixiv.net" }, responseType: "blob", onload(e) { resolve(e.response); }, onerror: reject, ontimeout: () => reject(new Error("Timeout")) }); }); const blob = await converter.appendPixivEffect({ id: taskId, qualityOption, illust: imgBlob, // seasonalEffect will be transfered to worker seasonalEffect: await effctBlob.arrayBuffer(), onProgress, signal }); historyDb.addImageEffect({ id: effectId, data: await effctBlob.arrayBuffer() }); return blob; } }); } create(option) { const { filenameTemplate, directoryTemplate, setProgress, useTranslatedTags, useFileSystemAccessApi, filenameConflictAction } = option; const index2 = "index" in option ? option.index : 0; const headers = this.getHeaders(); const templateData = this.getTemplateData( useTranslatedTags ? { tags: this.translatedTags.join("_"), page: String(index2) } : { page: String(index2) } ); return { headers, taskId: this.getTaskId(), src: this.getSrc(index2), path: this.getSavePath(directoryTemplate, filenameTemplate, this.getExt(index2), templateData), timeout: this.getDownloadTimeout(index2), onProgress: setProgress, useFileSystemAccessApi, filenameConflictAction }; } createMulti(option) { if (!this.isStringArray(this.src)) throw new Error(`Artwork ${this.id} only have one media.`); const { filenameTemplate, directoryTemplate, setProgress, useTranslatedTags, useFileSystemAccessApi, filenameConflictAction } = option; const taskId = this.getTaskId(); const headers = this.getHeaders(); const onFileSaved = setProgress ? this.getMultipleMediaDownloadCB(setProgress) : undefined; const overwriteData = useTranslatedTags ? { tags: this.translatedTags.join("_") } : {}; return this.src.map((src, i) => { return { headers, taskId, src, path: this.getSavePath( directoryTemplate, filenameTemplate, this.getExt(i), this.getTemplateData({ ...overwriteData, page: String(i) }) ), timeout: this.getDownloadTimeout(), onFileSaved, useFileSystemAccessApi, filenameConflictAction }; }); } createBundle(option) { if (!this.isStringArray(this.src)) throw new Error(`Artwork ${this.id} only have one media.`); const { filenameTemplate, directoryTemplate, setProgress, useTranslatedTags, useFileSystemAccessApi, filenameConflictAction } = option; const taskId = this.getTaskId(); const headers = this.getHeaders(); const onXhrLoaded = setProgress ? this.getMultipleMediaDownloadCB(setProgress) : undefined; const overwriteData = useTranslatedTags ? { tags: this.translatedTags.join("_") } : {}; const path = this.getSavePath( directoryTemplate, filenameTemplate, "zip", this.getTemplateData({ ...overwriteData, page: String(this.src.length) }) ); const filenameTemplateWithPage = filenameTemplate.includes(`{${SupportedTemplate.PAGE}}`) ? filenameTemplate : filenameTemplate + `_{${SupportedTemplate.PAGE}}`; const filenames = this.src.map((_, i) => { return this.getSavePath( "", filenameTemplateWithPage, this.getExt(i), this.getTemplateData({ ...overwriteData, page: String(i) }) ); }); return this.src.map((src, i) => { return { headers, taskId, src, path, timeout: this.getDownloadTimeout(i), onXhrLoaded, beforeFileSave: this.handleBundleFactory(filenames), onError: this.handleBundleErrorFactory(), onAbort: this.handleBundleAbortFactory(), useFileSystemAccessApi, filenameConflictAction }; }); } createConvert(option) { if (!this.isStringArray(this.src)) throw new Error(`Artwork ${this.id} only have one media.`); const { filenameTemplate, directoryTemplate, setProgress, qualityOption, useTranslatedTags, useFileSystemAccessApi, filenameConflictAction } = option; const taskId = this.getTaskId(); const headers = this.getHeaders(); const onXhrLoaded = setProgress ? this.getMultipleMediaDownloadCB(setProgress) : undefined; const beforeFileSave = this.handleConvertFactory(qualityOption, setProgress); const templateData = this.getTemplateData( useTranslatedTags ? { tags: this.translatedTags.join("_"), page: String(0) } : { page: String(0) } ); const path = this.getSavePath( directoryTemplate, filenameTemplate, qualityOption.format, templateData ); return this.src.map((src, i) => { return { headers, taskId, src, path, timeout: this.getDownloadTimeout(i), onXhrLoaded, beforeFileSave, useFileSystemAccessApi, filenameConflictAction }; }); } createSeasonalEffect(option) { const { filenameTemplate, directoryTemplate, setProgress, qualityOption, useTranslatedTags, useFileSystemAccessApi, filenameConflictAction } = option; const index2 = "index" in option ? option.index : 0; const templateData = this.getTemplateData( useTranslatedTags ? { tags: this.translatedTags.join("_"), page: String(index2) } : { page: String(index2) } ); return { headers: this.getHeaders(), taskId: this.getTaskId(), src: this.getSrc(index2), path: this.getSavePath( directoryTemplate, filenameTemplate, qualityOption.format, templateData ), timeout: this.getDownloadTimeout(index2), onProgress: setProgress, beforeFileSave: this.handleSeasonalEffectFactory(qualityOption, setProgress), useFileSystemAccessApi, filenameConflictAction }; } } async function addBookmark(illustId, token, optionalParams) { const { btn: btn2, tags, restrict } = optionalParams; try { await pixivApi.addBookmark(illustId, token, tags, restrict); if (!btn2) return; const bookmarkBtnRef = findBookmarkBtn(btn2); 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; } default: break; } } catch (error) { logger.error(error); } } function findBookmarkBtn(btn2) { var _a2, _b2, _c2, _d2, _e; const bookmarkBtnRef = {}; if (!btn2.dataset.type) { const favBtn = (_b2 = (_a2 = btn2.parentElement) == null ? undefined : _a2.nextElementSibling) == null ? undefined : _b2.querySelector( 'button[type="button"]' ); if (favBtn) { bookmarkBtnRef.kind = "sub"; bookmarkBtnRef.button = favBtn; } else { const favBtn2 = (_c2 = btn2.parentElement) == null ? undefined : _c2.querySelector("div._one-click-bookmark"); if (favBtn2) { bookmarkBtnRef.kind = "rank"; bookmarkBtnRef.button = favBtn2; } } } else if (btn2.dataset.type === ThumbnailBtnType.PixivToolbar) { const favBtn = (_e = (_d2 = btn2.parentElement) == null ? undefined : _d2.parentElement) == null ? undefined : _e.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 isArtworkPage(illustId) { return location.pathname.includes(`/artworks/${illustId}`); } async function likeIllust(illustId, token) { var _a2, _b2, _c2; await pixivApi.likeIllust(illustId, token); if (!isArtworkPage(illustId)) return; const likeBtn = (_c2 = (_b2 = (_a2 = document.querySelector( `${ThumbnailButton.tagNameLowerCase}[data-type="pixiv-toolbar"]` )) == null ? undefined : _a2.parentElement) == null ? undefined : _b2.previousElementSibling) == null ? undefined : _c2.firstElementChild; if (!likeBtn) return; likeBtn.disabled = true; likeBtn.style.color = "#0096fa"; likeBtn.style.cursor = "default"; } class Pixiv extends SiteInject { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "pixiv/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{artist}_{title}_{id}_p{page}" ); siteFeature.$update((state2) => { return { ...state2, ugoiraFormat: legacyConfig.ugoiraFormat ?? "zip", mixSeasonalEffect: legacyConfig.mixEffect ?? false, tagLocale: legacyConfig.tagLang ?? PixivTagLocale.JAPANESE, compressMultiIllusts: legacyConfig.bundleIllusts ?? false, compressManga: legacyConfig.bundleManga ?? false, addBookmark: false, bookmarkWithTags: legacyConfig.addBookmarkWithTags ?? false, privateBookmarkIfR18: legacyConfig.privateR18 ?? false, likeIllustWhenDownloading: legacyConfig.likeIllust ?? false }; }); } const themeWatcher2 = new ReactiveValue( () => document.documentElement.getAttribute("data-theme") === "dark", (update2) => { const observer2 = new MutationObserver((records) => { const isThemeChanged = records.some((record) => record.attributeName === "data-theme"); isThemeChanged && update2(); }); observer2.observe(document.documentElement, { attributes: true, childList: false, subtree: false }); return () => { observer2.disconnect(); }; } ); clientSetting.setThemeWatcher(themeWatcher2); super(); __publicField(this, "firstObserverCbRunFlag", true); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ async avatar(url) { let userId; let matchReg; if (matchReg = regexp.series.exec(url)) { userId = matchReg[1]; } else if (matchReg = regexp.userPage.exec(url)) { userId = matchReg[1] || matchReg[2]; } else { userId = getSelfId() ?? ""; } if (!userId) return ""; try { const userData = await pixivApi.getUserData(userId); return userData.imageBig; } catch (error) { logger.error(error); return ""; } }, parseMetaByArtworkId: (id) => { const tagLang = siteFeature.tagLocale ?? PixivTagLocale.JAPANESE; return pixivParser.parse(id, { tagLang, type: "api" }); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfigs = this.getDownloadConfig(meta); await downloader.download(downloadConfigs, { signal }); const { comment: comment2, id, tags, artist, userId, title } = meta; const historyData = { pid: Number(id), user: artist, userId: Number(userId), title, comment: comment2, tags }; historyDb.add(historyData); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "illust", type: "include", name: () => t("downloader.category.filter.pixiv_illust"), checked: true, fn(meta) { return meta.illustType === IllustType.illusts; } }, { id: "manga", type: "include", name: () => t("downloader.category.filter.pixiv_manga"), checked: true, fn(meta) { return meta.illustType === IllustType.manga; } }, { id: "ugoira", type: "include", name: () => t("downloader.category.filter.pixiv_ugoira"), checked: true, fn(meta) { return meta.illustType === IllustType.ugoira; } } ], enableTagFilter: true }, pageOption: { self_bookmark_public: { name: () => t("downloader.download_type.pixiv_bookmark_public"), match(url) { const userIdMatch = regexp.userPage.exec(url); if (!userIdMatch) return false; const userId = userIdMatch[1] || userIdMatch[2]; return userId === getSelfId(); }, filterInGenerator: true, fn: (pageRange, checkValidity) => { return pixivParser.bookmarkGenerator(pageRange, checkValidity, getSelfId()); } }, self_bookmark_private: { name: () => t("downloader.download_type.pixiv_bookmark_private"), match(url) { const userIdMatch = regexp.userPage.exec(url); if (!userIdMatch) return false; const userId = userIdMatch[1] || userIdMatch[2]; return userId === getSelfId(); }, filterInGenerator: true, fn: (pageRange, checkValidity) => { return pixivParser.bookmarkGenerator(pageRange, checkValidity, getSelfId(), "hide"); } }, user_page_works: { name: () => t("downloader.download_type.pixiv_works"), match: regexp.userPage, filterInGenerator: true, fn: (pageRange, checkValidity) => { const userIdMatch = regexp.userPage.exec(location.href); const userId = userIdMatch[1] || userIdMatch[2]; return pixivParser.illustMangaGenerator(pageRange, checkValidity, userId); } }, user_page_bookmark: { name: () => t("downloader.download_type.pixiv_bookmark"), match: regexp.userPage, filterInGenerator: true, fn: (pageRange, checkValidity) => { const userIdMatch = regexp.userPage.exec(location.href); const userId = userIdMatch[1] || userIdMatch[2]; return pixivParser.bookmarkGenerator(pageRange, checkValidity, userId); } }, follow_latest_all: { name: () => t("downloader.download_type.pixiv_follow_latest_all"), match: regexp.followLatest, filterInGenerator: true, fn: (pageRange, checkValidity) => { return pixivParser.followLatestGenerator(pageRange, checkValidity, "all"); } }, follow_latest_r18: { name: () => t("downloader.download_type.pixiv_follow_latest_r18"), match: regexp.followLatest, filterInGenerator: true, fn: (pageRange, checkValidity) => { return pixivParser.followLatestGenerator(pageRange, checkValidity, "r18"); } }, series: { name: () => t("downloader.download_type.pixiv_series"), match: regexp.series, filterInGenerator: true, fn: (pageRange, checkValidity) => { const matchSeries = regexp.series.exec(location.pathname); return pixivParser.seriesGenerator(pageRange, checkValidity, matchSeries[2]); } }, tagged_artwork: { name: "tagged_artwork", match: () => false, // use for user tag download filterInGenerator: true, fn: (pageRange, checkValidity, userId, category, tag, bookmarkRest = "show") => { if (category === "bookmarks") { return pixivParser.taggedArtworkGenerator( pageRange, checkValidity, userId, category, tag, bookmarkRest ); } else { return pixivParser.taggedArtworkGenerator( pageRange, checkValidity, userId, category, tag ); } } } } })); } static get hostname() { return "www.pixiv.net"; } inject() { super.inject(); this.downloadArtwork = this.downloadArtwork.bind(this); new MutationObserver((records) => { this.injectThumbnailButtons(records); this.pageActions(); }).observe(document.body, { childList: true, subtree: true }); } getSupportedTemplate() { return PixivDownloadConfig.supportedTemplate; } injectThumbnailButtons(records) { const addedNodes = []; records.forEach((record) => { if (!record.addedNodes.length) return; record.addedNodes.forEach((node) => { if (node.nodeType === Node.ELEMENT_NODE && node.tagName.toLowerCase() !== ThumbnailButton.tagNameLowerCase && node.tagName !== "IMG") { addedNodes.push(node); } }); }); if (!addedNodes.length) return; if (this.firstObserverCbRunFlag) { createThumbnailBtn(document.querySelectorAll("a"), this.downloadArtwork); this.firstObserverCbRunFlag = false; } else { fixPixivPreviewer(addedNodes); const thumbnails = addedNodes.reduce((prev, current) => { return prev.concat( current instanceof HTMLAnchorElement ? [current] : Array.from(current.querySelectorAll("a")) ); }, []); createThumbnailBtn(thumbnails, this.downloadArtwork); } } pageActions() { var _a2; const pathname = location.pathname; let param; switch (true) { case !!(param = regexp.artworksPage.exec(pathname)): { const id = param[1]; createToolbarBtn(id, this.downloadArtwork); createWorkExpanedViewBtn(id, this.downloadArtwork); createPresentationBtn(id, this.downloadArtwork); createPreviewModalBtn(id, this.downloadArtwork); createMangaViewerBtn(id, this.downloadArtwork); break; } case regexp.userPageTags.test(pathname): { const { downloading, batchDownload } = this.useBatchDownload(); const handleDownload = (props) => { const { userId, category, tag, rest } = props; return batchDownload("tagged_artwork", userId, category, tag, rest); }; createFrequentTagBtn(downloading, handleDownload); createTagListBtn(downloading, handleDownload); break; } case regexp.historyPage.test(pathname): { createThumbnailBtn( document.querySelectorAll("span[style]._history-item"), this.downloadArtwork ); break; } case !!(param = regexp.unlisted.exec(pathname)): { const unlistedId = param[0]; const canonicalUrlEL = document.head.querySelector('link[rel="canonical"]') || document.head.querySelector( 'link[data-next-head][href^="https://www.pixiv.net/artworks/"]' ); const canonicalUrl = canonicalUrlEL == null ? undefined : canonicalUrlEL.getAttribute("href"); if (!canonicalUrl) throw new Error(`Cannot get canonical url for ${unlistedId}`); const id = (_a2 = regexp.artworksPage.exec(canonicalUrl)) == null ? undefined : _a2[1]; if (!id) throw new Error(`Cannot get artwork id for ${unlistedId}`); createUnlistedToolbar(id, this.downloadArtwork, unlistedId); createWorkExpanedViewBtn(id, this.downloadArtwork, unlistedId); createPresentationBtn(id, this.downloadArtwork, unlistedId); createPreviewModalBtn(id, this.downloadArtwork, unlistedId); createMangaViewerBtn(id, this.downloadArtwork, unlistedId); break; } } } isMultiImageMeta(meta) { return Array.isArray(meta.src) && meta.src.length > 1; } getConvertQualityOption(ugoiraFormat) { switch (ugoiraFormat) { case ConvertFormat.GIF: return { format: ConvertFormat.GIF, quality: convertSetting.gifQuality }; case ConvertFormat.PNG: return { format: ConvertFormat.PNG, cnum: convertSetting.pngColor }; case ConvertFormat.WEBM: return { format: ConvertFormat.WEBM, bitrate: convertSetting.webmBitrate }; case ConvertFormat.WEBP: { const { losslessWebp: lossless, webpQuality: quality, webpMehtod: method } = convertSetting; return { format: ConvertFormat.WEBP, lossless, quality, method }; } case ConvertFormat.MP4: return { format: ConvertFormat.MP4, bitrate: convertSetting.mp4Bitrate }; default: return; } } getDownloadConfig(meta, setProgress, page) { const { compressManga, compressMultiIllusts, mixSeasonalEffect, ugoiraFormat, tagLocale } = siteFeature; const defaultFormat = ConvertFormat.MP4; const qualityOption = this.getConvertQualityOption(ugoiraFormat); const option = { ...downloadSetting, useTranslatedTags: !!tagLocale && tagLocale !== PixivTagLocale.JAPANESE, setProgress }; if ("ugoiraMeta" in meta) { if (qualityOption) { return new PixivDownloadConfig(meta).createConvert({ ...option, qualityOption }); } return new PixivDownloadConfig(meta).createBundle(option); } if (this.isMultiImageMeta(meta)) { if (page !== undefined) { if (mixSeasonalEffect) { return new PixivDownloadConfig(meta).createSeasonalEffect({ ...option, index: page, qualityOption: qualityOption || this.getConvertQualityOption(defaultFormat) }); } return new PixivDownloadConfig(meta).create({ ...option, index: page }); } if (meta.illustType === IllustType.manga && compressManga || meta.illustType === IllustType.illusts && compressMultiIllusts) { return new PixivDownloadConfig(meta).createBundle(option); } return new PixivDownloadConfig(meta).createMulti(option); } if (mixSeasonalEffect) { return new PixivDownloadConfig(meta).createSeasonalEffect({ ...option, qualityOption: qualityOption || this.getConvertQualityOption(defaultFormat) }); } return new PixivDownloadConfig(meta).create(option); } async downloadArtwork(btn2) { const { id, page, unlistedId } = btn2.dataset; const pageNum = page !== undefined ? +page : undefined; const tagLang = siteFeature.tagLocale ?? PixivTagLocale.JAPANESE; let pixivMeta; if (!unlistedId) { const shouldAddBookmark = siteFeature.addBookmark; const shouldLikeIllust = siteFeature.likeIllustWhenDownloading; if (shouldAddBookmark || shouldLikeIllust) { pixivMeta = await pixivParser.parse(id, { tagLang, type: "html" }); const { bookmarkData, token, tags: tags2, likeData } = pixivMeta; if (!bookmarkData && shouldAddBookmark) { const addedTags = siteFeature.bookmarkWithTags ? tags2 : undefined; const restrict = siteFeature.privateBookmarkIfR18 && tags2.includes("R-18") ? BookmarkRestrict.private : BookmarkRestrict.public; addBookmark(id, token, { btn: btn2, tags: addedTags, restrict }); } if (!likeData && shouldLikeIllust) { likeIllust(id, token); } } else { pixivMeta = await pixivParser.parse(id, { tagLang, type: "api" }); } } else { pixivMeta = await pixivParser.parse(unlistedId, { tagLang, type: "unlisted" }); } const downloadConfigs = this.getDownloadConfig( pixivMeta, (progress) => { if (progress > 0) { btn2.setProgress(progress); } else { btn2.setStatus(ThumbnailBtnStatus.Loading); } }, pageNum ); await downloader.download(downloadConfigs, { priority: 1 }); const { comment: comment2, tags, artist, userId, title } = pixivMeta; const historyData = { pid: Number(id), user: artist, userId: Number(userId), title, comment: comment2, tags }; if (page !== undefined) { historyData.page = Number(page); } if (unlistedId) { historyData.unlistedId = unlistedId; } historyDb.add(historyData); } } class MoebooruParser extends ParserBase { constructor() { super(...arguments); __privateAdd(this, _MoebooruParser_instances); } parseCsrfToken() { const el = document.querySelector('meta[name="csrf-token"]'); if (!el) throw new Error("Can not find csrf-token element."); return el.content; } isLatestData(data) { return "file_ext" in data; } buildMeta(data, tagType) { if (data.status === "deleted") throw new Error(`Post ${data.id} is deleted.`); const { id, file_url, md5, created_at, source: source2, rating, score } = data; const file_ext = this.isLatestData(data) ? data.file_ext : file_url.match(/\.(\w+)$/)[1]; const artists = []; const characters = []; const tags = data.tags.split(" ").map((tag) => { const type = tagType[tag] ?? "unknown"; if (type === "artist") { artists.push(tag); } else if (type === "character") { characters.push(tag); } return type + ":" + tag; }); return { id: String(id), src: file_url, extendName: file_ext, artist: artists.join(",") || this.UNKNOWN_ARTIST, character: characters.join(",") || this.UNKNOWN_CHARACTER, title: md5, tags, createDate: new Date(created_at * 1e3).toISOString(), rating, score, source: source2 }; } isFavorite(id, votesData) { return id in votesData && votesData[id] === 3; } /** * for post_list, popular */ parsePostsList(htmlText) { return { posts: __privateMethod(this, _MoebooruParser_instances, parsePostListData_fn).call(this, htmlText), tags: __privateMethod(this, _MoebooruParser_instances, parseTagListData_fn).call(this, htmlText) }; } /** * for post/show, pool */ parsePostAndPool(htmlText) { const [dataStr] = new RegExp("(?<=Post\\.register_resp\\().+(?=\\);)").exec(htmlText); return JSON.parse(dataStr); } parseBlacklistByDoc(doc) { const el = doc.querySelector("script#user-blacklisted-tags"); if (!el) throw new Error("Can not get blacklist."); const blacklistArr = JSON.parse(el.textContent ?? "[]"); return __privateMethod(this, _MoebooruParser_instances, parseBlacklist_fn).call(this, blacklistArr); } // konachan stores blacklist as `blacklisted_tags` cookies parseBlacklistByCookie() { const blacklistMatch = document.cookie.match(new RegExp("(?<=blacklisted_tags=).+?(?=;)")); let blacklistStr = blacklistMatch ? decodeURIComponent(blacklistMatch[0]) : "[]"; if (blacklistStr === '[""]') blacklistStr = "[]"; return __privateMethod(this, _MoebooruParser_instances, parseBlacklist_fn).call(this, JSON.parse(blacklistStr)); } /** * is_blacklisted * https://github.com/moebooru/moebooru/blob/master/app/javascript/src/legacy/post.coffee#L315 */ isBlacklisted(matchTags, blacklist) { return blacklist.some((blacklistItem) => { const { require: require2, exclude } = blacklistItem; const hasTag = (tag) => matchTags.includes(tag); return require2.every(hasTag) && !exclude.some(hasTag); }); } } _MoebooruParser_instances = new WeakSet(); parsePostListData_fn = function(docText) { const matchData = docText.match(new RegExp("(?<=Post\\.register\\().+(?=\\))", "g")); return matchData.map((dataStr) => JSON.parse(dataStr)); }; parseTagListData_fn = function(docText) { const [tagStr] = docText.match(new RegExp("(?<=Post\\.register_tags\\().+(?=\\))")); return JSON.parse(tagStr); }; /** * init_blacklisted * https://github.com/moebooru/moebooru/blob/master/app/javascript/src/legacy/post.coffee#L429 */ parseBlacklist_fn = function(blacklist) { return blacklist.map((blacklist2) => { const matchRatingTag = blacklist2.replace(/(rating:[qes])\w+/, "$1"); const tags = matchRatingTag.split(" "); const require2 = []; const exclude = []; tags.forEach((tag) => { if (tag.charAt(0) === "-") { exclude.push(tag.slice(1)); } else { require2.push(tag); } }); return { tags, original_tag_string: blacklist2, require: require2, exclude }; }); }; class MoebooruApi extends ApiBase { isBadResponse(obj) { return "success" in obj && !obj.success; } async getJSON(url, init2) { const json = await super.getJSON(url, init2); if (this.isBadResponse(json)) { throw new JsonDataError(json.reason); } return json; } async getPost(id) { const [data] = await this.getJSON( `/post.json?tags=id:${id}` ); return data; } async getPosts(params) { let url = "/post.json?"; Object.entries(params).forEach(([key, val]) => { if (typeof val === "number") { val = String(val); } else if (Array.isArray(val)) { val = val.join("+"); } url += `&${key}=${val}`; }); return this.getJSON(url); } // pool only have one page async getPool(poolId, page = 1) { return this.getJSON( `/pool/show.json?id=${poolId}&page=${page}` ); } async getPopularByDate(params) { let url; const { month = "", year = "" } = params; if (params.period === "month") { url = `/post/popular_by_month.json?month=${month}&year=${year}`; } else { url = `/post/popular_by_${params.period}.json?day=${params.day ?? ""}&month=${month}&year=${year}`; } return this.getJSON(url); } async getPostHtml(id) { return this.getHtml(`/post/show/${id}`); } async getPostsHtml(tags, page) { Array.isArray(tags) && tags.join("+"); return this.getHtml(`/post?page=${page}&tags=${tags}`); } async getPopularHtmlByPeriod(period) { return this.getHtml(`/post/popular_recent?period=${period}`); } async getPoolHtml(poolId) { return this.getHtml(`/pool/show/${poolId}`); } // blacklist can be updated via ajax so we shouldn't get blacklist from current document. async getBlacklistDoc() { return this.getDoc("/static/more"); } async getPopularHtmlByDate(params) { let url; const { month = "", year = "" } = params; if (params.period === "month") { url = `/post/popular_by_month?month=${month}&year=${year}`; } else { url = `/post/popular_by_${params.period}?day=${params.day ?? ""}&month=${month}&year=${year}`; } return this.getHtml(url); } async addFavorite(id, token) { const res = await this.fetch("/post/vote.json", { method: "POST", headers: { "x-csrf-token": token, "content-type": "application/x-www-form-urlencoded; charset=UTF-8" }, body: `id=${id}&score=3` }); if (!res.ok) throw new RequestError(res.url, res.status); } } class Moebooru extends SiteInject { constructor() { if (clientSetting.version === null) { siteFeature.addBookmark ?? (siteFeature.addBookmark = false); } super(); __privateAdd(this, _Moebooru_instances); __publicField(this, "blacklist", null); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: "/favicon.ico", parseMetaByArtworkId: async (id) => { const htmlText = await this.api.getPostHtml(id); const { posts, tags } = this.parser.parsePostAndPool(htmlText); return this.parser.buildMeta(posts[0], tags); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfig = new BooruDownloadConfig(meta).create({ ...downloadSetting, cfClearance: userAuthentication.cf_clearance || undefined }); await downloader.download(downloadConfig, { signal }); const { id, tags, artist, title, rating, source: source2 } = meta; historyDb.add({ pid: Number(id), user: artist, title, tags, rating, source: source2 }); }, afterDownload: () => { this.blacklist && (this.blacklist = null); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "exclude_blacklist", type: "exclude", name: () => t("downloader.category.filter.exclude_blacklist"), checked: true, fn: async (meta) => { if (!meta.tags) return false; this.blacklist ?? (this.blacklist = await this.getBlacklist()); return this.parser.isBlacklisted(meta.tags, this.blacklist); } }, { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn() { return true; } } ], enableTagFilter: true }, pageOption: { posts: { name: () => t("downloader.download_type.moebooru_posts"), match: () => location.pathname === "/post", filterInGenerator: true, fn: (pageRange, checkValidity, tags) => { tags ?? (tags = new URLSearchParams(location.search).get("tags") ?? ""); const getPostData = async (page) => { const POSTS_PER_PAGE = 40; const htmlText = await this.api.getPostsHtml(tags, page); const { posts, tags: tagType } = this.parser.parsePostsList(htmlText); const data = posts.map((post) => ({ ...post, tagType })); return { lastPage: data.length < POSTS_PER_PAGE, data }; }; return this.parser.paginationGenerator( pageRange, getPostData, __privateMethod(this, _Moebooru_instances, buildMetaByGeneratorData_fn).bind(this), __privateMethod(this, _Moebooru_instances, validityCallbackFactory_fn).call(this, checkValidity) ); } }, popular_1d: { name: () => t("downloader.download_type.moebooru_popular_1d"), match: () => location.pathname === "/post/popular_recent", filterInGenerator: true, fn: (_, checkValidity) => { return this.parser.paginationGenerator( [1, 1], __privateMethod(this, _Moebooru_instances, getPopularDataFactory_fn).call(this, "1d"), __privateMethod(this, _Moebooru_instances, buildMetaByGeneratorData_fn).bind(this), __privateMethod(this, _Moebooru_instances, validityCallbackFactory_fn).call(this, checkValidity) ); } }, popular_1w: { name: () => t("downloader.download_type.moebooru_popular_1w"), match: () => location.pathname === "/post/popular_recent", filterInGenerator: true, fn: (_, checkValidity) => { return this.parser.paginationGenerator( [1, 1], __privateMethod(this, _Moebooru_instances, getPopularDataFactory_fn).call(this, "1w"), __privateMethod(this, _Moebooru_instances, buildMetaByGeneratorData_fn).bind(this), __privateMethod(this, _Moebooru_instances, validityCallbackFactory_fn).call(this, checkValidity) ); } }, popular_1m: { name: () => t("downloader.download_type.moebooru_popular_1m"), match: () => location.pathname === "/post/popular_recent", filterInGenerator: true, fn: (_, checkValidity) => { return this.parser.paginationGenerator( [1, 1], __privateMethod(this, _Moebooru_instances, getPopularDataFactory_fn).call(this, "1m"), __privateMethod(this, _Moebooru_instances, buildMetaByGeneratorData_fn).bind(this), __privateMethod(this, _Moebooru_instances, validityCallbackFactory_fn).call(this, checkValidity) ); } }, popular_1y: { name: () => t("downloader.download_type.moebooru_popular_1y"), match: () => location.pathname === "/post/popular_recent", filterInGenerator: true, fn: (_, checkValidity) => { return this.parser.paginationGenerator( [1, 1], __privateMethod(this, _Moebooru_instances, getPopularDataFactory_fn).call(this, "1y"), __privateMethod(this, _Moebooru_instances, buildMetaByGeneratorData_fn).bind(this), __privateMethod(this, _Moebooru_instances, validityCallbackFactory_fn).call(this, checkValidity) ); } }, popular_by_date: { name: () => t("downloader.download_type.moebooru_popular_date"), match: /\/post\/popular_by_(day|week|month)/, filterInGenerator: true, fn: (_, checkValidity) => { const period = new RegExp("(?<=popular_by_)day|week|month").exec(location.pathname)[0]; const searchParams = new URLSearchParams(location.search); let params; if (searchParams.size === 0) { params = { period }; } else { params = { period, month: searchParams.get("month") || undefined, year: searchParams.get("year") || undefined }; if (params.period !== "month") { params.day = searchParams.get("day") || undefined; } } const getPopularData = async () => { const htmlText = await this.api.getPopularHtmlByDate(params); const { posts, tags: tagType } = this.parser.parsePostsList(htmlText); const data = posts.map((post) => ({ ...post, tagType })); return { lastPage: true, data }; }; return this.parser.paginationGenerator( [1, 1], getPopularData, __privateMethod(this, _Moebooru_instances, buildMetaByGeneratorData_fn).bind(this), __privateMethod(this, _Moebooru_instances, validityCallbackFactory_fn).call(this, checkValidity) ); } }, pool: { name: () => t("downloader.download_type.moebooru_pool"), match: /\/pool\/show\//, filterInGenerator: true, fn: (_, checkValidity, poolId) => { poolId ?? (poolId = new RegExp("(?<=show\\/)[0-9]+").exec(location.pathname)[0]); const getPoolData = async () => { const htmlText = await this.api.getPoolHtml(poolId); const { posts, tags: tagType } = this.parser.parsePostAndPool(htmlText); const data = posts.map((post) => ({ ...post, tagType })); return { lastPage: true, data }; }; return this.parser.paginationGenerator( [1, 1], getPoolData, __privateMethod(this, _Moebooru_instances, buildMetaByGeneratorData_fn).bind(this), __privateMethod(this, _Moebooru_instances, validityCallbackFactory_fn).call(this, checkValidity) ); } } } })); } getSupportedTemplate() { return BooruDownloadConfig.supportedTemplate; } createThumbnailBtn(containers) { if (!containers.length) return; containers.forEach((el) => { const idMathch = new RegExp("(?<=\\/post\\/show\\/)\\d+|(?<=\\/post\\/browse#)\\d+").exec(el.href); if (!idMathch) return; const id = idMathch[0]; const oldBtn = el.querySelector(ThumbnailButton.tagNameLowerCase); if (oldBtn) { if (oldBtn.dataset.id === id) return; oldBtn.remove(); } else { if (el.href.includes("/post/show")) { el.style.height = "fit-content"; if (!el.classList.contains("thumb")) { const image = el.querySelector("img"); if (image.src.includes("blacklisted-preview.png")) return; image.onload = () => { if (image.src.includes("blacklisted-preview.png") && image.nextElementSibling) { image.nextElementSibling.remove(); } }; el.style.marginBottom = "1em"; image.style.marginBottom = "0px"; } } el.parentElement.style.display = "flex"; el.style.position = "relative"; } el.appendChild( new ThumbnailButton({ id, onClick: __privateMethod(this, _Moebooru_instances, downloadArtwork_fn).bind(this) }) ); }); } createArtworkBtn(id) { const btnContainer = document.querySelector( "#post-view > .content > div:has(:is(img, video))" ); if (!btnContainer) throw new Error("Can not find button container"); btnContainer.style.position = "relative"; btnContainer.style.width = "max-content"; btnContainer.appendChild( new ArtworkButton({ id, site: btnContainer.querySelector("video") ? "vjs_video" : "moebooru_image", onClick: __privateMethod(this, _Moebooru_instances, downloadArtwork_fn).bind(this) }) ); } createScrollerBtn() { const scrollerList = document.querySelector("ul.post-browser-posts"); if (!scrollerList) return; const ob = new MutationObserver((records) => { const containers = []; records.forEach((record) => { if (!record.addedNodes.length) return; record.addedNodes.forEach((node) => { if (!(node instanceof HTMLElement)) return; const thumbs = node.querySelectorAll("a.thumb"); if (thumbs.length) containers.push(...thumbs); }); }); this.createThumbnailBtn(containers); }); ob.observe(scrollerList, { subtree: true, childList: true }); } createImageBrowseBtn() { const postId = document.querySelector("span.post-id"); if (!postId) return; const createBtn = () => { var _a2; (_a2 = document.querySelector( `${ThumbnailButton.tagNameLowerCase}[data-type="${ThumbnailBtnType.YandeBrowse}"]` )) == null ? undefined : _a2.remove(); const id = postId.textContent; if (!id) return; document.body.appendChild( new ThumbnailButton({ id, type: ThumbnailBtnType.YandeBrowse, onClick: __privateMethod(this, _Moebooru_instances, downloadArtwork_fn).bind(this) }) ); }; createBtn(); new MutationObserver(createBtn).observe(postId, { childList: true }); } inject() { super.inject(); const pathname = location.pathname; const galleryMatch = pathname.match(new RegExp("(?<=\\/post\\/show\\/)\\d+")); if (galleryMatch) { this.createArtworkBtn(galleryMatch[0]); } else if (pathname === "/post/browse") { this.createScrollerBtn(); this.createImageBrowseBtn(); } else { const btnContainers = document.querySelectorAll( "a.thumb, div.post div.col1 a" ); this.createThumbnailBtn(Array.from(btnContainers)); } } } _Moebooru_instances = new WeakSet(); /** * register * https://github.com/moebooru/moebooru/blob/master/app/javascript/src/legacy/post.coffee#L286 */ validityCallbackFactory_fn = function(checkValidity) { return async (data) => { const tags = data.tags.split(" "); tags.push("rating:" + data.rating.charAt(0)); tags.push("status:" + data.status); return await checkValidity({ id: String(data.id), tags }) ? PostValidState.VALID : PostValidState.INVALID; }; }; buildMetaByGeneratorData_fn = function(data) { return this.parser.buildMeta(data, data.tagType); }; getPopularDataFactory_fn = function(period) { return async () => { const htmlText = await this.api.getPopularHtmlByPeriod(period); const { posts, tags: tagType } = this.parser.parsePostsList(htmlText); const data = posts.map((post) => ({ ...post, tagType })); return { lastPage: true, data }; }; }; addBookmark_fn2 = async function(id) { try { const token = this.parser.parseCsrfToken(); await this.api.addFavorite(id, token); this.toast({ message: "You have favorited this post", timeout: 2e3 }); } catch (error) { logger.error(error); this.toast({ message: error.message, type: "error" }); } }; downloadArtwork_fn = async function(btn2) { const id = btn2.dataset.id; const htmlText = await this.api.getPostHtml(id); const { posts, tags: tagType, votes } = this.parser.parsePostAndPool(htmlText); const mediaMeta = this.parser.buildMeta(posts[0], tagType); const downloadConfig = new BooruDownloadConfig(mediaMeta).create({ ...downloadSetting, cfClearance: userAuthentication.cf_clearance || undefined, setProgress: (progress) => { btn2.setProgress(progress); } }); if (siteFeature.addBookmark && !this.parser.isFavorite(id, votes)) { __privateMethod(this, _Moebooru_instances, addBookmark_fn2).call(this, id); } await downloader.download(downloadConfig, { priority: 1 }); const { tags, artist, title, rating, source: source2 } = mediaMeta; historyDb.add({ pid: Number(id), user: artist, title, tags, rating, source: source2 }); }; class Yande extends Moebooru { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "yande/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); } super(); __publicField(this, "api", new MoebooruApi()); __publicField(this, "parser", new MoebooruParser()); } async getBlacklist() { const doc = await this.api.getBlacklistDoc(); return this.parser.parseBlacklistByDoc(doc); } static get hostname() { return "yande.re"; } } class ATFbooru extends AbstractDanbooru { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "ATFbooru/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); } super(); __publicField(this, "api", new DanbooruApi()); __publicField(this, "parser", new DanbooruParser()); __publicField(this, "commentaryAccessible"); } static get hostname() { return "booru.allthefallen.moe"; } getAvatar() { return "/favicon.svg"; } // check if user has permission to access artist commentary. async isCommentaryAccessible() { try { await this.api.getArtistCommentary("703816"); } catch (error) { if (!(error instanceof RequestError)) return false; throw error; } return true; } async getMetaByPostId(id) { let mediaMeta; if (this.commentaryAccessible ?? (this.commentaryAccessible = await this.isCommentaryAccessible())) { const { post, comment: commentary } = await this.getPostAndComment(id); mediaMeta = this.parser.buildMetaByApi(post, commentary); } else { const doc = await this.api.getPostDoc(id); mediaMeta = this.parser.buildMetaByDoc(doc); } return mediaMeta; } } class Konachan extends Moebooru { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "konachan/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); userAuthentication.cf_clearance ?? (userAuthentication.cf_clearance = ""); } super(); __privateAdd(this, _Konachan_instances); __publicField(this, "api", new MoebooruApi()); __publicField(this, "parser", new MoebooruParser()); } async getBlacklist() { return this.parser.parseBlacklistByCookie(); } static get hostname() { return ["konachan.com", "konachan.net"]; } inject() { super.inject(); if (/pool\/show\//.test(location.pathname)) { __privateMethod(this, _Konachan_instances, fixPoolImageStyle_fn).call(this); } } } _Konachan_instances = new WeakSet(); fixPoolImageStyle_fn = function() { document.querySelectorAll("ul#post-list-posts li").forEach((el) => { el.style.width = "auto"; const innerEl = el.firstElementChild; innerEl.style.width = "auto"; innerEl.style.height = "auto"; }); }; class Sakugabooru extends Moebooru { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "sakugabooru/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); } super(); __publicField(this, "api", new MoebooruApi()); __publicField(this, "parser", new MoebooruParser()); } async getBlacklist() { const doc = await this.api.getBlacklistDoc(); return this.parser.parseBlacklistByDoc(doc); } static get hostname() { return "www.sakugabooru.com"; } } class Safebooru extends GelbooruV020 { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "safebooru/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); userAuthentication.cf_clearance ?? (userAuthentication.cf_clearance = ""); } super(); __publicField(this, "api", new GelbooruApiV020()); __publicField(this, "parser", new GelbooruParserV020()); } static get hostname() { return "safebooru.org"; } getThumbnailSelector() { return ".thumb:not(.blacklisted-image) > a:first-child"; } } class GelbooruApiV025 extends GelbooruApiV020 { async getPosts() { throw new Error("This is a Gelbooru v0.2.0 method, do not use."); } async getPostsV025(params) { let url = "/index.php?page=dapi&s=post&q=index&json=1"; Object.entries(params).forEach(([key, val]) => { if (typeof val === "number") { val = String(val); } else if (Array.isArray(val)) { val = val.join("+"); } url += `&${key}=${val}`; }); const res = await this.fetch(url); if (!res.ok) throw new RequestError(url, res.status); const postsListData = await res.json(); if (postsListData.post) { return postsListData.post; } else { return []; } } } class Gelbooru extends GelbooruV020 { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "gelbooru/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); } clientSetting.setThemeWatcher({ get current() { return !!document.head.querySelector('link[href*="dark.css"]'); } }); super(); __publicField(this, "api", new GelbooruApiV025()); __publicField(this, "parser", new GelbooruParserV020()); } static get hostname() { return "gelbooru.com"; } getAvatar() { return "/user_avatars/honkonymous.png"; } getThumbnailSelector() { return '.thumb > a:first-child, .thumbnail-preview > a[id], .commentThumbnail > a:first-child, .thumbnail-container > span[id] > a, a:has(img[style*="max-height: 100px"]), a:has(div.profileThumbnailPadding)'; } setThumbnailStyle(btnContainer) { switch (true) { case this.isMyfavorites(): { btnContainer.setAttribute("style", "position: relative; display: inline-block;"); break; } case this.isPool(): { btnContainer.setAttribute( "style", "position: relative; display: inline-block; margin: 10px;" ); const image = btnContainer.querySelector("img"); image.style.margin = "0px"; image.style.padding = "0px"; break; } case this.isPostsList(): { btnContainer.style.position = "relative"; break; } case this.isPostView(): { btnContainer.setAttribute( "style", "position: relative; display: inline-block; margin: 10px;" ); const image = btnContainer.querySelector("img"); image.style.margin = "0px"; break; } case this.isAccountProfile(): { btnContainer.setAttribute("style", "position: relative; display: inline-block"); btnContainer.firstElementChild.style.padding = "0px"; break; } } } createArtworkBtn(id) { let btnContainer = document.querySelector(".image-container > picture"); if (btnContainer) { btnContainer.setAttribute("style", "position: relative; display: inline-block;"); } else { const video = document.querySelector("#gelcomVideoPlayer"); if (!video) return; btnContainer = document.createElement("div"); btnContainer.setAttribute("style", "position: relative; width: fit-content; font-size: 0px;"); video.parentElement.insertBefore(btnContainer, video); btnContainer.appendChild(video); } btnContainer.appendChild( new ArtworkButton({ id, site: btnContainer.nodeName === "DIV" ? "native_video" : undefined, onClick: this.downloadArtwork.bind(this) }) ); } inject() { super.inject(); if (this.isPostView()) this.createThumbnailBtn(); } } class E621ngApi extends ApiBase { constructor(option) { super(option); __privateAdd(this, _authParams); const [username, apiKey] = option.authorization; const UA = `Pixiv Downloader/${"1.10.0"} (by drunkg00se on e621)`; __privateSet(this, _authParams, new URLSearchParams({ username, apiKey, _client: UA })); } get username() { return __privateGet(this, _authParams).get("username"); } set username(username) { username !== __privateGet(this, _authParams).get("username") && __privateGet(this, _authParams).set("username", username); } get apiKey() { return __privateGet(this, _authParams).get("apiKey"); } set apiKey(apiKey) { apiKey !== __privateGet(this, _authParams).get("apiKey") && __privateGet(this, _authParams).set("apiKey", apiKey); } async getJSON(url, init2) { const fullUrl = new URL(url, location.origin); url += fullUrl.search === "" ? "?" : "&"; url += __privateGet(this, _authParams).toString(); logger.info("Fetch url:", url); const res = await this.fetch(url, init2); if (!res.ok) throw new RequestError(res.url, res.status); return await res.json(); } getFavorites(params) { const searchParams = new URLSearchParams( Object.entries(params).map(([key, val]) => [key, String(val)]) ); return this.getJSON(`/favorites.json?${searchParams.toString()}`); } getPool(poolId) { return this.getJSON(`/pools/${poolId}.json`); } getPosts(params) { const { limit, page } = params; if (limit < 0 || limit > 320) throw new RangeError("limit should between 0 and 320."); if (page < 1 || page > 750) throw new RangeError("Page should between 1 and 750."); const searchParams = new URLSearchParams( Object.entries(params).map( ([key, val]) => typeof val === "string" ? [key, val] : [key, String(val)] ) ); return this.getJSON(`/posts.json?${searchParams.toString()}`); } getPost(id) { return this.getJSON(`/posts/${id}.json`); } getCurrentUserProfile(userId) { return this.getJSON(`/users/${userId}.json`); } async addFavorites(postId, token) { const res = await this.fetch(`/favorites.json?${__privateGet(this, _authParams).toString()}`, { method: "POST", headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8", "x-csrf-token": token }, body: `post_id=${postId}` }); if (!res.ok) throw new RequestError("/favorites.json", res.status); return res.json(); } } _authParams = new WeakMap(); class E621ngParser extends ParserBase { buildMeta(postData) { const { id, file, tags: fullTags, description, created_at, rating, sources, score, is_favorited: isFavorited } = postData; const { ext, url, md5 } = file; if (!url) throw new Error(`Url can not be null: Post ${id}`); const tags = []; for (const [type, tagArr] of Object.entries(fullTags)) { tagArr.forEach((tag) => tags.push(`${type}:${tag}`)); } return { id: String(id), src: url, extendName: ext, artist: ("artist" in fullTags ? fullTags.artist : fullTags.director).join(",") || this.UNKNOWN_ARTIST, character: fullTags.character.join(",") || this.UNKNOWN_CHARACTER, title: md5, comment: description, tags, createDate: created_at, score: score.total, source: sources.join("\n"), rating, isFavorited }; } parseCsrfToken() { var _a2; return (_a2 = document.head.querySelector('meta[name="csrf-token"]')) == null ? undefined : _a2.content; } parseCurrentUserId() { var _a2; return (_a2 = document.head.querySelector('meta[name="current-user-id"]')) == null ? undefined : _a2.content; } } class E621ng extends SiteInject { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "e621/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{id}_{artist}_{character}" ); siteFeature.addBookmark ?? (siteFeature.addBookmark = false); userAuthentication.$update((state2) => { return { ...state2, apiKey: "", username: "" }; }); } super(); __privateAdd(this, _E621ng_instances); __publicField(this, "api", new E621ngApi({ rateLimit: 2, authorization: [userAuthentication.username ?? "", userAuthentication.apiKey ?? ""] })); __publicField(this, "parser", new E621ngParser()); __publicField(this, "profile", null); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: "/packs/static/main-logo-2653c015c5870ec4ff08.svg", parseMetaByArtworkId: async (id) => { const { post } = await this.api.getPost(+id); return this.parser.buildMeta(post); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfig = new BooruDownloadConfig(meta).create({ ...downloadSetting }); await downloader.download(downloadConfig, { priority: 1, signal }); const { tags, artist, title, comment: comment2, source: source2, rating } = meta; historyDb.add({ pid: Number(meta.id), user: artist, title, comment: comment2, tags, source: source2, rating }); }, beforeDownload: async () => { __privateMethod(this, _E621ng_instances, throwIfNotAuthorized_fn).call(this); const userId = this.parser.parseCurrentUserId(); if (!userId) throw new Error("Cannot get user id."); this.profile = await this.api.getCurrentUserProfile(+userId); }, afterDownload: () => { this.profile = null; }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn(meta) { return ( // https://e621.net/help/supported_filetypes !!meta.extendName && /bmp|jp(e)?g|png|tif|gif|exif|svg|webp/i.test(meta.extendName) ); } }, { id: "allow_video", type: "include", name: () => t("downloader.category.filter.video"), checked: true, fn(meta) { return !!meta.extendName && /mp4|avi|mov|mkv|flv|wmv|webm|mpeg|mpg|m4v/i.test(meta.extendName); } } ], enableTagFilter: true }, pageOption: { pool: { name: "Pool", match: () => __privateMethod(this, _E621ng_instances, isPoolView_fn).call(this), filterInGenerator: true, fn: (pageRange, checkValidity) => { var _a2; const poolId = (_a2 = new RegExp("(?<=\\/pools\\/)[0-9]+").exec(location.pathname)) == null ? undefined : _a2[0]; if (!poolId) throw new Error("Invalid pool id"); const postsPerPage = this.profile.per_page; const getPostsMetaByPage = async (page) => { const data = (await this.api.getPosts({ limit: postsPerPage, page, tags: `pool:${poolId} order:id` })).posts; return { lastPage: data.length < postsPerPage, data }; }; return this.parser.paginationGenerator( pageRange, getPostsMetaByPage, (data) => this.parser.buildMeta(data), __privateMethod(this, _E621ng_instances, validityCallbackFactory_fn2).call(this, checkValidity) ); } }, post_list: { name: "Posts", match: () => __privateMethod(this, _E621ng_instances, isPostsPage_fn).call(this), filterInGenerator: true, fn: (pageRange, checkValidity) => { const searchParam = new URLSearchParams(new URL(location.href).search); const tags = searchParam.get("tags") || ""; const limit = +(searchParam.get("limit") || this.profile.per_page); const getPostsMetaByPage = async (page) => { const data = (await this.api.getPosts({ limit, page, tags })).posts; return { lastPage: data.length < limit, data }; }; return this.parser.paginationGenerator( pageRange, getPostsMetaByPage, (data) => this.parser.buildMeta(data), __privateMethod(this, _E621ng_instances, validityCallbackFactory_fn2).call(this, checkValidity) ); } }, favorites: { name: "Favorites", match: () => __privateMethod(this, _E621ng_instances, isFavoritesPage_fn).call(this), filterInGenerator: true, fn: (pageRange, checkValidity) => { const searchParam = new URLSearchParams(new URL(location.href).search); const limit = +(searchParam.get("limit") || this.profile.per_page); const userId = +(searchParam.get("user_id") || this.profile.id); if (!userId) throw new Error("Cannot get user id."); const getPostsMetaByPage = async (page) => { const data = (await this.api.getFavorites({ limit, page, user_id: userId })).posts; return { lastPage: data.length < limit, data }; }; return this.parser.paginationGenerator( pageRange, getPostsMetaByPage, (data) => this.parser.buildMeta(data), __privateMethod(this, _E621ng_instances, validityCallbackFactory_fn2).call(this, checkValidity) ); } }, pool_gallery_button: { name: "pool_gallery_button", match: () => false, filterInGenerator: true, fn: (pageRange, checkValidity, poolId) => { if (!poolId) throw new Error("Invalid pool id"); const getPostsMetaByPage = async (page) => { const limit = this.profile.per_page; const data = (await this.api.getPosts({ limit, page, tags: `pool:${poolId}` })).posts; return { lastPage: data.length < limit, data }; }; return this.parser.paginationGenerator( pageRange, getPostsMetaByPage, (data) => this.parser.buildMeta(data), __privateMethod(this, _E621ng_instances, validityCallbackFactory_fn2).call(this, checkValidity) ); } }, show_downloader_in_pool_gallery: { name: "pool_gallery", match: /\/pools\/gallery/ } } })); toStore(() => [userAuthentication.username, userAuthentication.apiKey]).subscribe( ([username, apiKey]) => { this.api.username = username; this.api.apiKey = apiKey; } ); } static get hostname() { return ["e621.net", "e926.net", "e6ai.net"]; } getSupportedTemplate() { return BooruDownloadConfig.supportedTemplate; } async downloadArtwork(btn2) { __privateMethod(this, _E621ng_instances, throwIfNotAuthorized_fn).call(this); const id = +btn2.dataset.id; const { post } = await this.api.getPost(id); const mediaMeta = this.parser.buildMeta(post); const downloadConfig = new BooruDownloadConfig(mediaMeta).create({ ...downloadSetting, setProgress: (progress) => { btn2.setProgress(progress); } }); if (siteFeature.addBookmark && !post.is_favorited) { __privateMethod(this, _E621ng_instances, addFavorites_fn).call(this, id); } await downloader.download(downloadConfig, { priority: 1 }); const { tags, artist, title, comment: comment2, source: source2, rating } = mediaMeta; historyDb.add({ pid: id, user: artist, title, comment: comment2, tags, source: source2, rating }); } createArtworkBtn() { const btnContainer = document.querySelector("#image-container"); if (!btnContainer) return; btnContainer.style.width = "fit-content"; btnContainer.style.position = "relative"; const id = btnContainer.dataset.id; btnContainer.appendChild( new ArtworkButton({ id, site: btnContainer.querySelector("video") ? "native_video" : undefined, onClick: this.downloadArtwork }) ); } createPoolThumbnailBtn() { const btnContainers = document.querySelectorAll("article.thumbnail > a"); if (!btnContainers.length) return; btnContainers.forEach((el) => { var _a2; const poolId = (_a2 = new RegExp("(?<=\\/pools\\/)[0-9]+").exec(el.href)) == null ? undefined : _a2[0]; if (!poolId) return; const { downloading, batchDownload } = this.useBatchDownload(); const onClick = (btn22) => { const poolId2 = btn22.dataset.id; return batchDownload("pool_gallery_button", poolId2); }; const btn2 = new DanbooruPoolButton({ id: poolId, downloading, onClick }); el.style.position = "relative"; el.appendChild(btn2); }); } createThumbnailBtn() { const btnContainers = document.querySelectorAll("article.thumbnail > a"); if (!btnContainers.length) return; btnContainers.forEach((el) => { var _a2; const id = (_a2 = new RegExp("(?<=\\/posts\\/)[0-9]+").exec(el.href)) == null ? undefined : _a2[0]; if (!id) return; el.style.position = "relative"; const btn2 = new ThumbnailButton({ id, onClick: this.downloadArtwork }); el.appendChild(btn2); }); } inject() { super.inject(); this.downloadArtwork = this.downloadArtwork.bind(this); if (__privateMethod(this, _E621ng_instances, isPostView_fn).call(this)) { this.createArtworkBtn(); } else if (__privateMethod(this, _E621ng_instances, isPoolGallery_fn).call(this)) { this.createPoolThumbnailBtn(); } else { this.createThumbnailBtn(); } } } _E621ng_instances = new WeakSet(); isPoolGallery_fn = function() { return location.pathname === "/pools/gallery"; }; isPoolView_fn = function() { return /\/pools\/[0-9]+/.test(location.pathname); }; isPostView_fn = function() { return /\/posts\/[0-9]+/.test(location.pathname); }; isFavoritesPage_fn = function() { return location.pathname === "/favorites"; }; isPostsPage_fn = function() { return location.pathname === "/posts"; }; isAuthorized_fn = function() { return this.api.username && this.api.apiKey; }; throwIfNotAuthorized_fn = function() { if (!__privateMethod(this, _E621ng_instances, isAuthorized_fn).call(this)) { const message2 = "Please input your username and apiKey in setting."; this.toast({ message: message2, type: "error" }); throw new Error(message2); } }; validityCallbackFactory_fn2 = function(checkValidity) { return async (data) => { const { id, file, tags: fullTags } = data; const tags = []; for (const tagArr of Object.values(fullTags)) { tagArr.forEach((tag) => { tags.push(tag); }); } return await checkValidity({ id: String(id), extendName: file.ext, tags }) ? PostValidState.VALID : PostValidState.INVALID; }; }; addFavorites_fn = async function(id) { try { const csrfToken = this.parser.parseCsrfToken(); if (!csrfToken) throw new Error("Cannot get csrf-token."); await this.api.addFavorites(id, csrfToken); this.toast({ message: "You have favorited this post", timeout: 2e3 }); } catch (error) { logger.error(error); this.toast({ message: error.message, type: "error" }); } }; class NijieParser extends ParserBase { constructor() { super(...arguments); __privateAdd(this, _NijieParser_instances); } buildMetaByView(id, doc) { var _a2, _b2, _c2, _d2; const [title, artist] = (((_a2 = doc.querySelector('meta[property="og:title"]')) == null ? undefined : _a2.content) ?? "").split(" | "); const comment2 = ((_b2 = doc.querySelector( "#illust_text p, #dojin_text p:not(.title), #view-honbun > p.m-bottom15:not(.gray)" )) == null ? undefined : _b2.textContent) ?? ""; const src = doc.querySelector( "#img_filter :is(img, video), p.image img, #gallery_new img#view_img" ).src; const userId = (_d2 = (_c2 = doc.querySelector('a[href*="members_illust"]')) == null ? undefined : _c2.href.match(new RegExp("(?<=id=)[0-9]+$"))) == null ? undefined : _d2[0]; if (!title || !artist || !src || !userId) throw new Error("Can not parse necessary data"); const matchExt = src.match(new RegExp("(?<=\\.)[a-z0-9]{3,4}$", "i")); if (!matchExt) throw new Error("Can not parse ext."); const postDateMatch = /[0-9:\- ]+$/.exec( doc.querySelector("#view-honbun > p, #created").textContent ); const goodCount = doc.querySelector("#good_cnt").textContent ?? 0; const tags = Array.from(doc.querySelectorAll("[tag_id].tag .tag_name")).map( (el) => el.textContent ); const isBookmarked = !!doc.querySelector('a[href*="bookmark_edit"]'); return { id, userId, src, extendName: matchExt[0], artist, title, tags, createDate: postDateMatch[0], comment: comment2, score: +goodCount, isBookmarked }; } parseDiffSrcByDoc(doc) { return Array.from( doc.querySelectorAll( '#img_filter :is(img[src*="pic.nijie.net"], video)' ) ).map((el) => { const src = el.src; const matchExt = src.match(new RegExp("(?<=\\.)[a-z0-9]{3,4}$", "i")); if (!matchExt) throw new Error("Can not parse ext."); return { src, extendName: matchExt[0] }; }); } mergeImageDiff(meta, diffs) { const src = []; const extendName = []; for (const diff of diffs) { src.push(diff.src); extendName.push(diff.extendName); } return { ...meta, src, extendName }; } parseUserPageArtworkIdByDoc(doc) { const thumbnails = doc.querySelectorAll( '.mem-index .nijiedao > a[href^="/view.php?id="]' ); return __privateMethod(this, _NijieParser_instances, parseIdByAnchors_fn).call(this, Array.from(thumbnails)); } parseUserFeedArtworkIdByDoc(doc) { const thumbnails = doc.querySelectorAll( '#main-left-main [illust_id] .picture > a[href^="/view.php?id="]' ); return __privateMethod(this, _NijieParser_instances, parseIdByAnchors_fn).call(this, Array.from(thumbnails)); } parseHistoryIllustArtworkIdByDoc(doc) { const thumbnails = doc.querySelectorAll( '.history_block > .picture > a[href*="id="]:has(img)' ); return __privateMethod(this, _NijieParser_instances, parseIdByAnchors_fn).call(this, Array.from(thumbnails)); } parseHistoryNuitaArtworkIdByDoc(doc) { const thumbnails = doc.querySelectorAll( '#center_column div[illust_id].illust_list .picture a[href*="id="]:has(img)' ); return __privateMethod(this, _NijieParser_instances, parseIdByAnchors_fn).call(this, Array.from(thumbnails)); } parseOkiniiriArtworkIdByDoc(doc) { const thumbnails = doc.querySelectorAll( '#content_delete .picture a[href*="id="]:has(img)' ); return __privateMethod(this, _NijieParser_instances, parseIdByAnchors_fn).call(this, Array.from(thumbnails)); } parseSearchArtworkIdByDoc(doc) { const thumbnails = doc.querySelectorAll( '#main-left-main [illust_id] .picture a[href*="id="]:has(img)' ); return __privateMethod(this, _NijieParser_instances, parseIdByAnchors_fn).call(this, Array.from(thumbnails)); } docHasDiff(doc) { return !!doc.querySelector('a[href*="#diff_"]'); } docIsDojin(doc) { return !!doc.querySelector("#dojin_left"); } docHasNextPagination(doc) { return !!doc.querySelector('.page_button > a[rel="next"]'); } } _NijieParser_instances = new WeakSet(); parseIdByAnchors_fn = function(elems) { if (!elems.length) return []; return elems.map((el) => { const idMatch = new RegExp("(?<=id=)[0-9]+$").exec(el.href); return idMatch[0]; }); }; class NijieApi extends ApiBase { getViewDoc(id) { return this.getDoc(`/view.php?id=${id}`); } getViewPopupDoc(id) { return this.getDoc(`/view_popup.php?id=${id}`); } getUserIllustsDoc(id, page) { return this.getDoc(`/members_illust.php?id=${id}&p=${page}`); } getUserDojinDoc(id) { return this.getDoc(`/members_dojin.php?id=${id}`); } getUserBookmarkDoc(id, page) { return this.getDoc(`/user_like_illust_view.php?p=${page}&id=${id}`); } getUserFeedDoc(page) { return this.getDoc(`/like_user_view.php?p=${page}`); } getHistoryIllustDoc() { return this.getDoc("/history_illust.php"); } getHistoryNuitaDoc() { return this.getDoc("/history_nuita.php"); } /** * @param id - The id of tag. * @param page * @param sort - 0: 新しくブクマした順; 1: 古くブクマした順 */ getOkiniiriDoc(id, page, sort) { return this.getDoc(`/okiniiri.php?p=${page}&id=${id}&sort=${sort}`); } getIllustSearchDoc(params) { const { mode = 0, type = "partial", sort = 0, illustType = 0, period = 0, userId = "0", word, page } = params; const searchParams = new URLSearchParams(); searchParams.append("mode", String(mode)); searchParams.append("type", String(type)); searchParams.append("sort", String(sort)); searchParams.append("illust_type", String(illustType)); searchParams.append("period", String(period)); searchParams.append("user_id", userId); searchParams.append("word", word); searchParams.append("p", String(page)); return this.getDoc(`/search.php?${searchParams.toString()}`); } async addBookmark(id, tags) { const params = new URLSearchParams(); params.append("id", id); params.append("tag", (tags == null ? undefined : tags.join(" ")) ?? ""); const url = "/bookmark_add.php"; const res = await this.fetch(url, { method: "POST", redirect: "manual", body: params }); if (res.type !== "opaqueredirect") throw new RequestError(url, res.status); } } class NijieDownloadConfig extends MayBeMultiIllustsConfig { constructor(meta) { super(meta); __publicField(this, "userId"); __publicField(this, "comment"); __publicField(this, "score"); this.userId = meta.userId; this.comment = meta.comment; this.score = meta.score; } static get supportedTemplate() { return { [SupportedTemplate.ID]: "{id}", [SupportedTemplate.ARTIST]: "{artist}", [SupportedTemplate.ARTISTID]: "{artistID}", [SupportedTemplate.DATE]: "{date} {date(YYYY-MM-DD)}", [SupportedTemplate.PAGE]: "{page}", [SupportedTemplate.SCORE]: "{score}: likeCount", [SupportedTemplate.TAGS]: "{tags}", [SupportedTemplate.TITLE]: "{title}" }; } getZipComment() { return this.comment; } getTemplateData(data) { return { id: this.id, artist: this.normalizeString(this.artist) || this.userId, artistID: this.userId, date: this.createDate, score: String(this.score), title: this.normalizeString(this.title) || this.id, tags: this.tags.join("_"), ...data }; } create(option) { const { filenameTemplate, directoryTemplate, setProgress, useFileSystemAccessApi, filenameConflictAction } = option; const index2 = "index" in option ? option.index : 0; return { taskId: this.getTaskId(), src: this.getSrc(index2), path: this.getSavePath( directoryTemplate, filenameTemplate, this.getExt(index2), this.getTemplateData({ page: String(index2) }) ), timeout: this.getDownloadTimeout(index2), onProgress: setProgress, useFileSystemAccessApi, filenameConflictAction }; } createMulti(option) { if (!this.isStringArray(this.src)) throw new Error(`Artwork ${this.id} only have one media.`); const { filenameTemplate, directoryTemplate, setProgress, useFileSystemAccessApi, filenameConflictAction } = option; const taskId = this.getTaskId(); const onFileSaved = setProgress ? this.getMultipleMediaDownloadCB(setProgress) : undefined; return this.src.map((src, i) => { return { taskId, src, path: this.getSavePath( directoryTemplate, filenameTemplate, this.getExt(i), this.getTemplateData({ page: String(i) }) ), timeout: this.getDownloadTimeout(), onFileSaved, useFileSystemAccessApi, filenameConflictAction }; }); } createBundle(option) { if (!this.isStringArray(this.src) || !this.isStringArray(this.ext)) throw new Error(`Artwork ${this.id} only have one media.`); const { filenameTemplate, directoryTemplate, setProgress, useFileSystemAccessApi, filenameConflictAction } = option; const taskId = this.getTaskId(); const onXhrLoaded = setProgress ? this.getMultipleMediaDownloadCB(setProgress) : undefined; const path = this.getSavePath( directoryTemplate, filenameTemplate, "zip", this.getTemplateData({ page: String(this.src.length) }) ); const filenameTemplateWithPage = filenameTemplate.includes(`{${SupportedTemplate.PAGE}}`) ? filenameTemplate : filenameTemplate + `_{${SupportedTemplate.PAGE}}`; const filenames = this.src.map((_, i) => { return this.getSavePath( "", filenameTemplateWithPage, this.getExt(i), this.getTemplateData({ page: String(i) }) ); }); return this.src.map((src, i) => { return { taskId, src, path, timeout: this.getDownloadTimeout(i), onXhrLoaded, beforeFileSave: this.handleBundleFactory(filenames), onError: this.handleBundleErrorFactory(), onAbort: this.handleBundleAbortFactory(), useFileSystemAccessApi, filenameConflictAction }; }); } } class Nijie extends SiteInject { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate(legacyConfig.folderPattern ?? "nijie/{artist}"); downloadSetting.setFilenameTemplate( legacyConfig.filenamePattern ?? "{artist}_{title}_{id}_p{page}" ); siteFeature.$update((state2) => { return { ...state2, compressMultiIllusts: legacyConfig.bundleIllusts ?? false, addBookmark: state2.addBookmark ?? false, bookmarkWithTags: state2.bookmarkWithTags ?? false }; }); } clientSetting.setThemeWatcher({ get current() { return !!document.querySelector('link[type="text/css"][href*="night_mode"]'); } }); super(); __privateAdd(this, _Nijie_instances); __publicField(this, "parser", new NijieParser()); __publicField(this, "api", new NijieApi({ rateLimit: 3 })); __privateAdd(this, _searchParams, new URLSearchParams(location.search)); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: () => { const userAvatarImg = document.querySelector( 'a[href*="members.php"].name img' ); return userAvatarImg ? userAvatarImg.src : "/pic/icon/nijie.png"; }, parseMetaByArtworkId: async (id) => { const viewDoc = await this.api.getViewDoc(id); const meta = this.parser.buildMetaByView(id, viewDoc); if (this.parser.docHasDiff(viewDoc)) { const popupDoc = await this.api.getViewPopupDoc(id); const imgDiffSrcs = this.parser.parseDiffSrcByDoc(popupDoc); return this.parser.mergeImageDiff(meta, imgDiffSrcs); } return meta; }, downloadArtworkByMeta: async (meta, signal) => { let downloadConfig; const option = { ...downloadSetting }; const bundleIllusts = siteFeature.compressMultiIllusts; if (Array.isArray(meta.src)) { downloadConfig = bundleIllusts ? new NijieDownloadConfig(meta).createBundle(option) : new NijieDownloadConfig(meta).createMulti(option); } else { downloadConfig = new NijieDownloadConfig(meta).create(option); } await downloader.download(downloadConfig, { signal }); const { id, artist, userId, title, comment: comment2, tags } = meta; const historyData = { pid: Number(id), user: artist, userId: Number(userId), title, comment: comment2, tags }; historyDb.add(historyData); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, // nijie post may contain both image and video. { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn(meta) { if (meta.extendName === undefined) return false; if (Array.isArray(meta.extendName)) { return meta.extendName.some((extendName) => regexp.imageExt.test(extendName)); } return regexp.imageExt.test(meta.extendName); } }, { id: "allow_video", type: "include", name: () => t("downloader.category.filter.video"), checked: true, fn(meta) { if (meta.extendName === undefined) return false; if (Array.isArray(meta.extendName)) { return meta.extendName.some((extendName) => regexp.videoExt.test(extendName)); } return regexp.videoExt.test(meta.extendName); } } ], enableTagFilter: true }, pageOption: { illusts: { name: "投稿イラスト", match: (url) => __privateMethod(this, _Nijie_instances, isSupportedUserPage_fn).call(this, url), filterInGenerator: false, fn: (pageRange) => { const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); if (!id) throw new Error("Invalid user ID."); const getIllustData = async (page) => { const doc = await this.api.getUserIllustsDoc(id, page); return { lastPage: !this.parser.docHasNextPagination(doc), data: this.parser.parseUserPageArtworkIdByDoc(doc) }; }; return this.parser.paginationGenerator(pageRange, getIllustData, (data) => data); } }, // dojin only have one page dojin: { name: "同人", match: (url) => __privateMethod(this, _Nijie_instances, isSupportedUserPage_fn).call(this, url), filterInGenerator: false, fn: () => { const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); if (!id) throw new Error("Invalid user ID."); return this.parser.paginationGenerator( [1, 1], async () => { const doc = await this.api.getUserDojinDoc(id); return { lastPage: true, data: this.parser.parseUserPageArtworkIdByDoc(doc) }; }, (data) => data ); } }, // user_like_illust_view may not always have 48 illusts per page. bookmark: { name: "ブックマーク", match: (url) => __privateMethod(this, _Nijie_instances, isSupportedUserPage_fn).call(this, url), filterInGenerator: false, fn: (pageRange) => { const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); if (!id) throw new Error("Invalid user ID."); return this.parser.paginationGenerator( pageRange, async (page) => { const doc = await this.api.getUserBookmarkDoc(id, page); return { lastPage: !this.parser.docHasNextPagination(doc), data: this.parser.parseUserPageArtworkIdByDoc(doc) }; }, (data) => data ); } }, feed: { name: "新着2次絵", match: /like_user_view\.php/, filterInGenerator: false, fn: (pageRange) => { return this.parser.paginationGenerator( pageRange, async (page) => { const doc = await this.api.getUserFeedDoc(page); return { lastPage: !this.parser.docHasNextPagination(doc), data: this.parser.parseUserFeedArtworkIdByDoc(doc) }; }, (data) => data ); } }, nuita: { name: "抜いた", match: (url) => __privateMethod(this, _Nijie_instances, isSupportedHistoryPage_fn).call(this, url), filterInGenerator: false, fn: () => { return this.parser.paginationGenerator( [1, 1], async () => { const doc = await this.api.getHistoryNuitaDoc(); return { lastPage: true, data: this.parser.parseHistoryNuitaArtworkIdByDoc(doc) }; }, (data) => data ); } }, history: { name: "閲覧", match: (url) => __privateMethod(this, _Nijie_instances, isSupportedHistoryPage_fn).call(this, url), filterInGenerator: false, fn: () => { return this.parser.paginationGenerator( [1, 1], async () => { const doc = await this.api.getHistoryIllustDoc(); return { lastPage: true, data: this.parser.parseHistoryIllustArtworkIdByDoc(doc) }; }, (data) => data ); } }, okiniiri: { name: "お気に入り", match: /okiniiri\.php/, filterInGenerator: false, fn: (pageRange) => { const tagId = __privateGet(this, _searchParams).get("id") ?? "0"; const sort = __privateGet(this, _searchParams).get("sort") ?? "0"; if (sort !== "0" && sort !== "1") throw new RangeError('Invalid sort params, must be "0" or "1"'); return this.parser.paginationGenerator( pageRange, async (page) => { const doc = await this.api.getOkiniiriDoc(tagId, String(page), sort); return { lastPage: !this.parser.docHasNextPagination(doc), data: this.parser.parseOkiniiriArtworkIdByDoc(doc) }; }, (data) => data ); } }, userIllustTagSearch: { name: "イラスト検索", match: /search\.php.+user_id=[0-9]+/, filterInGenerator: false, fn: (pageRange) => { const mode = Number(__privateGet(this, _searchParams).get("mode")); const type = __privateGet(this, _searchParams).get("type") || undefined; const sort = Number(__privateGet(this, _searchParams).get("sort")); const illustType = Number( __privateGet(this, _searchParams).get("illust_type") ); const period = Number(__privateGet(this, _searchParams).get("p")); const userId = __privateGet(this, _searchParams).get("user_id"); const word = __privateGet(this, _searchParams).get("word") ?? ""; if (!userId) throw new Error("User id is required."); return this.parser.paginationGenerator( pageRange, async (page) => { const doc = await this.api.getIllustSearchDoc({ mode, type, sort, illustType, period, userId, word, page }); return { lastPage: !this.parser.docHasNextPagination(doc), data: this.parser.parseSearchArtworkIdByDoc(doc) }; }, (data) => data ); } } } })); } static get hostname() { return "nijie.info"; } getSupportedTemplate() { return NijieDownloadConfig.supportedTemplate; } async downloadArtwork(btn2) { const { id, page } = btn2.dataset; let viewDoc; let popupDoc; if (__privateMethod(this, _Nijie_instances, isViewPage_fn).call(this) && id === __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this)) { viewDoc = document; } else { viewDoc = await this.api.getViewDoc(id); } const meta = this.parser.buildMetaByView(id, viewDoc); const { userId, comment: comment2, tags, artist, title, isBookmarked } = meta; if (!isBookmarked && siteFeature.addBookmark) { __privateMethod(this, _Nijie_instances, addBookmark_fn3).call(this, id, siteFeature.bookmarkWithTags ? tags : undefined); } const option = { ...downloadSetting, setProgress: (progress) => { btn2.setProgress(progress); } }; const pageNum = page ? +page : undefined; let downloadConfig; if (pageNum === 0 || !this.parser.docHasDiff(viewDoc)) { downloadConfig = new NijieDownloadConfig(meta).create(option); } else { if (__privateMethod(this, _Nijie_instances, isViewPopupPage_fn).call(this) && id === __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this)) { popupDoc = document; } else { popupDoc = await this.api.getViewPopupDoc(id); } const imgDiffSrcs = this.parser.parseDiffSrcByDoc(popupDoc); const diffMeta = this.parser.mergeImageDiff(meta, imgDiffSrcs); if (pageNum) { downloadConfig = new NijieDownloadConfig(diffMeta).create({ ...option, index: pageNum }); } else { const bundleIllusts = siteFeature.compressMultiIllusts; downloadConfig = bundleIllusts ? new NijieDownloadConfig(diffMeta).createBundle(option) : new NijieDownloadConfig(diffMeta).createMulti(option); } } await downloader.download(downloadConfig, { priority: 1 }); const historyData = { pid: Number(id), user: artist, userId: Number(userId), title, comment: comment2, tags }; if (page !== undefined) { historyData.page = Number(page); } historyDb.add(historyData); } createThumbnailBtn() { const btnStyle2 = { thumbnails: { selector: "p.nijiedao:has(> img.ngtag)", setStyle: undefined, diff: undefined }, memberAndHotImage: { selector: "p.nijiedao > a:has(img.ngtag)", setStyle: (el) => { el.style.display = "inline-block"; }, diff: undefined }, imgDiff: { selector: 'a[href*="#diff_"]:has(img.ngtag:not([data-original]))', // :not([data-original]): exclude dojin thumbnails setStyle: (el) => { const img = el.querySelector("img"); const margin = getComputedStyle(img).margin; img.style.margin = "0px"; el.style.display = "inline-block"; el.style.margin = margin; }, diff: (el) => { var _a2; return (_a2 = new RegExp("(?<=#diff_)[0-9]+$").exec(el.href)) == null ? undefined : _a2[0]; } }, dojinDiff: { selector: 'a[href*="#diff_"]:has(img[data-original].ngtag)', setStyle: (el) => { const img = el.querySelector("img"); img.style.width = "auto"; el.style.display = "inline-block"; }, diff: (el) => { var _a2; return (_a2 = new RegExp("(?<=#diff_)[0-9]+$").exec(el.href)) == null ? undefined : _a2[0]; } }, otherDojin: { selector: 'a[href*="id="]:has(>.other_dojin_block)', setStyle: (el) => { el.style.display = "block"; }, diff: undefined }, similar: { selector: "#nuitahito li:has(img.ngtag)", setStyle: undefined, diff: undefined }, // nuitahito // okazu rank: { selector: "p.illust:has(> img.ngtag)", setStyle: (el) => { const okazuThumbnailIcon = el.previousElementSibling; if (!okazuThumbnailIcon) return; okazuThumbnailIcon.style.zIndex = "1"; }, diff: undefined }, dictionary: { selector: 'a[href*="id="]:has(> :is(img.tagsimage, img.mozamoza:not([illust_id])))', // .tagsimage: dictionary block in 'search.php' setStyle: (el) => { const img = el.querySelector("img"); const width = getComputedStyle(img).width; el.style.width = width; img.style.width = "100%"; el.style.display = "inline-block"; }, diff: undefined }, responseOdai: { selector: '#response_odai li a[href*="id="]:has(img)', setStyle: (el) => { el.style.display = "inline-block"; }, diff: undefined }, historyIllust: { selector: '.history_block > .picture > a[href*="id="]:has(img)', setStyle: (el) => { el.style.display = "inline-block"; }, diff: undefined } }; for (const { selector, setStyle, diff } of Object.values(btnStyle2)) { const btnContainers = document.querySelectorAll(selector); if (!btnContainers.length) continue; btnContainers.forEach((container) => { if (container.querySelector(ThumbnailButton.tagNameLowerCase)) return; let id; if (container instanceof HTMLAnchorElement) { id = new RegExp("(?<=id=)[0-9]+").exec(container.href)[0]; } else { const img = container.querySelector("img.ngtag"); id = img.getAttribute("illust_id"); } const page = diff == null ? undefined : diff(container); setStyle == null ? undefined : setStyle(container); container.style.position = "relative"; container.appendChild( new ThumbnailButton({ id, page: page ? +page : undefined, onClick: this.downloadArtwork }) ); }); } } createIllustSettingBtn() { const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); const container = document.querySelector("#illust_setting"); if (!id || !container) return; container.appendChild( new ThumbnailButton({ id, type: ThumbnailBtnType.NijieIllust, onClick: this.downloadArtwork }) ); } // sticky button does not work properly on 'view.php' due to `overflow: hidden` on ancestor `#main-center-none` createImgFilterBtn() { const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); const containers = document.querySelectorAll("#img_filter"); if (!id || !containers.length) return; if (containers.length === 1) { if (containers[0].dataset.index) return; if (!document.querySelector('#img_diff > a[href*="diff_"]')) return; } containers.forEach((container, idx) => { const media = container.querySelector( 'img[src*="pic.nijie.net"], video' ); if (!media) return; const { marginBottom } = getComputedStyle(media); container.style.marginBottom = marginBottom; media.style.marginBottom = "0px"; const filterImg = container.querySelector("img.filter"); filterImg && (filterImg.style.zIndex = "auto"); container.appendChild( new ArtworkButton({ id, page: idx, site: "nijie", onClick: this.downloadArtwork }) ); }); } createDojinHeaderBtn() { const container = document.querySelector("#dojin_header > .right > ul"); const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); if (!container || !id) return; container.appendChild( new ThumbnailButton({ id, type: ThumbnailBtnType.NijieIllust, onClick: this.downloadArtwork }) ); } createDojinCoverBtn() { const container = document.querySelector( '#dojin_left .image > a[href*="view_popup"]' ); const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); if (!container || !id) return; container.style.display = "inline-block"; container.style.position = "relative"; container.style.width = "fit-content"; container.appendChild(new ArtworkButton({ id, page: 0, onClick: this.downloadArtwork })); } createOdaiBtn() { const id = __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this); const container = document.querySelector( "#gallery_new > p > a:has(img#view_img)" ); if (!id || !container) return; const img = container.querySelector("img:not(.filter)"); const { marginBottom } = getComputedStyle(img); img.style.marginBottom = "0px"; container.style.marginBottom = marginBottom; container.style.display = "inline-block"; container.style.position = "relative"; container.appendChild(new ArtworkButton({ id, page: 0, onClick: this.downloadArtwork })); } observeOzakuListChange() { const ozakuList = document.querySelector("#okazu_list"); if (!ozakuList) return; const observer2 = new MutationObserver(() => this.createThumbnailBtn()); observer2.observe(ozakuList, { subtree: true, childList: true }); } inject() { super.inject(); this.downloadArtwork = this.downloadArtwork.bind(this); this.createThumbnailBtn(); if (__privateMethod(this, _Nijie_instances, isViewPage_fn).call(this)) { this.createIllustSettingBtn(); this.createImgFilterBtn(); this.createDojinHeaderBtn(); this.createDojinCoverBtn(); this.createOdaiBtn(); } else if (__privateMethod(this, _Nijie_instances, isViewPopupPage_fn).call(this)) { this.createIllustSettingBtn(); this.createImgFilterBtn(); } else if (__privateMethod(this, _Nijie_instances, isOkazuPage_fn).call(this)) { this.observeOzakuListChange(); } } } _searchParams = new WeakMap(); _Nijie_instances = new WeakSet(); isViewPage_fn = function() { return location.pathname === "/view.php"; }; isViewPopupPage_fn = function() { return location.pathname === "/view_popup.php"; }; isOkazuPage_fn = function() { return location.pathname === "/okazu.php"; }; isSupportedUserPage_fn = function(url) { return /members\.php|members_illust\.php|members_dojin\.php|user_like_illust_view\.php/.test( url ); }; isSupportedHistoryPage_fn = function(url) { return /history_nuita\.php|history_illust\.php/.test(url); }; getSearchId_fn = function() { return __privateGet(this, _searchParams).get("id"); }; addBookmark_fn3 = async function(id, tags) { var _a2; try { await this.api.addBookmark(id, tags); if ((__privateMethod(this, _Nijie_instances, isViewPage_fn).call(this) || __privateMethod(this, _Nijie_instances, isViewPopupPage_fn).call(this)) && __privateMethod(this, _Nijie_instances, getSearchId_fn).call(this) === id) { const bookmarkBtn = document.querySelector("a#bukuma-do"); if (!bookmarkBtn) return; bookmarkBtn.id = "bukuma"; bookmarkBtn.setAttribute( "href", bookmarkBtn.getAttribute("href").replace("bookmark.php", "bookmark_edit.php") ); (_a2 = bookmarkBtn.lastChild) == null ? void 0 : _a2.remove(); const text2 = document.createElement("span"); text2.textContent = "ブックマーク編集"; bookmarkBtn.appendChild(text2); } } catch (error) { logger.error(error); } }; class Rule34VaultDownloadConfig extends BooruDownloadConfig { static get supportedTemplate() { return { [SupportedTemplate.ID]: "{id}", [SupportedTemplate.ARTIST]: "{artist}", [SupportedTemplate.CHARACTER]: "{character}", [SupportedTemplate.DATE]: "{date}, {date(YYYY-MM-DD)}", [SupportedTemplate.SCORE]: "{score}" }; } getTemplateData() { return { id: this.id, artist: this.artist, character: this.character, date: this.createDate, score: String(this.score) }; } create(option) { return super.create({ ...option, cfClearance: undefined }); } } var Rule34VaultPostType = /* @__PURE__ */ ((Rule34VaultPostType2) => { Rule34VaultPostType2[Rule34VaultPostType2["JPG"] = 0] = "JPG"; Rule34VaultPostType2[Rule34VaultPostType2["MP4"] = 1] = "MP4"; return Rule34VaultPostType2; })(Rule34VaultPostType || {}); var Rule34VaultTagType = /* @__PURE__ */ ((Rule34VaultTagType2) => { Rule34VaultTagType2[Rule34VaultTagType2["GENERAL"] = 1] = "GENERAL"; Rule34VaultTagType2[Rule34VaultTagType2["COPYRIGHT"] = 2] = "COPYRIGHT"; Rule34VaultTagType2[Rule34VaultTagType2["CHARACTER"] = 4] = "CHARACTER"; Rule34VaultTagType2[Rule34VaultTagType2["ARTIST"] = 8] = "ARTIST"; return Rule34VaultTagType2; })(Rule34VaultTagType || {}); var TagSearchSortType = /* @__PURE__ */ ((TagSearchSortType2) => { TagSearchSortType2[TagSearchSortType2["LATEST"] = 0] = "LATEST"; TagSearchSortType2[TagSearchSortType2["TOP_RATED"] = 1] = "TOP_RATED"; TagSearchSortType2[TagSearchSortType2["MOST_VIEWED"] = 2] = "MOST_VIEWED"; return TagSearchSortType2; })(TagSearchSortType || {}); class Rule34VaultApi extends ApiBase { constructor() { super(...arguments); __privateAdd(this, _Rule34VaultApi_instances); } getPostData(id) { return this.getJSON(`/api/v2/post/${id}`); } getUserProfile(username) { return this.getJSON(`/api/v2/account/user/${username}`); } searchPlaylist(id, param, token) { return __privateMethod(this, _Rule34VaultApi_instances, requestSearch_fn).call(this, `/api/v2/post/search/playlist/${id}`, param, token); } searchTags(param, token) { return __privateMethod(this, _Rule34VaultApi_instances, requestSearch_fn).call(this, "/api/v2/post/search/root", param, token); } searchUserBookmark(userId, param, token) { return __privateMethod(this, _Rule34VaultApi_instances, requestSearch_fn).call(this, `/api/v2/post/search/bookmarked/${userId}`, param, token); } searchUserSubscriptions(userId, param, token) { return __privateMethod(this, _Rule34VaultApi_instances, requestSearch_fn).call(this, `/api/v2/post/search/tag-subscriptions/${userId}`, { ...param, sortBy: 0 /* LATEST */ }, token); } addBookmark(id, token) { return this.getJSON("/api/v2/post/action/bookmark", { method: "POST", headers: { "content-type": "application/json; charset=utf-8", authorization: `Bearer ${token}` }, body: `{"postId":${id}}` }); } } _Rule34VaultApi_instances = new WeakSet(); /** * Token: used to filter user's blacklisted tags * Blacklist does not apply to bookmarks and cums */ requestSearch_fn = function(url, param, token) { return this.getJSON(url, { method: "POST", headers: { "content-type": "application/json; charset=utf-8", ...token ? { authorization: `Bearer ${token}` } : {} }, body: JSON.stringify(param) }); }; class Rule34VaultParser extends ParserBase { constructor() { super(...arguments); __privateAdd(this, _tagTypeMap, { [Rule34VaultTagType.GENERAL]: "general", [Rule34VaultTagType.COPYRIGHT]: "copyright", [Rule34VaultTagType.CHARACTER]: "character", [Rule34VaultTagType.ARTIST]: "artist" }); } buildMeta(postData) { const { id, type, data, posted, likes = 0, tags: postTags } = postData; const tagsToFilter = []; const tagsToStore = []; const artists = []; const characters = []; for (const { value, type: type2 } of postTags) { tagsToFilter.push(value); const tag = value.replaceAll(" ", "_"); tagsToStore.push(`${__privateGet(this, _tagTypeMap)[type2]}:${tag}`); if (type2 === Rule34VaultTagType.ARTIST) { artists.push(tag); } else if (type2 === Rule34VaultTagType.CHARACTER) { characters.push(tag); } } const extendName = type === Rule34VaultPostType.JPG ? "jpg" : "mp4"; const src = `https://r34xyz.b-cdn.net/posts/${id / 1e3 | 0}/${id}/${id}.${extendName}`; return { id: String(id), src, extendName, artist: artists.join(",") || this.UNKNOWN_ARTIST, character: characters.join(",") || this.UNKNOWN_CHARACTER, tags: tagsToFilter, tagsToStore, score: likes, source: data.sources ? data.sources.join("\n") : "", createDate: posted, title: "" }; } getCurrentUserToken() { return localStorage.getItem("token") || undefined; } getCurrentUserProfile() { const profileStr = localStorage.getItem("user"); return profileStr ? JSON.parse(profileStr) : undefined; } } _tagTypeMap = new WeakMap(); class Rule34Vault extends SiteInject { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate("Rule34Vault/{artist}"); downloadSetting.setFilenameTemplate("{id}_{artist}_{character}"); buttonPosition.$update((state2) => { return { ...state2, thumbnailBtnUnitY: BtnLengthUnit.PX, "--pdl-btn-left-percent": 100, "--pdl-btn-top-px": 40 }; }); siteFeature.addBookmark ?? (siteFeature.addBookmark = false); } super(); __privateAdd(this, _Rule34Vault_instances); __publicField(this, "api", new Rule34VaultApi()); __publicField(this, "parser", new Rule34VaultParser()); __publicField(this, "pageType", [ "app-home-page", "app-post-page", "app-highest-posts-page", "app-hot-posts-page", "app-playlists-page", "app-playlist-page", "app-comments-page", "app-upgrade-to-premium-page", "app-user-page", "app-feed-page", "app-sign-in-page", "app-sign-up-page" ]); __privateAdd(this, _currentPage); __privateAdd(this, _notifyPageChange); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: async (url) => { try { let avatarEl; if (/\/playlists\/view\/[0-9]+/.test(url)) { avatarEl = await waitDom("app-playlist-page-content app-user-avatar div.img"); } else { avatarEl = document.body.querySelector("mat-sidenav app-user-avatar div.img"); } const style = avatarEl.getAttribute("style"); const avatarSrc = new RegExp('(?<=url\\(").+(?="\\))').exec(style)[0]; return avatarSrc; } catch (error) { return "/assets/icons/icon-512x512.png"; } }, parseMetaByArtworkId: async (id) => { const postData = await this.api.getPostData(id); return this.parser.buildMeta(postData); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfig = new Rule34VaultDownloadConfig(meta).create({ ...downloadSetting }); await downloader.download(downloadConfig, { signal }); const { id, tagsToStore, artist, title, source: source2 } = meta; historyDb.add({ pid: Number(id), user: artist, title, tags: tagsToStore, source: source2 }); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn(meta) { return !!meta.extendName && regexp.imageExt.test(meta.extendName); } }, { id: "allow_video", type: "include", name: () => t("downloader.category.filter.video"), checked: true, fn(meta) { return !!meta.extendName && regexp.videoExt.test(meta.extendName); } } ], enableTagFilter: true }, pageOption: { playlist: { name: "Playlist", match: () => __privateGet(this, _currentPage).current === "app-playlist-page", filterInGenerator: false, fn: (pageRange) => { const idMathch = new RegExp("(?<=\\/playlists\\/view\\/)[0-9]+").exec(location.pathname); if (!idMathch) throw new Error("Cannot get playlist id."); const id = idMathch[0]; const token = this.parser.getCurrentUserToken(); let cursor; const getPostIdByPage = async (page) => { const POST_PER_PAGE = 30; const skip = (page - 1) * POST_PER_PAGE; const param = { checkHasMore: true, countTotal: false, skip, take: POST_PER_PAGE }; if (cursor) param.cursor = cursor; const { cursor: nextCursor, hasMore, items } = await this.api.searchPlaylist(id, param, token); cursor = nextCursor; return { lastPage: !hasMore, data: items }; }; return this.parser.paginationGenerator( pageRange, getPostIdByPage, (item) => String(item.id) ); } }, tagSearch: { name: "All Posts", match: () => __privateGet(this, _currentPage).current === "app-home-page", filterInGenerator: false, fn: (pageRange) => { const { includeTags, sortBy, type } = __privateMethod(this, _Rule34Vault_instances, getCurrentSearchSetting_fn).call(this); const token = this.parser.getCurrentUserToken(); let cursor; const getPostIdByPage = async (page) => { const POST_PER_PAGE = 30; const skip = (page - 1) * POST_PER_PAGE; const param = { includeTags, sortBy, checkHasMore: true, countTotal: true, skip, take: POST_PER_PAGE }; if (cursor) param.cursor = cursor; if (type !== undefined) param.type = type; const { cursor: nextCursor, hasMore, items } = await this.api.searchTags(param, token); cursor = nextCursor; return { lastPage: !hasMore, data: items }; }; return this.parser.paginationGenerator( pageRange, getPostIdByPage, (item) => String(item.id) ); } }, bookmarkSearch: { name: "Bookmarks", match: () => __privateGet(this, _currentPage).current === "app-home-page", filterInGenerator: false, fn: (pageRange) => { var _a2; const token = this.parser.getCurrentUserToken(); const userId = (_a2 = this.parser.getCurrentUserProfile()) == null ? undefined : _a2.id; if (!token || !userId) throw new Error("Please sign in."); const { includeTags, sortBy, type } = __privateMethod(this, _Rule34Vault_instances, getCurrentSearchSetting_fn).call(this); let cursor; const getPostIdByPage = async (page) => { const POST_PER_PAGE = 30; const skip = (page - 1) * POST_PER_PAGE; const param = { includeTags, sortBy, checkHasMore: true, countTotal: true, skip, take: POST_PER_PAGE }; if (cursor) param.cursor = cursor; if (type !== undefined) param.type = type; const { cursor: nextCursor, hasMore, items } = await this.api.searchUserBookmark(String(userId), param, token); cursor = nextCursor; return { lastPage: !hasMore, data: items }; }; return this.parser.paginationGenerator( pageRange, getPostIdByPage, (item) => String(item.id) ); } }, subscriptionSearch: { name: "Subscriptions", match: () => __privateGet(this, _currentPage).current === "app-home-page", filterInGenerator: false, fn: (pageRange) => { var _a2; const token = this.parser.getCurrentUserToken(); const userId = (_a2 = this.parser.getCurrentUserProfile()) == null ? undefined : _a2.id; if (!token || !userId) throw new Error("Please sign in."); const { includeTags, type } = __privateMethod(this, _Rule34Vault_instances, getCurrentSearchSetting_fn).call(this); let cursor; const getPostIdByPage = async (page) => { const POST_PER_PAGE = 30; const skip = (page - 1) * POST_PER_PAGE; const param = { includeTags, checkHasMore: true, countTotal: true, skip, take: POST_PER_PAGE }; if (cursor) param.cursor = cursor; if (type !== undefined) param.type = type; const { cursor: nextCursor, hasMore, items } = await this.api.searchUserSubscriptions(String(userId), param, token); cursor = nextCursor; return { lastPage: !hasMore, data: items }; }; return this.parser.paginationGenerator( pageRange, getPostIdByPage, (item) => String(item.id) ); } } } })); __privateSet(this, _currentPage, new ReactiveValue( () => document.querySelector(this.pageType.join(",")).tagName.toLowerCase(), (update2) => { __privateSet(this, _notifyPageChange, update2); return () => { __privateSet(this, _notifyPageChange, undefined); }; } )); } static get hostname() { return "rule34vault.com"; } getSupportedTemplate() { return Rule34VaultDownloadConfig.supportedTemplate; } inject() { super.inject(); const root2 = document.querySelector("app-root"); if (!root2) return; new MutationObserver((records) => { const thumbnailBtnTagName = ThumbnailButton.tagNameLowerCase.toUpperCase(); const addedElementNodes = records.filter((record) => record.type === "childList" && record.addedNodes.length).flatMap((record) => Array.from(record.addedNodes)).filter( (node) => node.nodeType === Node.ELEMENT_NODE && node.tagName !== thumbnailBtnTagName ); if (!addedElementNodes.length) return; const getIdByAnchor = (el) => { var _a2; const id = (_a2 = new RegExp("(?<=\\/post\\/)[0-9]+$").exec(el.getAttribute("href"))) == null ? undefined : _a2[0]; return id ?? ""; }; addedElementNodes.forEach((el) => { switch (true) { case el.matches('a[href^="/post/"].box'): { const id = getIdByAnchor(el); if (!id) return; __privateMethod(this, _Rule34Vault_instances, createThumbnailButton_fn).call(this, id, el.querySelector("div.box-inner")); break; } case el.matches('a[href^="/post/"].masonry-item'): { const id = getIdByAnchor(el); if (!id) return; __privateMethod(this, _Rule34Vault_instances, createThumbnailButton_fn).call(this, id, el); break; } case el.matches('div.post-con:has(a[href^="/post/"])'): { const anchor = el.querySelector('a[href^="/post/"]'); const id = getIdByAnchor(anchor); if (!id) return; __privateMethod(this, _Rule34Vault_instances, createThumbnailButton_fn).call(this, id, el.querySelector("div.con")); break; } case el.matches("app-post-page-content, app-tag-list"): { __privateMethod(this, _Rule34Vault_instances, createOrUpdatePostActionButton_fn).call(this); break; } case el.matches(this.pageType.join(",")): { if (!__privateGet(this, _notifyPageChange)) { throw new TypeError("Type of #notifyPageChange is undefined."); } __privateGet(this, _notifyPageChange).call(this); break; } } }); }).observe(root2, { subtree: true, childList: true }); } } _currentPage = new WeakMap(); _notifyPageChange = new WeakMap(); _Rule34Vault_instances = new WeakSet(); getCurrentSearchSetting_fn = function() { const searchParam = new URLSearchParams(location.search); const sort = searchParam.get("sort"); const mediaType = searchParam.get("type"); const sortBy = sort === "likes" ? TagSearchSortType.TOP_RATED : sort === "views" ? TagSearchSortType.MOST_VIEWED : TagSearchSortType.LATEST; const type = mediaType === "image" ? Rule34VaultPostType.JPG : mediaType === "video" ? Rule34VaultPostType.MP4 : undefined; const encodedTag = location.pathname.slice(1); const includeTags = encodedTag ? decodeURIComponent(decodeURIComponent(encodedTag)).replaceAll("_", " ").split("|") : []; return { includeTags, sortBy, type }; }; addBookmark_fn4 = async function(id) { try { const token = this.parser.getCurrentUserToken(); if (!token) throw new Error("Token not found, please sign in"); await this.api.addBookmark(id, token); this.toast({ message: "You have favorited this post", timeout: 2e3 }); } catch (error) { logger.error(error); this.toast({ message: error.message, type: "error" }); } }; downloadArtwork_fn2 = async function(btn2) { const id = btn2.dataset.id; const postData = await this.api.getPostData(id); const mediaMeta = this.parser.buildMeta(postData); const downloadConfig = new Rule34VaultDownloadConfig(mediaMeta).create({ ...downloadSetting, setProgress: (progress) => { btn2.setProgress(progress); } }); if (siteFeature.addBookmark) { __privateMethod(this, _Rule34Vault_instances, addBookmark_fn4).call(this, id); } await downloader.download(downloadConfig, { priority: 1 }); const { tagsToStore, artist, title, source: source2 } = mediaMeta; historyDb.add({ pid: Number(id), user: artist, title, tags: tagsToStore, source: source2 }); }; createOrUpdatePostActionButton_fn = function() { var _a2; const id = (_a2 = new RegExp("(?<=\\/post\\/)[0-9]+").exec(location.pathname)) == null ? undefined : _a2[0]; if (!id) return; const container = document.querySelector("app-post-actions div.con"); if (!container) return; const btn2 = container.querySelector(ThumbnailButton.tagNameLowerCase); if (btn2) { btn2.dataset.id = id; } else { const wrapper = document.createElement("div"); wrapper.setAttribute("style", "margin-right: 8px;"); wrapper.classList.add("chip"); wrapper.appendChild( new ThumbnailButton({ id, type: ThumbnailBtnType.Rule34VaultPostAction, onClick: (btn22) => __privateMethod(this, _Rule34Vault_instances, downloadArtwork_fn2).call(this, btn22) }) ); container.insertBefore(wrapper, container.firstElementChild); } }; createThumbnailButton_fn = function(id, container) { if (container.querySelector(ThumbnailButton.tagNameLowerCase)) return; container.appendChild( new ThumbnailButton({ id, onClick: (btn2) => __privateMethod(this, _Rule34Vault_instances, downloadArtwork_fn2).call(this, btn2) }) ); }; class Rule34PahealApi extends ApiBase { getPostViewDoc(id) { return this.getDoc(`/post/view/${id}`); } getPostListDoc(page, tag) { let url = "/post/list/"; url += tag !== undefined ? `${tag}/${page}` : String(page); return this.getDoc(url); } } class Rule34PahealParser extends ParserBase { constructor() { super(...arguments); __privateAdd(this, _Rule34PahealParser_instances); } buildMetaByDoc(id, doc) { var _a2; const mediaEl = doc.querySelector("#main_image"); if (!mediaEl) throw new Error("Cannot find media element."); let src; let extendName; if (mediaEl.tagName === "IMG") { const mimeType = mediaEl.getAttribute("data-mime"); if (!mimeType) throw new Error("Cannot get mimetype."); extendName = mimeType.split("/")[1]; src = mediaEl.src; } else { const videoSourceEl = mediaEl.querySelector("source"); if (!videoSourceEl) throw new Error("Cannot get video source."); const mimeType = videoSourceEl.getAttribute("type"); if (!mimeType) throw new Error("Cannot get mimetype."); extendName = mimeType.split("/")[1]; src = videoSourceEl.src; } const md5 = src.slice(src.lastIndexOf("/") + 1); const tags = Array.from( doc.querySelectorAll('tr[data-row="Tags"] .tag') ).map((el) => el.textContent.toLowerCase()); const createDate = (_a2 = doc.querySelector('tr[data-row="Uploader"] time')) == null ? undefined : _a2.dateTime; if (!createDate) throw new Error("Cannot get upload date."); const sourceEl = doc.querySelector('tr[data-row="Source Link"] td a'); const source2 = (sourceEl == null ? undefined : sourceEl.href) || ""; return { id, src, extendName, artist: "", title: md5, tags, createDate, source: source2 }; } buildMetasByPostsListDoc(doc) { const thumbnailEls = doc.querySelectorAll(".shm-image-list .shm-thumb.thumb"); if (!thumbnailEls.length) return []; return Array.from(thumbnailEls).map((el) => __privateMethod(this, _Rule34PahealParser_instances, buildMetaByThumbnailEl_fn).call(this, el)); } docHasNextPage(doc) { return Array.from(doc.querySelectorAll("#paginator a")).some( (el) => el.textContent === "Next" ); } } _Rule34PahealParser_instances = new WeakSet(); // the thumbnail element does not contain source url. buildMetaByThumbnailEl_fn = function(el) { const imgEl = el.querySelector("img"); const fileAnchor = el.querySelector('a[href*="r34i.paheal-cdn.net"]'); if (!imgEl || !fileAnchor) throw new Error("Cannot find image or flie link."); const src = fileAnchor.href; const md5 = src.slice(src.lastIndexOf("/") + 1); const { ext, tags, postId } = el.dataset; const imgTitleData = imgEl.title.split("\n"); const createDate = imgTitleData[imgTitleData.length - 1]; return { id: postId, src, extendName: ext, artist: "", title: md5, tags: tags.split(" "), createDate }; }; class Rule34PahealDownloadConfig extends MediaDownloadConfig { constructor(meta) { super(meta); } static get supportedTemplate() { return { [SupportedTemplate.ID]: "{id}", [SupportedTemplate.DATE]: "{date}, {date(YYYY-MM-DD)}", [SupportedTemplate.TAGS]: "{tags}", [SupportedTemplate.MD5]: "{md5}" }; } getTemplateData() { return { id: this.id, tags: this.tags.join(" "), date: this.createDate, md5: this.title }; } create(option) { const { filenameTemplate, filenameConflictAction, directoryTemplate, useFileSystemAccessApi, setProgress } = option; return { taskId: this.getTaskId(), src: this.getSrc(), path: this.getSavePath( directoryTemplate, filenameTemplate, this.getExt(), this.getTemplateData() ), timeout: this.getDownloadTimeout(), onProgress: setProgress, useFileSystemAccessApi, filenameConflictAction }; } } class Rule34Paheal extends SiteInject { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate("Rule34Paheal"); } super(); __privateAdd(this, _Rule34Paheal_instances); __privateAdd(this, _api, new Rule34PahealApi()); __privateAdd(this, _parser, new Rule34PahealParser()); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: "/favicon.ico", parseMetaByArtworkId: async (id) => { const postDoc = await __privateGet(this, _api).getPostViewDoc(id); return __privateGet(this, _parser).buildMetaByDoc(id, postDoc); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfig = new Rule34PahealDownloadConfig(meta).create({ ...downloadSetting }); await downloader.download(downloadConfig, { signal }); const { tags, title, source: source2, id } = meta; historyDb.add({ pid: Number(id), title, tags, source: source2 }); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn(meta) { return !!meta.extendName && regexp.imageExt.test(meta.extendName); } }, { id: "allow_video", type: "include", name: () => t("downloader.category.filter.video"), checked: true, fn(meta) { return !!meta.extendName && regexp.videoExt.test(meta.extendName); } } ], enableTagFilter: true }, pageOption: { posts: { name: "Posts", match: () => __privateMethod(this, _Rule34Paheal_instances, isPostList_fn).call(this), filterInGenerator: true, fn: (pageRange, checkValidity) => { if (pageRange && pageRange[0] > 500) throw new Error("Only 500 pages of results are searchable"); const tagPageStr = location.pathname.slice("/post/list/".length); const [tagOrPage, maybePage] = tagPageStr.split("/"); let tag; if (maybePage === undefined && /^[0-9]+$/.test(tagOrPage)) { tag = undefined; } else { tag = tagOrPage === "" ? undefined : tagOrPage; } const getArtworkMetasByPage = async (page) => { const doc = await __privateGet(this, _api).getPostListDoc(page, tag); return { lastPage: !__privateGet(this, _parser).docHasNextPage(doc), data: __privateGet(this, _parser).buildMetasByPostsListDoc(doc) }; }; return __privateGet(this, _parser).paginationGenerator( pageRange, getArtworkMetasByPage, (data) => data.id, async (data) => { return await checkValidity(data) ? PostValidState.VALID : PostValidState.INVALID; } ); } } } })); } static get hostname() { return "rule34.paheal.net"; } getSupportedTemplate() { return Rule34PahealDownloadConfig.supportedTemplate; } inject() { super.inject(); if (__privateMethod(this, _Rule34Paheal_instances, isPostView_fn2).call(this)) { __privateMethod(this, _Rule34Paheal_instances, createArtworkButton_fn).call(this); } else { __privateMethod(this, _Rule34Paheal_instances, createThumbnailButtons_fn).call(this); } } } _api = new WeakMap(); _parser = new WeakMap(); _Rule34Paheal_instances = new WeakSet(); isPostList_fn = function() { return location.pathname.startsWith("/post/list"); }; isPostView_fn2 = function() { return /\/post\/view\/[0-9]+/.test(location.pathname); }; downloadArtwork_fn3 = async function(btn2) { const id = btn2.dataset.id; const postDoc = await __privateGet(this, _api).getPostViewDoc(id); const mediaMeta = __privateGet(this, _parser).buildMetaByDoc(id, postDoc); const downloadConfig = new Rule34PahealDownloadConfig(mediaMeta).create({ ...downloadSetting, setProgress: (progress) => { btn2.setProgress(progress); } }); await downloader.download(downloadConfig, { priority: 1 }); const { tags, title, source: source2 } = mediaMeta; historyDb.add({ pid: Number(id), title, tags, source: source2 }); }; createArtworkButton_fn = function() { const idMatch = new RegExp("(?<=view\\/)[0-9]+").exec(location.pathname); if (!idMatch) return; const id = idMatch[0]; const blockBodyEL = document.querySelector(":is(#Imagemain, #Videomain) .blockbody"); const mediaEl = document.querySelector("#main_image"); if (!blockBodyEL || !mediaEl) throw new Error("Cannot find container or media."); const btnContainer = document.createElement("div"); btnContainer.classList.add("pdl-wrapper"); btnContainer.appendChild(mediaEl); btnContainer.appendChild( new ArtworkButton({ id, site: mediaEl.tagName === "VIDEO" ? "native_video" : undefined, onClick: (btn2) => __privateMethod(this, _Rule34Paheal_instances, downloadArtwork_fn3).call(this, btn2) }) ); blockBodyEL.appendChild(btnContainer); const styleSheet = new CSSStyleSheet(); styleSheet.replaceSync( ".pdl-wrapper { position: relative; width: fit-content; margin: 0 auto; font-size: 0px; } #main_image { max-width: 100% !important; }" // !important breaks zoomer: Full Size. ); document.adoptedStyleSheets = [...document.adoptedStyleSheets, styleSheet]; }; createThumbnailButtons_fn = function() { const thumbnailEls = document.querySelectorAll(".shm-thumb.thumb"); if (!thumbnailEls.length) return; thumbnailEls.forEach((el) => { const id = el.dataset.postId; if (!id) throw new Error("Cannot get post id."); const container = el.querySelector("a.shm-thumb-link"); if (!container) throw new Error("Cannot find button container"); container.setAttribute("style", "display: inline-block; position: relative; font-size: 0px;"); container.appendChild( new ThumbnailButton({ id, onClick: (btn2) => __privateMethod(this, _Rule34Paheal_instances, downloadArtwork_fn3).call(this, btn2) }) ); }); }; class Rule34UsApi extends ApiBase { getPostViewDoc(id) { return this.getDoc(`/index.php?r=posts/view&id=${id}`); } getPostListDoc(page, tags = "all") { return this.getDoc(`/index.php?r=posts/index&page=${page}&q=${tags}`); } getFavoritesDoc(id, page) { return this.getDoc(`/index.php?r=favorites/view&id=${id}&page=${page}`); } addBookmark(id) { return this.fetch(`/index.php?r=favorites/create&id=${id}`); } } class Rule34UsParser extends ParserBase { constructor() { super(...arguments); __privateAdd(this, _Rule34UsParser_instances); } buildMetaByDoc(id, doc) { var _a2; const mediaEl = doc.querySelector( ".content_push > :is(video, img[width])" ); if (!mediaEl) throw new Error("Cannot find media element."); let src; if (mediaEl.tagName === "IMG") { src = mediaEl.src; } else { const sourceEl = mediaEl.querySelector("source"); if (!sourceEl) { if (!mediaEl.src) throw new Error("Cannot get src of the video"); src = mediaEl.src; } else { src = sourceEl.src; } } const [md5, extendName] = src.slice(src.lastIndexOf("/") + 1).split("."); const tagListEl = doc.querySelector("ul.tag-list-left"); const artists = __privateMethod(this, _Rule34UsParser_instances, getTagsByEL_fn).call(this, tagListEl, "ul > .artist-tag > a"); const characters = __privateMethod(this, _Rule34UsParser_instances, getTagsByEL_fn).call(this, tagListEl, "ul > .character-tag > a"); const copyrights = __privateMethod(this, _Rule34UsParser_instances, getTagsByEL_fn).call(this, tagListEl, "ul > .copyright-tag > a"); const metadatas = __privateMethod(this, _Rule34UsParser_instances, getTagsByEL_fn).call(this, tagListEl, "ul > .metadata-tag > a"); const generals = __privateMethod(this, _Rule34UsParser_instances, getTagsByEL_fn).call(this, tagListEl, "ul > .general-tag > a"); const tags = [...artists, ...characters, ...copyrights, ...metadatas, ...generals]; const tagsWithType = [ ...artists.map((v) => "artist:" + v), ...characters.map((v) => "character:" + v), ...copyrights.map((v) => "copyright:" + v), ...metadatas.map((v) => "metadata:" + v), ...generals.map((v) => "general:" + v) ]; const scoreStr = (_a2 = tagListEl.querySelector('span[id^="psc"]')) == null ? undefined : _a2.textContent; return { id, src, extendName, title: md5, artist: artists.join(",") || this.UNKNOWN_ARTIST, character: characters.join(",") || this.UNKNOWN_CHARACTER, tags, tagsWithType, createDate: "", score: scoreStr ? +scoreStr : 0, source: "" }; } parsePostIdsByDoc(doc) { const thumbnailEls = doc.querySelectorAll('a[href*="posts/view&id="]'); if (!thumbnailEls.length) return []; return Array.from(thumbnailEls).map((el) => { const idMatch = new RegExp("(?<=id=)[0-9]+$").exec(el.href); if (!idMatch) throw new Error("Cannot match post id."); return idMatch[0]; }); } } _Rule34UsParser_instances = new WeakSet(); getTagsByEL_fn = function(el, selector) { return Array.from(el.querySelectorAll(selector)).map( (el2) => el2.textContent.replaceAll(" ", "_") ); }; class Rule34UsDownloadConfig extends BooruDownloadConfig { static get supportedTemplate() { return { [SupportedTemplate.ID]: "{id}", [SupportedTemplate.ARTIST]: "{artist}", [SupportedTemplate.CHARACTER]: "{character}", [SupportedTemplate.MD5]: "{md5}", [SupportedTemplate.SCORE]: "{score}" }; } getTemplateData() { return { id: this.id, artist: this.artist, character: this.character, md5: this.title, title: this.title, score: String(this.score) }; } } class Rule34Us extends SiteInject { constructor() { if (clientSetting.version === null) { downloadSetting.setDirectoryTemplate("Rule34us/{artist}"); siteFeature.addBookmark ?? (siteFeature.addBookmark = false); } super(); __privateAdd(this, _Rule34Us_instances); __privateAdd(this, _api2, new Rule34UsApi()); __privateAdd(this, _parser2, new Rule34UsParser()); __privateAdd(this, _searchParam, new URLSearchParams(location.search)); __publicField(this, "useBatchDownload", this.app.initBatchDownloader({ avatar: "/v1/counter/7.gif", parseMetaByArtworkId: async (id) => { const doc = await __privateGet(this, _api2).getPostViewDoc(id); return __privateGet(this, _parser2).buildMetaByDoc(id, doc); }, downloadArtworkByMeta: async (meta, signal) => { const downloadConfig = new Rule34UsDownloadConfig(meta).create({ ...downloadSetting }); await downloader.download(downloadConfig, { signal }); const { tagsWithType, title, artist, id } = meta; historyDb.add({ pid: Number(id), title, tags: tagsWithType, user: artist }); }, filterOption: { filters: [ { id: "exclude_downloaded", type: "exclude", name: () => t("downloader.category.filter.exclude_downloaded"), checked: false, fn(meta) { return !!meta.id && historyDb.has(meta.id); } }, { id: "allow_image", type: "include", name: () => t("downloader.category.filter.image"), checked: true, fn(meta) { return !!meta.extendName && regexp.imageExt.test(meta.extendName); } }, { id: "allow_video", type: "include", name: () => t("downloader.category.filter.video"), checked: true, fn(meta) { return !!meta.extendName && regexp.videoExt.test(meta.extendName); } } ], enableTagFilter: true }, pageOption: { posts: { name: "Posts", match: () => __privateMethod(this, _Rule34Us_instances, isPostList_fn2).call(this), filterInGenerator: false, fn: (pageRange) => { const tags = __privateMethod(this, _Rule34Us_instances, getQueryTag_fn).call(this) || undefined; return __privateGet(this, _parser2).paginationGenerator( pageRange, async (page) => { const THUMBS_PER_PAGE = 42; const doc = await __privateGet(this, _api2).getPostListDoc(page - 1, tags); const data = __privateGet(this, _parser2).parsePostIdsByDoc(doc); return { lastPage: data.length < THUMBS_PER_PAGE, data }; }, (data) => data ); } }, // search with fav:id favorites: { name: "Favorites", match: () => __privateMethod(this, _Rule34Us_instances, isFavorites_fn).call(this), filterInGenerator: false, fn: (pageRange) => { const id = __privateMethod(this, _Rule34Us_instances, getQueryId_fn).call(this); if (!id) throw new Error("Cannot get user id."); return __privateGet(this, _parser2).paginationGenerator( pageRange, async (page) => { const THUMBS_PER_PAGE = 30; const doc = await __privateGet(this, _api2).getFavoritesDoc(id, page - 1); const data = __privateGet(this, _parser2).parsePostIdsByDoc(doc); return { lastPage: data.length < THUMBS_PER_PAGE, data }; }, (data) => data ); } } } })); } static get hostname() { return "rule34.us"; } getSupportedTemplate() { return Rule34UsDownloadConfig.supportedTemplate; } inject() { super.inject(); if (__privateMethod(this, _Rule34Us_instances, isPostView_fn3).call(this)) { __privateMethod(this, _Rule34Us_instances, createArtworkButton_fn2).call(this); } else { __privateMethod(this, _Rule34Us_instances, createThumbnailButtons_fn2).call(this); } } } _api2 = new WeakMap(); _parser2 = new WeakMap(); _searchParam = new WeakMap(); _Rule34Us_instances = new WeakSet(); isPostList_fn2 = function() { return __privateGet(this, _searchParam).get("r") === "posts/index"; }; isPostView_fn3 = function() { return __privateGet(this, _searchParam).get("r") === "posts/view"; }; isFavorites_fn = function() { return __privateGet(this, _searchParam).get("r") === "favorites/view"; }; getQueryTag_fn = function() { return __privateGet(this, _searchParam).get("q"); }; getQueryId_fn = function() { return __privateGet(this, _searchParam).get("id"); }; addBookmark_fn5 = function(id) { _unsafeWindow.addFav(id); }; downloadArtwork_fn4 = async function(btn2) { const id = btn2.dataset.id; const postDoc = await __privateGet(this, _api2).getPostViewDoc(id); const mediaMeta = __privateGet(this, _parser2).buildMetaByDoc(id, postDoc); const downloadConfig = new Rule34UsDownloadConfig(mediaMeta).create({ ...downloadSetting, setProgress: (progress) => { btn2.setProgress(progress); } }); if (siteFeature.addBookmark) __privateMethod(this, _Rule34Us_instances, addBookmark_fn5).call(this, id); await downloader.download(downloadConfig, { priority: 1 }); const { tagsWithType, title, artist } = mediaMeta; historyDb.add({ pid: Number(id), title, tags: tagsWithType, user: artist }); }; createThumbnailButtons_fn2 = function() { const thumbnailEls = document.querySelectorAll('a[href*="posts/view&id="]'); if (!thumbnailEls.length) return; const fitContent = __privateMethod(this, _Rule34Us_instances, isFavorites_fn).call(this) ? false : true; thumbnailEls.forEach((el) => { const idMatch = new RegExp("(?<=id=)[0-9]+$").exec(el.href); if (!idMatch) throw new Error("Cannot match post id."); const id = idMatch[0]; el.setAttribute( "style", "display: block; position: relative; height: inherit; font-size: 0px" ); fitContent && (el.style.width = "fit-content"); el.appendChild( new ThumbnailButton({ id, onClick: (btn2) => __privateMethod(this, _Rule34Us_instances, downloadArtwork_fn4).call(this, btn2) }) ); }); }; createArtworkButton_fn2 = function() { var _a2; const id = __privateMethod(this, _Rule34Us_instances, getQueryId_fn).call(this); if (!id) throw new Error("Cannot get post id."); const mediaEl = document.querySelector( ".content_push > :is(video, img[width])" ); if (!mediaEl) throw new Error("Cannot find media element."); const wrapper = document.createElement("div"); wrapper.setAttribute( "style", "position: relative; width: fit-content; margin: 0 auto; font-size: 0px" ); (_a2 = mediaEl.parentElement) == null ? undefined : _a2.insertBefore(wrapper, mediaEl); wrapper.appendChild(mediaEl); wrapper.appendChild( new ArtworkButton({ id, site: mediaEl.tagName === "VIDEO" ? "native_video" : undefined, onClick: (btn2) => __privateMethod(this, _Rule34Us_instances, downloadArtwork_fn4).call(this, btn2) }) ); }; function getSiteInjector() { const sitesAdapters = [ Pixiv, Danbooru, Rule34, Yande, ATFbooru, Konachan, Sakugabooru, Safebooru, Gelbooru, E621ng, Nijie, Rule34Vault, Rule34Paheal, Rule34Us ]; const hostname = location.hostname; for (const sites of sitesAdapters) { if (typeof sites.hostname === "string") { if (hostname === sites.hostname) { return sites; } } else { if (sites.hostname.includes(hostname)) { return sites; } } } } const siteInject = getSiteInjector(); siteInject && new siteInject().inject(); })(Dexie, GIF, WebMMuxer, Mp4Muxer, dayjs, JSZip);