MemoizeX

Enhanced Twitter/X data management with browsing history tracking and webhook notifications.

K instalaci tototo skriptu si budete muset nainstalovat rozšíření jako Tampermonkey, Greasemonkey nebo Violentmonkey.

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

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Userscripts.

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

K instalaci tohoto skriptu si budete muset nainstalovat manažer uživatelských skriptů.

(Už mám manažer uživatelských skriptů, nechte mě ho nainstalovat!)

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

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

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

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

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

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

(Už mám manažer uživatelských stylů, nechte mě ho nainstalovat!)

// ==UserScript==
// @name               MemoizeX
// @name:zh-CN         MemoizeX - Twitter 数据增强工具
// @name:zh-TW         MemoizeX - Twitter 資料增強工具
// @namespace          https://github.com/memoizex
// @version            1.0.3
// @author             MemoizeX
// @description        Enhanced Twitter/X data management with browsing history tracking and webhook notifications.
// @description:zh-CN  增强 Twitter(X) 数据管理能力,支持浏览历史记录和 Webhook 通知。
// @description:zh-TW  增強 Twitter(X) 資料管理能力,支援瀏覽歷史記錄和 Webhook 通知。
// @license            MIT
// @icon               
// @homepage           https://github.com/memoizex/memoizex
// @homepageURL        https://github.com/memoizex/memoizex
// @supportURL         https://github.com/memoizex/memoizex/issues
// @match              *://twitter.com/*
// @match              *://x.com/*
// @match              *://mobile.x.com/*
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dayjs.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/dexie.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/dexie-export-import.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/FileSaver.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/i18next.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/dist/preact.min.js
// @require            https://cdn.jsdelivr.net/npm/[email protected]/hooks/dist/hooks.umd.js
// @require            https://cdn.jsdelivr.net/npm/@preact/[email protected]/dist/signals-core.min.js
// @require            https://cdn.jsdelivr.net/npm/@preact/[email protected]/dist/signals.min.js
// @require            https://cdn.jsdelivr.net/npm/@tanstack/[email protected]/build/umd/index.production.js
// @connect            *
// @grant              GM.xmlHttpRequest
// @grant              GM_addStyle
// @grant              GM_registerMenuCommand
// @grant              GM_xmlhttpRequest
// @grant              unsafeWindow
// @run-at             document-start
// ==/UserScript==

(o=>{if(typeof GM_addStyle=="function"){GM_addStyle(o);return}const t=document.createElement("style");t.textContent=o,document.head.append(t)})(` #memoizex-root *,#memoizex-root :before,#memoizex-root :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: }#memoizex-root ::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: }#memoizex-root *,#memoizex-root :before,#memoizex-root :after{box-sizing:border-box;border-width:0;border-style:solid;border-color:#e5e7eb}#memoizex-root :before,#memoizex-root :after{--tw-content: ""}#memoizex-root,#memoizex-root :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}#memoizex-root{margin:0;line-height:inherit}#memoizex-root hr{height:0;color:inherit;border-top-width:1px}#memoizex-root abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}#memoizex-root h1,#memoizex-root h2,#memoizex-root h3,#memoizex-root h4,#memoizex-root h5,#memoizex-root h6{font-size:inherit;font-weight:inherit}#memoizex-root a{color:inherit;text-decoration:inherit}#memoizex-root b,#memoizex-root strong{font-weight:bolder}#memoizex-root code,#memoizex-root kbd,#memoizex-root samp,#memoizex-root pre{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}#memoizex-root small{font-size:80%}#memoizex-root sub,#memoizex-root sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}#memoizex-root sub{bottom:-.25em}#memoizex-root sup{top:-.5em}#memoizex-root table{text-indent:0;border-color:inherit;border-collapse:collapse}#memoizex-root button,#memoizex-root input,#memoizex-root optgroup,#memoizex-root select,#memoizex-root 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}#memoizex-root button,#memoizex-root select{text-transform:none}#memoizex-root button,#memoizex-root input:where([type=button]),#memoizex-root input:where([type=reset]),#memoizex-root input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}#memoizex-root :-moz-focusring{outline:auto}#memoizex-root :-moz-ui-invalid{box-shadow:none}#memoizex-root progress{vertical-align:baseline}#memoizex-root ::-webkit-inner-spin-button,#memoizex-root ::-webkit-outer-spin-button{height:auto}#memoizex-root [type=search]{-webkit-appearance:textfield;outline-offset:-2px}#memoizex-root ::-webkit-search-decoration{-webkit-appearance:none}#memoizex-root ::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}#memoizex-root summary{display:list-item}#memoizex-root blockquote,#memoizex-root dl,#memoizex-root dd,#memoizex-root h1,#memoizex-root h2,#memoizex-root h3,#memoizex-root h4,#memoizex-root h5,#memoizex-root h6,#memoizex-root hr,#memoizex-root figure,#memoizex-root p,#memoizex-root pre{margin:0}#memoizex-root fieldset{margin:0;padding:0}#memoizex-root legend{padding:0}#memoizex-root ol,#memoizex-root ul,#memoizex-root menu{list-style:none;margin:0;padding:0}#memoizex-root dialog{padding:0}#memoizex-root textarea{resize:vertical}#memoizex-root input::-moz-placeholder,#memoizex-root textarea::-moz-placeholder{opacity:1;color:#9ca3af}#memoizex-root input::placeholder,#memoizex-root textarea::placeholder{opacity:1;color:#9ca3af}#memoizex-root button,#memoizex-root [role=button]{cursor:pointer}#memoizex-root :disabled{cursor:default}#memoizex-root img,#memoizex-root svg,#memoizex-root video,#memoizex-root canvas,#memoizex-root audio,#memoizex-root iframe,#memoizex-root embed,#memoizex-root object{display:block;vertical-align:middle}#memoizex-root img,#memoizex-root video{max-width:100%;height:auto}#memoizex-root [hidden]:where(:not([hidden=until-found])){display:none}#memoizex-root,#memoizex-root [data-theme]{background-color:var(--fallback-b1,oklch(var(--b1)/1));color:var(--fallback-bc,oklch(var(--bc)/1))}@supports not (color: oklch(0% 0 0)){#memoizex-root{color-scheme:light;--fallback-p: #491eff;--fallback-pc: #d4dbff;--fallback-s: #ff41c7;--fallback-sc: #fff9fc;--fallback-a: #00cfbd;--fallback-ac: #00100d;--fallback-n: #2b3440;--fallback-nc: #d7dde4;--fallback-b1: #ffffff;--fallback-b2: #e5e6e6;--fallback-b3: #e5e6e6;--fallback-bc: #1f2937;--fallback-in: #00b3f0;--fallback-inc: #000000;--fallback-su: #00ca92;--fallback-suc: #000000;--fallback-wa: #ffc22d;--fallback-wac: #000000;--fallback-er: #ff6f70;--fallback-erc: #000000}@media(prefers-color-scheme:dark){#memoizex-root{color-scheme:dark;--fallback-p: #7582ff;--fallback-pc: #050617;--fallback-s: #ff71cf;--fallback-sc: #190211;--fallback-a: #00c7b5;--fallback-ac: #000e0c;--fallback-n: #2a323c;--fallback-nc: #a6adbb;--fallback-b1: #1d232a;--fallback-b2: #191e24;--fallback-b3: #15191e;--fallback-bc: #a6adbb;--fallback-in: #00b3f0;--fallback-inc: #000000;--fallback-su: #00ca92;--fallback-suc: #000000;--fallback-wa: #ffc22d;--fallback-wac: #000000;--fallback-er: #ff6f70;--fallback-erc: #000000}}}#memoizex-root{-webkit-tap-highlight-color:transparent}#memoizex-root *{scrollbar-color:color-mix(in oklch,currentColor 35%,transparent) transparent}#memoizex-root *:hover{scrollbar-color:color-mix(in oklch,currentColor 60%,transparent) transparent}#memoizex-root{color-scheme:light;--in: 72.06% .191 231.6;--su: 64.8% .15 160;--wa: 84.71% .199 83.87;--er: 71.76% .221 22.18;--pc: 15.2344% .017892 200.026556;--sc: 15.787% .020249 356.29965;--ac: 15.8762% .029206 78.618794;--nc: 84.7148% .013247 313.189598;--inc: 0% 0 0;--suc: 0% 0 0;--wac: 0% 0 0;--erc: 0% 0 0;--rounded-box: 16px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--p: 76.172% .089459 200.026556;--s: 78.9351% .101246 356.29965;--a: 79.3811% .146032 78.618794;--n: 23.5742% .066235 313.189598;--b1: 97.7882% .00418 56.375637;--b2: 93.9822% .007638 61.449292;--b3: 91.5861% .006811 53.440502;--bc: 23.5742% .066235 313.189598;--rounded-btn: 30.4px;--tab-border: 2px;--tab-radius: 11.2px}@media(prefers-color-scheme:dark){#memoizex-root{color-scheme:dark;--b2: 26.8053% .020556 277.508664;--b3: 24.7877% .019009 277.508664;--pc: 15.0922% .036614 346.812432;--sc: 14.8405% .029709 301.883095;--ac: 16.6785% .024826 66.558491;--nc: 87.8891% .006515 275.524078;--inc: 17.6526% .018676 212.846491;--suc: 17.4199% .043903 148.024881;--wac: 19.1068% .026849 112.757109;--erc: 13.6441% .041266 24.430965;--rounded-box: 16px;--rounded-btn: 8px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;--tab-radius: 8px;--p: 75.4611% .18307 346.812432;--s: 74.2023% .148546 301.883095;--a: 83.3927% .124132 66.558491;--n: 39.4456% .032576 275.524078;--b1: 28.8229% .022103 277.508664;--bc: 97.7477% .007913 106.545019;--in: 88.263% .09338 212.846491;--su: 87.0995% .219516 148.024881;--wa: 95.5338% .134246 112.757109;--er: 68.2204% .206328 24.430965}}#memoizex-root [data-theme=cupcake]{color-scheme:light;--in: 72.06% .191 231.6;--su: 64.8% .15 160;--wa: 84.71% .199 83.87;--er: 71.76% .221 22.18;--pc: 15.2344% .017892 200.026556;--sc: 15.787% .020249 356.29965;--ac: 15.8762% .029206 78.618794;--nc: 84.7148% .013247 313.189598;--inc: 0% 0 0;--suc: 0% 0 0;--wac: 0% 0 0;--erc: 0% 0 0;--rounded-box: 16px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--p: 76.172% .089459 200.026556;--s: 78.9351% .101246 356.29965;--a: 79.3811% .146032 78.618794;--n: 23.5742% .066235 313.189598;--b1: 97.7882% .00418 56.375637;--b2: 93.9822% .007638 61.449292;--b3: 91.5861% .006811 53.440502;--bc: 23.5742% .066235 313.189598;--rounded-btn: 30.4px;--tab-border: 2px;--tab-radius: 11.2px}#memoizex-root [data-theme=dark]{color-scheme:dark;--in: 72.06% .191 231.6;--su: 64.8% .15 160;--wa: 84.71% .199 83.87;--er: 71.76% .221 22.18;--pc: 13.138% .0392 275.75;--sc: 14.96% .052 342.55;--ac: 14.902% .0334 183.61;--inc: 0% 0 0;--suc: 0% 0 0;--wac: 0% 0 0;--erc: 0% 0 0;--rounded-box: 16px;--rounded-btn: 8px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;--tab-radius: 8px;--p: 65.69% .196 275.75;--s: 74.8% .26 342.55;--a: 74.51% .167 183.61;--n: 31.3815% .021108 254.139175;--nc: 74.6477% .0216 264.435964;--b1: 25.3267% .015896 252.417568;--b2: 23.2607% .013807 253.100675;--b3: 21.1484% .01165 254.087939;--bc: 74.6477% .0216 264.435964}#memoizex-root [data-theme=emerald]{color-scheme:light;--b2: 93% 0 0;--b3: 86% 0 0;--in: 72.06% .191 231.6;--su: 64.8% .15 160;--wa: 84.71% .199 83.87;--er: 71.76% .221 22.18;--inc: 0% 0 0;--suc: 0% 0 0;--wac: 0% 0 0;--erc: 0% 0 0;--rounded-box: 16px;--rounded-btn: 8px;--rounded-badge: 30.4px;--border-btn: 1px;--tab-border: 1px;--tab-radius: 8px;--p: 76.6626% .135433 153.450024;--pc: 33.3872% .040618 162.240129;--s: 61.3028% .202368 261.294233;--sc: 100% 0 0;--a: 72.7725% .149783 33.200363;--ac: 0% 0 0;--n: 35.5192% .032071 262.988584;--nc: 98.4625% .001706 247.838921;--b1: 100% 0 0;--bc: 35.5192% .032071 262.988584;--animation-btn: 0;--animation-input: 0;--btn-focus-scale: 1}#memoizex-root [data-theme=cyberpunk]{color-scheme:light;--b2: 87.8943% .16647 104.32;--b3: 81.2786% .15394 104.32;--in: 72.06% .191 231.6;--su: 64.8% .15 160;--wa: 84.71% .199 83.87;--er: 71.76% .221 22.18;--bc: 18.902% .0358 104.32;--pc: 14.844% .0418 6.35;--sc: 16.666% .0368 204.72;--ac: 14.372% .04352 310.43;--inc: 0% 0 0;--suc: 0% 0 0;--wac: 0% 0 0;--erc: 0% 0 0;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;--p: 74.22% .209 6.35;--s: 83.33% .184 204.72;--a: 71.86% .2176 310.43;--n: 23.04% .065 269.31;--nc: 94.51% .179 104.32;--b1: 94.51% .179 104.32;--rounded-box: 0;--rounded-btn: 0;--rounded-badge: 0;--tab-radius: 0}#memoizex-root [data-theme=valentine]{color-scheme:light;--b2: 88.0567% .024834 337.06289;--b3: 81.4288% .022964 337.06289;--pc: 13.7239% .030755 15.066527;--sc: 14.3942% .029258 293.189609;--ac: 14.2537% .014961 197.828857;--inc: 90.923% .043042 262.880917;--suc: 12.541% .033982 149.213788;--wac: 13.3168% .031484 58.31834;--erc: 14.614% .0414 27.33;--rounded-box: 16px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;--p: 68.6197% .153774 15.066527;--s: 71.971% .14629 293.189609;--a: 71.2685% .074804 197.828857;--n: 54.6053% .143342 358.004839;--nc: 90.2701% .037202 336.955191;--b1: 94.6846% .026703 337.06289;--bc: 37.3085% .081131 4.606426;--in: 54.615% .215208 262.880917;--su: 62.7052% .169912 149.213788;--wa: 66.584% .157422 58.31834;--er: 73.07% .207 27.33;--rounded-btn: 30.4px;--tab-radius: 11.2px}#memoizex-root [data-theme=lofi]{color-scheme:light;--inc: 15.908% .0206 205.9;--suc: 18.026% .0306 164.14;--wac: 17.674% .027 79.94;--erc: 15.732% .03 28.47;--border-btn: 1px;--tab-border: 1px;--p: 15.9066% 0 0;--pc: 100% 0 0;--s: 21.455% .001566 17.278957;--sc: 100% 0 0;--a: 26.8618% 0 0;--ac: 100% 0 0;--n: 0% 0 0;--nc: 100% 0 0;--b1: 100% 0 0;--b2: 96.1151% 0 0;--b3: 92.268% .001082 17.17934;--bc: 0% 0 0;--in: 79.54% .103 205.9;--su: 90.13% .153 164.14;--wa: 88.37% .135 79.94;--er: 78.66% .15 28.47;--rounded-box: 4px;--rounded-btn: 2px;--rounded-badge: 2px;--tab-radius: 2px;--animation-btn: 0;--animation-input: 0;--btn-focus-scale: 1}#memoizex-root [data-theme=dracula]{color-scheme:dark;--b2: 26.8053% .020556 277.508664;--b3: 24.7877% .019009 277.508664;--pc: 15.0922% .036614 346.812432;--sc: 14.8405% .029709 301.883095;--ac: 16.6785% .024826 66.558491;--nc: 87.8891% .006515 275.524078;--inc: 17.6526% .018676 212.846491;--suc: 17.4199% .043903 148.024881;--wac: 19.1068% .026849 112.757109;--erc: 13.6441% .041266 24.430965;--rounded-box: 16px;--rounded-btn: 8px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;--tab-radius: 8px;--p: 75.4611% .18307 346.812432;--s: 74.2023% .148546 301.883095;--a: 83.3927% .124132 66.558491;--n: 39.4456% .032576 275.524078;--b1: 28.8229% .022103 277.508664;--bc: 97.7477% .007913 106.545019;--in: 88.263% .09338 212.846491;--su: 87.0995% .219516 148.024881;--wa: 95.5338% .134246 112.757109;--er: 68.2204% .206328 24.430965}#memoizex-root [data-theme=cmyk]{color-scheme:light;--b2: 93% 0 0;--b3: 86% 0 0;--bc: 20% 0 0;--pc: 14.3544% .02666 239.443325;--sc: 12.8953% .040552 359.339283;--ac: 18.8458% .037948 105.306968;--nc: 84.3557% 0 0;--inc: 13.6952% .0189 217.284104;--suc: 89.3898% .032505 321.406278;--wac: 14.2473% .031969 52.023412;--erc: 12.4027% .041677 28.717543;--rounded-box: 16px;--rounded-btn: 8px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;--tab-radius: 8px;--p: 71.7722% .133298 239.443325;--s: 64.4766% .202758 359.339283;--a: 94.2289% .189741 105.306968;--n: 21.7787% 0 0;--b1: 100% 0 0;--in: 68.4759% .094499 217.284104;--su: 46.949% .162524 321.406278;--wa: 71.2364% .159843 52.023412;--er: 62.0133% .208385 28.717543}#memoizex-root [data-theme=business]{color-scheme:dark;--b2: 22.6487% 0 0;--b3: 20.944% 0 0;--bc: 84.8707% 0 0;--pc: 88.3407% .019811 251.473931;--sc: 12.8185% .005481 229.389418;--ac: 13.4542% .033545 35.791525;--nc: 85.4882% .00265 253.041249;--inc: 12.5233% .028702 240.033697;--suc: 14.0454% .018919 156.59611;--wac: 15.4965% .023141 81.519177;--erc: 90.3221% .029356 29.674507;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;--tab-radius: 8px;--p: 41.7036% .099057 251.473931;--s: 64.0924% .027405 229.389418;--a: 67.271% .167726 35.791525;--n: 27.441% .01325 253.041249;--b1: 24.3535% 0 0;--in: 62.6163% .143511 240.033697;--su: 70.2268% .094594 156.59611;--wa: 77.4824% .115704 81.519177;--er: 51.6105% .14678 29.674507;--rounded-box: 4px;--rounded-btn: 2px;--rounded-badge: 2px}#memoizex-root [data-theme=winter]{color-scheme:light;--pc: 91.372% .051 257.57;--sc: 88.5103% .03222 282.339433;--ac: 11.988% .038303 335.171434;--nc: 83.9233% .012704 257.651965;--inc: 17.6255% .017178 214.515264;--suc: 16.0988% .015404 197.823719;--wac: 17.8345% .009167 71.47031;--erc: 14.6185% .022037 20.076293;--rounded-box: 16px;--rounded-btn: 8px;--rounded-badge: 30.4px;--animation-btn: .25s;--animation-input: .2s;--btn-focus-scale: .95;--border-btn: 1px;--tab-border: 1px;--tab-radius: 8px;--p: 56.86% .255 257.57;--s: 42.5516% .161098 282.339433;--a: 59.9398% .191515 335.171434;--n: 19.6166% .063518 257.651965;--b1: 100% 0 0;--b2: 97.4663% .011947 259.822565;--b3: 93.2686% .016223 262.751375;--bc: 41.8869% .053885 255.824911;--in: 88.1275% .085888 214.515264;--su: 80.4941% .077019 197.823719;--wa: 89.1725% .045833 71.47031;--er: 73.0926% .110185 20.076293}#memoizex-root .alert{display:grid;width:100%;grid-auto-flow:row;align-content:flex-start;align-items:center;justify-items:center;gap:16px;text-align:center;border-radius:var(--rounded-box, 16px);border-width:1px;--tw-border-opacity: 1;border-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)));padding:16px;--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));--alert-bg: var(--fallback-b2,oklch(var(--b2)/1));--alert-bg-mix: var(--fallback-b1,oklch(var(--b1)/1));background-color:var(--alert-bg)}@media(min-width:640px){#memoizex-root .alert{grid-auto-flow:column;grid-template-columns:auto minmax(auto,1fr);justify-items:start;text-align:start}}#memoizex-root .avatar{position:relative;display:inline-flex}#memoizex-root .avatar>div{display:block;aspect-ratio:1 / 1;overflow:hidden}#memoizex-root .avatar img{height:100%;width:100%;-o-object-fit:cover;object-fit:cover}#memoizex-root .avatar.placeholder>div{display:flex;align-items:center;justify-content:center}#memoizex-root .badge{display:inline-flex;align-items:center;justify-content:center;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-timing-function:cubic-bezier(0,0,.2,1);transition-duration:.2s;height:20px;font-size:14px;line-height:20px;width:-moz-fit-content;width:fit-content;padding-left:9.008px;padding-right:9.008px;border-radius:var(--rounded-badge, 30.4px);border-width:1px;--tw-border-opacity: 1;border-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)))}@media(hover:hover){#memoizex-root .checkbox-accent:hover{--tw-border-opacity: 1;border-color:var(--fallback-a,oklch(var(--a)/var(--tw-border-opacity)))}#memoizex-root .label a:hover{--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)))}#memoizex-root .menu li>*:not(ul,.menu-title,details,.btn):active,#memoizex-root .menu li>*:not(ul,.menu-title,details,.btn).active,#memoizex-root .menu li>details>summary:active{--tw-bg-opacity: 1;background-color:var(--fallback-n,oklch(var(--n)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-nc,oklch(var(--nc)/var(--tw-text-opacity)))}#memoizex-root .table tr.hover:hover,#memoizex-root .table tr.hover:nth-child(2n):hover{--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)))}#memoizex-root .\\!table tr.hover:hover,#memoizex-root .\\!table tr.hover:nth-child(2n):hover{--tw-bg-opacity: 1 !important;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)))!important}#memoizex-root .table-zebra tr.hover:hover,#memoizex-root .table-zebra tr.hover:nth-child(2n):hover{--tw-bg-opacity: 1;background-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-bg-opacity)))}}#memoizex-root .btn{display:inline-flex;height:48px;min-height:48px;flex-shrink:0;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none;flex-wrap:wrap;align-items:center;justify-content:center;border-radius:var(--rounded-btn, 8px);border-color:transparent;border-color:oklch(var(--btn-color, var(--b2)) / var(--tw-border-opacity));padding-left:16px;padding-right:16px;text-align:center;font-size:14px;line-height:1em;gap:8px;font-weight:600;text-decoration-line:none;transition-duration:.2s;transition-timing-function:cubic-bezier(0,0,.2,1);border-width:var(--border-btn, 1px);transition-property:color,background-color,border-color,opacity,box-shadow,transform;--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));--tw-shadow: 0 1px 2px 0 rgb(0 0 0 / .05);--tw-shadow-colored: 0 1px 2px 0 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-color:var(--fallback-bc,oklch(var(--bc)/1));background-color:oklch(var(--btn-color, var(--b2)) / var(--tw-bg-opacity));--tw-bg-opacity: 1;--tw-border-opacity: 1}#memoizex-root .btn-disabled,#memoizex-root .btn[disabled],#memoizex-root .btn:disabled{pointer-events:none}#memoizex-root :where(.btn:is(input[type=checkbox])),#memoizex-root :where(.btn:is(input[type=radio])){width:auto;-webkit-appearance:none;-moz-appearance:none;appearance:none}#memoizex-root .btn:is(input[type=checkbox]):after,#memoizex-root .btn:is(input[type=radio]):after{--tw-content: attr(aria-label);content:var(--tw-content)}#memoizex-root .card{position:relative;display:flex;flex-direction:column;border-radius:var(--rounded-box, 16px)}#memoizex-root .card:focus{outline:2px solid transparent;outline-offset:2px}#memoizex-root .card figure{display:flex;align-items:center;justify-content:center}#memoizex-root .card.image-full{display:grid}#memoizex-root .card.image-full:before{position:relative;content:"";z-index:10;border-radius:var(--rounded-box, 16px);--tw-bg-opacity: 1;background-color:var(--fallback-n,oklch(var(--n)/var(--tw-bg-opacity)));opacity:.75}#memoizex-root .card.image-full:before,#memoizex-root .card.image-full>*{grid-column-start:1;grid-row-start:1}#memoizex-root .card.image-full>figure img{height:100%;-o-object-fit:cover;object-fit:cover}#memoizex-root .card.image-full>.card-body{position:relative;z-index:20;--tw-text-opacity: 1;color:var(--fallback-nc,oklch(var(--nc)/var(--tw-text-opacity)))}#memoizex-root .checkbox{flex-shrink:0;--chkbg: var(--fallback-bc,oklch(var(--bc)/1));--chkfg: var(--fallback-b1,oklch(var(--b1)/1));height:24px;width:24px;cursor:pointer;-webkit-appearance:none;-moz-appearance:none;appearance:none;border-radius:var(--rounded-btn, 8px);border-width:1px;border-color:var(--fallback-bc,oklch(var(--bc)/var(--tw-border-opacity)));--tw-border-opacity: .2}#memoizex-root .divider{display:flex;flex-direction:row;align-items:center;align-self:stretch;margin-top:16px;margin-bottom:16px;height:16px;white-space:nowrap}#memoizex-root .divider:before,#memoizex-root .divider:after{height:2px;width:100%;flex-grow:1;--tw-content: "";content:var(--tw-content);background-color:var(--fallback-bc,oklch(var(--bc)/.1))}#memoizex-root .dropdown{position:relative;display:inline-block}#memoizex-root .dropdown>*:not(summary):focus{outline:2px solid transparent;outline-offset:2px}#memoizex-root .dropdown .dropdown-content{position:absolute}#memoizex-root .dropdown:is(:not(details)) .dropdown-content{visibility:hidden;opacity:0;transform-origin:top;--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));transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-timing-function:cubic-bezier(0,0,.2,1);transition-duration:.2s}#memoizex-root .dropdown-end .dropdown-content{inset-inline-end:0px}#memoizex-root .dropdown-left .dropdown-content{bottom:auto;inset-inline-end:100%;top:0;transform-origin:right}#memoizex-root .dropdown-right .dropdown-content{bottom:auto;inset-inline-start:100%;top:0;transform-origin:left}#memoizex-root .dropdown-bottom .dropdown-content{bottom:auto;top:100%;transform-origin:top}#memoizex-root .dropdown-top .dropdown-content{bottom:100%;top:auto;transform-origin:bottom}#memoizex-root .dropdown-end.dropdown-right .dropdown-content,#memoizex-root .dropdown-end.dropdown-left .dropdown-content{bottom:0;top:auto}#memoizex-root .dropdown.dropdown-open .dropdown-content,#memoizex-root .dropdown:not(.dropdown-hover):focus .dropdown-content,#memoizex-root .dropdown:focus-within .dropdown-content{visibility:visible;opacity:1}@media(hover:hover){#memoizex-root .dropdown.dropdown-hover:hover .dropdown-content{visibility:visible;opacity:1}#memoizex-root .btm-nav>*.disabled:hover,#memoizex-root .btm-nav>*[disabled]:hover{pointer-events:none;--tw-border-opacity: 0;background-color:var(--fallback-n,oklch(var(--n)/var(--tw-bg-opacity)));--tw-bg-opacity: .1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));--tw-text-opacity: .2}#memoizex-root .btn:hover{--tw-border-opacity: 1;border-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-bg-opacity)))}@supports (color: color-mix(in oklab,black,black)){#memoizex-root .btn:hover{background-color:color-mix(in oklab,oklch(var(--btn-color, var(--b2)) / var(--tw-bg-opacity, 1)) 90%,black);border-color:color-mix(in oklab,oklch(var(--btn-color, var(--b2)) / var(--tw-border-opacity, 1)) 90%,black)}}@supports not (color: oklch(0% 0 0)){#memoizex-root .btn:hover{background-color:var(--btn-color, var(--fallback-b2));border-color:var(--btn-color, var(--fallback-b2))}}#memoizex-root .btn.glass:hover{--glass-opacity: 25%;--glass-border-opacity: 15%}#memoizex-root .btn-ghost:hover{border-color:transparent}@supports (color: oklch(0% 0 0)){#memoizex-root .btn-ghost:hover{background-color:var(--fallback-bc,oklch(var(--bc)/.2))}}#memoizex-root .btn-outline.btn-primary:hover{--tw-text-opacity: 1;color:var(--fallback-pc,oklch(var(--pc)/var(--tw-text-opacity)))}@supports (color: color-mix(in oklab,black,black)){#memoizex-root .btn-outline.btn-primary:hover{background-color:color-mix(in oklab,var(--fallback-p,oklch(var(--p)/1)) 90%,black);border-color:color-mix(in oklab,var(--fallback-p,oklch(var(--p)/1)) 90%,black)}}#memoizex-root .btn-outline.btn-secondary:hover{--tw-text-opacity: 1;color:var(--fallback-sc,oklch(var(--sc)/var(--tw-text-opacity)))}@supports (color: color-mix(in oklab,black,black)){#memoizex-root .btn-outline.btn-secondary:hover{background-color:color-mix(in oklab,var(--fallback-s,oklch(var(--s)/1)) 90%,black);border-color:color-mix(in oklab,var(--fallback-s,oklch(var(--s)/1)) 90%,black)}}#memoizex-root .btn-outline.btn-warning:hover{--tw-text-opacity: 1;color:var(--fallback-wac,oklch(var(--wac)/var(--tw-text-opacity)))}@supports (color: color-mix(in oklab,black,black)){#memoizex-root .btn-outline.btn-warning:hover{background-color:color-mix(in oklab,var(--fallback-wa,oklch(var(--wa)/1)) 90%,black);border-color:color-mix(in oklab,var(--fallback-wa,oklch(var(--wa)/1)) 90%,black)}}#memoizex-root .btn-disabled:hover,#memoizex-root .btn[disabled]:hover,#memoizex-root .btn:disabled:hover{--tw-border-opacity: 0;background-color:var(--fallback-n,oklch(var(--n)/var(--tw-bg-opacity)));--tw-bg-opacity: .2;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));--tw-text-opacity: .2}@supports (color: color-mix(in oklab,black,black)){#memoizex-root .btn:is(input[type=checkbox]:checked):hover,#memoizex-root .btn:is(input[type=radio]:checked):hover{background-color:color-mix(in oklab,var(--fallback-p,oklch(var(--p)/1)) 90%,black);border-color:color-mix(in oklab,var(--fallback-p,oklch(var(--p)/1)) 90%,black)}}#memoizex-root .dropdown.dropdown-hover:hover .dropdown-content{--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))}#memoizex-root :where(.menu li:not(.menu-title,.disabled)>*:not(ul,details,.menu-title)):not(.active,.btn):hover,#memoizex-root :where(.menu li:not(.menu-title,.disabled)>details>summary:not(.menu-title)):not(.active,.btn):hover{cursor:pointer;outline:2px solid transparent;outline-offset:2px}@supports (color: oklch(0% 0 0)){#memoizex-root :where(.menu li:not(.menu-title,.disabled)>*:not(ul,details,.menu-title)):not(.active,.btn):hover,#memoizex-root :where(.menu li:not(.menu-title,.disabled)>details>summary:not(.menu-title)):not(.active,.btn):hover{background-color:var(--fallback-bc,oklch(var(--bc)/.1))}}}#memoizex-root .dropdown:is(details) summary::-webkit-details-marker{display:none}#memoizex-root .footer{display:grid;width:100%;grid-auto-flow:row;place-items:start;-moz-column-gap:16px;column-gap:16px;row-gap:40px;font-size:14px;line-height:20px}#memoizex-root .footer>*{display:grid;place-items:start;gap:8px}@media(min-width:48rem){#memoizex-root .footer{grid-auto-flow:column}#memoizex-root .footer-center{grid-auto-flow:row dense}}#memoizex-root .label{display:flex;-webkit-user-select:none;-moz-user-select:none;user-select:none;align-items:center;justify-content:space-between;padding:8px 4px}#memoizex-root .input{flex-shrink:1;-webkit-appearance:none;-moz-appearance:none;appearance:none;height:48px;padding-left:16px;padding-right:16px;font-size:16px;line-height:2;line-height:24px;border-radius:var(--rounded-btn, 8px);border-width:1px;border-color:transparent;--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))}#memoizex-root .input[type=number]::-webkit-inner-spin-button,#memoizex-root .input-md[type=number]::-webkit-inner-spin-button{margin-top:-16px;margin-bottom:-16px;margin-inline-end:-16px}#memoizex-root .input-xs[type=number]::-webkit-inner-spin-button{margin-top:-4px;margin-bottom:-4px;margin-inline-end:-0px}#memoizex-root .input-sm[type=number]::-webkit-inner-spin-button{margin-top:0;margin-bottom:0;margin-inline-end:-0px}#memoizex-root .join{display:inline-flex;align-items:stretch;border-radius:var(--rounded-btn, 8px)}#memoizex-root .join :where(.join-item){border-start-end-radius:0;border-end-end-radius:0;border-end-start-radius:0;border-start-start-radius:0}#memoizex-root .join .join-item:not(:first-child):not(:last-child),#memoizex-root .join *:not(:first-child):not(:last-child) .join-item{border-start-end-radius:0;border-end-end-radius:0;border-end-start-radius:0;border-start-start-radius:0}#memoizex-root .join .join-item:first-child:not(:last-child),#memoizex-root .join *:first-child:not(:last-child) .join-item{border-start-end-radius:0;border-end-end-radius:0}#memoizex-root .join .dropdown .join-item:first-child:not(:last-child),#memoizex-root .join *:first-child:not(:last-child) .dropdown .join-item{border-start-end-radius:inherit;border-end-end-radius:inherit}#memoizex-root .join :where(.join-item:first-child:not(:last-child)),#memoizex-root .join :where(*:first-child:not(:last-child) .join-item){border-end-start-radius:inherit;border-start-start-radius:inherit}#memoizex-root .join .join-item:last-child:not(:first-child),#memoizex-root .join *:last-child:not(:first-child) .join-item{border-end-start-radius:0;border-start-start-radius:0}#memoizex-root .join :where(.join-item:last-child:not(:first-child)),#memoizex-root .join :where(*:last-child:not(:first-child) .join-item){border-start-end-radius:inherit;border-end-end-radius:inherit}@supports not selector(:has(*)){#memoizex-root :where(.join *){border-radius:inherit}}@supports selector(:has(*)){#memoizex-root :where(.join *:has(.join-item)){border-radius:inherit}}#memoizex-root .link{cursor:pointer;text-decoration-line:underline}#memoizex-root .menu{display:flex;flex-direction:column;flex-wrap:wrap;font-size:14px;line-height:20px;padding:8px}#memoizex-root .menu :where(li ul){position:relative;white-space:nowrap;margin-inline-start:16px;padding-inline-start:8px}#memoizex-root .menu :where(li:not(.menu-title)>*:not(ul,details,.menu-title,.btn)),#memoizex-root .menu :where(li:not(.menu-title)>details>summary:not(.menu-title)){display:grid;grid-auto-flow:column;align-content:flex-start;align-items:center;gap:8px;grid-auto-columns:minmax(auto,max-content) auto max-content;-webkit-user-select:none;-moz-user-select:none;user-select:none}#memoizex-root .menu li.disabled{cursor:not-allowed;-webkit-user-select:none;-moz-user-select:none;user-select:none;color:var(--fallback-bc,oklch(var(--bc)/.3))}#memoizex-root .menu :where(li>.menu-dropdown:not(.menu-dropdown-show)){display:none}#memoizex-root :where(.menu li){position:relative;display:flex;flex-shrink:0;flex-direction:column;flex-wrap:wrap;align-items:stretch}#memoizex-root :where(.menu li) .badge{justify-self:end}#memoizex-root .modal{pointer-events:none;position:fixed;top:0;right:0;bottom:0;left:0;margin:0;display:grid;height:100%;max-height:none;width:100%;max-width:none;justify-items:center;padding:0;opacity:0;overscroll-behavior:contain;z-index:999;background-color:transparent;color:inherit;transition-duration:.2s;transition-timing-function:cubic-bezier(0,0,.2,1);transition-property:transform,opacity,visibility;overflow-y:hidden}#memoizex-root :where(.modal){align-items:center}#memoizex-root .modal-box{max-height:calc(100vh - 5em);grid-column-start:1;grid-row-start:1;width:91.666667%;max-width:512px;--tw-scale-x: .9;--tw-scale-y: .9;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));border-bottom-right-radius:var(--rounded-box, 16px);border-bottom-left-radius:var(--rounded-box, 16px);border-top-left-radius:var(--rounded-box, 16px);border-top-right-radius:var(--rounded-box, 16px);--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)));padding:24px;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-timing-function:cubic-bezier(0,0,.2,1);transition-duration:.2s;box-shadow:#00000040 0 25px 50px -12px;overflow-y:auto;overscroll-behavior:contain}#memoizex-root .modal-open,#memoizex-root .modal:target,#memoizex-root .modal-toggle:checked+.modal,#memoizex-root .modal[open]{pointer-events:auto;visibility:visible;opacity:1}#memoizex-root:has(:is(.modal-open,.modal:target,.modal-toggle:checked+.modal,.modal[open])){overflow:hidden;scrollbar-gutter:stable}#memoizex-root .progress{position:relative;width:100%;-webkit-appearance:none;-moz-appearance:none;appearance:none;overflow:hidden;height:8px;border-radius:var(--rounded-box, 16px);background-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .select{display:inline-flex;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none;-webkit-appearance:none;-moz-appearance:none;appearance:none;height:48px;min-height:48px;padding-inline-start:16px;padding-inline-end:40px;font-size:14px;line-height:20px;line-height:2;border-radius:var(--rounded-btn, 8px);border-width:1px;border-color:transparent;--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)));background-image:linear-gradient(45deg,transparent 50%,currentColor 50%),linear-gradient(135deg,currentColor 50%,transparent 50%);background-position:calc(100% - 20px) calc(1px + 50%),calc(100% - 16.1px) calc(1px + 50%);background-size:4px 4px,4px 4px;background-repeat:no-repeat}#memoizex-root .select[multiple]{height:auto}#memoizex-root .stats{display:inline-grid;border-radius:var(--rounded-box, 16px);--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)))}#memoizex-root :where(.stats){grid-auto-flow:column;overflow-x:auto}#memoizex-root .\\!table{position:relative!important;width:100%!important;border-radius:var(--rounded-box, 16px)!important;text-align:left!important;font-size:14px!important;line-height:20px!important}#memoizex-root .table{position:relative;width:100%;border-radius:var(--rounded-box, 16px);text-align:left;font-size:14px;line-height:20px}#memoizex-root .\\!table :where(.table-pin-rows thead tr){position:sticky!important;top:0!important;z-index:1!important;--tw-bg-opacity: 1 !important;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))!important}#memoizex-root .table :where(.table-pin-rows thead tr){position:sticky;top:0;z-index:1;--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))}#memoizex-root .\\!table :where(.table-pin-rows tfoot tr){position:sticky!important;bottom:0!important;z-index:1!important;--tw-bg-opacity: 1 !important;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))!important}#memoizex-root .table :where(.table-pin-rows tfoot tr){position:sticky;bottom:0;z-index:1;--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))}#memoizex-root .\\!table :where(.table-pin-cols tr th){position:sticky!important;left:0!important;right:0!important;--tw-bg-opacity: 1 !important;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))!important}#memoizex-root .table :where(.table-pin-cols tr th){position:sticky;left:0;right:0;--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))}#memoizex-root .table-zebra tbody tr:nth-child(2n) :where(.table-pin-cols tr th){--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)))}#memoizex-root .textarea{min-height:48px;flex-shrink:1;padding:8px 16px;font-size:14px;line-height:20px;line-height:2;border-radius:var(--rounded-btn, 8px);border-width:1px;border-color:transparent;--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))}#memoizex-root .timeline{position:relative;display:flex}#memoizex-root :where(.timeline>li){position:relative;display:grid;flex-shrink:0;align-items:center;grid-template-rows:var(--timeline-row-start, minmax(0, 1fr)) auto var( --timeline-row-end, minmax(0, 1fr) );grid-template-columns:var(--timeline-col-start, minmax(0, 1fr)) auto var( --timeline-col-end, minmax(0, 1fr) )}#memoizex-root .timeline>li>hr{width:100%;border-width:0px}#memoizex-root :where(.timeline>li>hr):first-child{grid-column-start:1;grid-row-start:2}#memoizex-root :where(.timeline>li>hr):last-child{grid-column-start:3;grid-column-end:none;grid-row-start:2;grid-row-end:auto}#memoizex-root .toggle{flex-shrink:0;--tglbg: var(--fallback-b1,oklch(var(--b1)/1));--handleoffset: 24px;--handleoffsetcalculator: calc(var(--handleoffset) * -1);--togglehandleborder: 0 0;height:24px;width:48px;cursor:pointer;-webkit-appearance:none;-moz-appearance:none;appearance:none;border-radius:var(--rounded-badge, 30.4px);border-width:1px;border-color:currentColor;background-color:currentColor;color:var(--fallback-bc,oklch(var(--bc)/.5));transition:background,box-shadow var(--animation-input, .2s) ease-out;box-shadow:var(--handleoffsetcalculator) 0 0 2px var(--tglbg) inset,0 0 0 2px var(--tglbg) inset,var(--togglehandleborder)}#memoizex-root .alert-error{border-color:var(--fallback-er,oklch(var(--er)/.2));--tw-text-opacity: 1;color:var(--fallback-erc,oklch(var(--erc)/var(--tw-text-opacity)));--alert-bg: var(--fallback-er,oklch(var(--er)/1));--alert-bg-mix: var(--fallback-b1,oklch(var(--b1)/1))}#memoizex-root .avatar-group :where(.avatar){overflow:hidden;border-radius:9999px;border-width:4px;--tw-border-opacity: 1;border-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-border-opacity)))}#memoizex-root .badge-accent{--tw-border-opacity: 1;border-color:var(--fallback-a,oklch(var(--a)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-a,oklch(var(--a)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-ac,oklch(var(--ac)/var(--tw-text-opacity)))}#memoizex-root .badge-success{border-color:transparent;--tw-bg-opacity: 1;background-color:var(--fallback-su,oklch(var(--su)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-suc,oklch(var(--suc)/var(--tw-text-opacity)))}#memoizex-root .badge-ghost{--tw-border-opacity: 1;border-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)))}#memoizex-root .badge-outline.badge-accent{--tw-text-opacity: 1;color:var(--fallback-a,oklch(var(--a)/var(--tw-text-opacity)))}#memoizex-root .badge-outline.badge-success{--tw-text-opacity: 1;color:var(--fallback-su,oklch(var(--su)/var(--tw-text-opacity)))}#memoizex-root .btm-nav>*:where(.active){border-top-width:2px;--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)))}#memoizex-root .btm-nav>*.disabled,#memoizex-root .btm-nav>*[disabled]{pointer-events:none;--tw-border-opacity: 0;background-color:var(--fallback-n,oklch(var(--n)/var(--tw-bg-opacity)));--tw-bg-opacity: .1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));--tw-text-opacity: .2}#memoizex-root .btm-nav>* .label{font-size:16px;line-height:24px}@media(prefers-reduced-motion:no-preference){#memoizex-root .btn{animation:button-pop var(--animation-btn, .25s) ease-out}}#memoizex-root .btn:active:hover,#memoizex-root .btn:active:focus{animation:button-pop 0s ease-out;transform:scale(var(--btn-focus-scale, .97))}@supports not (color: oklch(0% 0 0)){#memoizex-root .btn{background-color:var(--btn-color, var(--fallback-b2));border-color:var(--btn-color, var(--fallback-b2))}#memoizex-root .btn-primary{--btn-color: var(--fallback-p)}#memoizex-root .btn-secondary{--btn-color: var(--fallback-s)}#memoizex-root .btn-neutral{--btn-color: var(--fallback-n)}#memoizex-root .btn-warning{--btn-color: var(--fallback-wa)}}@supports (color: color-mix(in oklab,black,black)){#memoizex-root .btn-outline.btn-primary.btn-active{background-color:color-mix(in oklab,var(--fallback-p,oklch(var(--p)/1)) 90%,black);border-color:color-mix(in oklab,var(--fallback-p,oklch(var(--p)/1)) 90%,black)}#memoizex-root .btn-outline.btn-secondary.btn-active{background-color:color-mix(in oklab,var(--fallback-s,oklch(var(--s)/1)) 90%,black);border-color:color-mix(in oklab,var(--fallback-s,oklch(var(--s)/1)) 90%,black)}#memoizex-root .btn-outline.btn-warning.btn-active{background-color:color-mix(in oklab,var(--fallback-wa,oklch(var(--wa)/1)) 90%,black);border-color:color-mix(in oklab,var(--fallback-wa,oklch(var(--wa)/1)) 90%,black)}}#memoizex-root .btn:focus-visible{outline-style:solid;outline-width:2px;outline-offset:2px}#memoizex-root .btn-primary{--tw-text-opacity: 1;color:var(--fallback-pc,oklch(var(--pc)/var(--tw-text-opacity)));outline-color:var(--fallback-p,oklch(var(--p)/1))}@supports (color: oklch(0% 0 0)){#memoizex-root .btn-primary{--btn-color: var(--p)}#memoizex-root .btn-secondary{--btn-color: var(--s)}#memoizex-root .btn-neutral{--btn-color: var(--n)}#memoizex-root .btn-warning{--btn-color: var(--wa)}}#memoizex-root .btn-secondary{--tw-text-opacity: 1;color:var(--fallback-sc,oklch(var(--sc)/var(--tw-text-opacity)));outline-color:var(--fallback-s,oklch(var(--s)/1))}#memoizex-root .btn-neutral{--tw-text-opacity: 1;color:var(--fallback-nc,oklch(var(--nc)/var(--tw-text-opacity)));outline-color:var(--fallback-n,oklch(var(--n)/1))}#memoizex-root .btn-warning{--tw-text-opacity: 1;color:var(--fallback-wac,oklch(var(--wac)/var(--tw-text-opacity)));outline-color:var(--fallback-wa,oklch(var(--wa)/1))}#memoizex-root .btn.glass{--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow);outline-color:currentColor}#memoizex-root .btn.glass.btn-active{--glass-opacity: 25%;--glass-border-opacity: 15%}#memoizex-root .btn-ghost{border-width:1px;border-color:transparent;background-color:transparent;color:currentColor;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000),var(--tw-ring-shadow, 0 0 #0000),var(--tw-shadow);outline-color:currentColor}#memoizex-root .btn-ghost.btn-active{border-color:transparent;background-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .btn-outline.btn-primary{--tw-text-opacity: 1;color:var(--fallback-p,oklch(var(--p)/var(--tw-text-opacity)))}#memoizex-root .btn-outline.btn-primary.btn-active{--tw-text-opacity: 1;color:var(--fallback-pc,oklch(var(--pc)/var(--tw-text-opacity)))}#memoizex-root .btn-outline.btn-secondary{--tw-text-opacity: 1;color:var(--fallback-s,oklch(var(--s)/var(--tw-text-opacity)))}#memoizex-root .btn-outline.btn-secondary.btn-active{--tw-text-opacity: 1;color:var(--fallback-sc,oklch(var(--sc)/var(--tw-text-opacity)))}#memoizex-root .btn-outline.btn-warning{--tw-text-opacity: 1;color:var(--fallback-wa,oklch(var(--wa)/var(--tw-text-opacity)))}#memoizex-root .btn-outline.btn-warning.btn-active{--tw-text-opacity: 1;color:var(--fallback-wac,oklch(var(--wac)/var(--tw-text-opacity)))}#memoizex-root .btn.btn-disabled,#memoizex-root .btn[disabled],#memoizex-root .btn:disabled{--tw-border-opacity: 0;background-color:var(--fallback-n,oklch(var(--n)/var(--tw-bg-opacity)));--tw-bg-opacity: .2;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));--tw-text-opacity: .2}#memoizex-root .btn:is(input[type=checkbox]:checked),#memoizex-root .btn:is(input[type=radio]:checked){--tw-border-opacity: 1;border-color:var(--fallback-p,oklch(var(--p)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-p,oklch(var(--p)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-pc,oklch(var(--pc)/var(--tw-text-opacity)))}#memoizex-root .btn:is(input[type=checkbox]:checked):focus-visible,#memoizex-root .btn:is(input[type=radio]:checked):focus-visible{outline-color:var(--fallback-p,oklch(var(--p)/1))}@keyframes button-pop{0%{transform:scale(var(--btn-focus-scale, .98))}40%{transform:scale(1.02)}to{transform:scale(1)}}#memoizex-root .card :where(figure:first-child){overflow:hidden;border-start-start-radius:inherit;border-start-end-radius:inherit;border-end-start-radius:unset;border-end-end-radius:unset}#memoizex-root .card :where(figure:last-child){overflow:hidden;border-start-start-radius:unset;border-start-end-radius:unset;border-end-start-radius:inherit;border-end-end-radius:inherit}#memoizex-root .card:focus-visible{outline:2px solid currentColor;outline-offset:2px}#memoizex-root .card.bordered{border-width:1px;--tw-border-opacity: 1;border-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)))}#memoizex-root .card.compact .card-body{padding:16px;font-size:14px;line-height:20px}#memoizex-root .card.image-full :where(figure){overflow:hidden;border-radius:inherit}#memoizex-root .checkbox:focus{box-shadow:none}#memoizex-root .checkbox:focus-visible{outline-style:solid;outline-width:2px;outline-offset:2px;outline-color:var(--fallback-bc,oklch(var(--bc)/1))}#memoizex-root .checkbox:disabled{border-width:0px;cursor:not-allowed;border-color:transparent;--tw-bg-opacity: 1;background-color:var(--fallback-bc,oklch(var(--bc)/var(--tw-bg-opacity)));opacity:.2}#memoizex-root .checkbox:checked,#memoizex-root .checkbox[aria-checked=true]{background-repeat:no-repeat;animation:checkmark var(--animation-input, .2s) ease-out;background-color:var(--chkbg);background-image:linear-gradient(-45deg,transparent 65%,var(--chkbg) 65.99%),linear-gradient(45deg,transparent 75%,var(--chkbg) 75.99%),linear-gradient(-45deg,var(--chkbg) 40%,transparent 40.99%),linear-gradient(45deg,var(--chkbg) 30%,var(--chkfg) 30.99%,var(--chkfg) 40%,transparent 40.99%),linear-gradient(-45deg,var(--chkfg) 50%,var(--chkbg) 50.99%)}#memoizex-root .checkbox:indeterminate{--tw-bg-opacity: 1;background-color:var(--fallback-bc,oklch(var(--bc)/var(--tw-bg-opacity)));background-repeat:no-repeat;animation:checkmark var(--animation-input, .2s) ease-out;background-image:linear-gradient(90deg,transparent 80%,var(--chkbg) 80%),linear-gradient(-90deg,transparent 80%,var(--chkbg) 80%),linear-gradient(0deg,var(--chkbg) 43%,var(--chkfg) 43%,var(--chkfg) 57%,var(--chkbg) 57%)}#memoizex-root .checkbox-accent{--chkbg: var(--fallback-a,oklch(var(--a)/1));--chkfg: var(--fallback-ac,oklch(var(--ac)/1));--tw-border-opacity: 1;border-color:var(--fallback-a,oklch(var(--a)/var(--tw-border-opacity)))}#memoizex-root .checkbox-accent:focus-visible{outline-color:var(--fallback-a,oklch(var(--a)/1))}#memoizex-root .checkbox-accent:checked,#memoizex-root .checkbox-accent[aria-checked=true]{--tw-border-opacity: 1;border-color:var(--fallback-a,oklch(var(--a)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-a,oklch(var(--a)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-ac,oklch(var(--ac)/var(--tw-text-opacity)))}@keyframes checkmark{0%{background-position-y:5px}50%{background-position-y:-2px}to{background-position-y:0}}#memoizex-root .divider:not(:empty){gap:16px}#memoizex-root .dropdown.dropdown-open .dropdown-content,#memoizex-root .dropdown:focus .dropdown-content,#memoizex-root .dropdown:focus-within .dropdown-content{--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))}#memoizex-root .label-text{font-size:14px;line-height:20px;--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)))}#memoizex-root .input input{--tw-bg-opacity: 1;background-color:var(--fallback-p,oklch(var(--p)/var(--tw-bg-opacity)));background-color:transparent}#memoizex-root .input input:focus{outline:2px solid transparent;outline-offset:2px}#memoizex-root .input[list]::-webkit-calendar-picker-indicator{line-height:1em}#memoizex-root .input-bordered{border-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .input:focus,#memoizex-root .input:focus-within{box-shadow:none;border-color:var(--fallback-bc,oklch(var(--bc)/.2));outline-style:solid;outline-width:2px;outline-offset:2px;outline-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .input:has(>input[disabled]),#memoizex-root .input-disabled,#memoizex-root .input:disabled,#memoizex-root .input[disabled]{cursor:not-allowed;--tw-border-opacity: 1;border-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)));color:var(--fallback-bc,oklch(var(--bc)/.4))}#memoizex-root .input:has(>input[disabled])::-moz-placeholder,#memoizex-root .input-disabled::-moz-placeholder,#memoizex-root .input:disabled::-moz-placeholder,#memoizex-root .input[disabled]::-moz-placeholder{color:var(--fallback-bc,oklch(var(--bc)/var(--tw-placeholder-opacity)));--tw-placeholder-opacity: .2}#memoizex-root .input:has(>input[disabled])::placeholder,#memoizex-root .input-disabled::placeholder,#memoizex-root .input:disabled::placeholder,#memoizex-root .input[disabled]::placeholder{color:var(--fallback-bc,oklch(var(--bc)/var(--tw-placeholder-opacity)));--tw-placeholder-opacity: .2}#memoizex-root .input:has(>input[disabled])>input[disabled]{cursor:not-allowed}#memoizex-root .input::-webkit-date-and-time-value{text-align:inherit}#memoizex-root .join>:where(*:not(:first-child)){margin-top:0;margin-bottom:0;margin-inline-start:-1px}#memoizex-root .join>:where(*:not(:first-child)):is(.btn){margin-inline-start:calc(var(--border-btn) * -1)}#memoizex-root .join-item:focus{isolation:isolate}#memoizex-root .link:focus{outline:2px solid transparent;outline-offset:2px}#memoizex-root .link:focus-visible{outline:2px solid currentColor;outline-offset:2px}#memoizex-root .loading{pointer-events:none;display:inline-block;aspect-ratio:1 / 1;width:24px;background-color:currentColor;-webkit-mask-size:100%;mask-size:100%;-webkit-mask-repeat:no-repeat;mask-repeat:no-repeat;-webkit-mask-position:center;mask-position:center;-webkit-mask-image:url("data:image/svg+xml,%3Csvg width='24' height='24' stroke='black' viewBox='0 0 24 24' xmlns='http://www.w3.org/2000/svg'%3E%3Cg transform-origin='center'%3E%3Ccircle cx='12' cy='12' r='9.5' fill='none' stroke-width='3' stroke-linecap='round'%3E%3CanimateTransform attributeName='transform' type='rotate' from='0 12 12' to='360 12 12' dur='2s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dasharray' values='0,150;42,150;42,150' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dashoffset' values='0;-16;-59' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3C/circle%3E%3C/g%3E%3C/svg%3E");mask-image:url("data:image/svg+xml,%3Csvg width='24' height='24' stroke='black' viewBox='0 0 24 24' xmlns='http://www.w3.org/2000/svg'%3E%3Cg transform-origin='center'%3E%3Ccircle cx='12' cy='12' r='9.5' fill='none' stroke-width='3' stroke-linecap='round'%3E%3CanimateTransform attributeName='transform' type='rotate' from='0 12 12' to='360 12 12' dur='2s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dasharray' values='0,150;42,150;42,150' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dashoffset' values='0;-16;-59' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3C/circle%3E%3C/g%3E%3C/svg%3E")}#memoizex-root .loading-spinner{-webkit-mask-image:url("data:image/svg+xml,%3Csvg width='24' height='24' stroke='black' viewBox='0 0 24 24' xmlns='http://www.w3.org/2000/svg'%3E%3Cg transform-origin='center'%3E%3Ccircle cx='12' cy='12' r='9.5' fill='none' stroke-width='3' stroke-linecap='round'%3E%3CanimateTransform attributeName='transform' type='rotate' from='0 12 12' to='360 12 12' dur='2s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dasharray' values='0,150;42,150;42,150' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dashoffset' values='0;-16;-59' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3C/circle%3E%3C/g%3E%3C/svg%3E");mask-image:url("data:image/svg+xml,%3Csvg width='24' height='24' stroke='black' viewBox='0 0 24 24' xmlns='http://www.w3.org/2000/svg'%3E%3Cg transform-origin='center'%3E%3Ccircle cx='12' cy='12' r='9.5' fill='none' stroke-width='3' stroke-linecap='round'%3E%3CanimateTransform attributeName='transform' type='rotate' from='0 12 12' to='360 12 12' dur='2s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dasharray' values='0,150;42,150;42,150' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3Canimate attributeName='stroke-dashoffset' values='0;-16;-59' keyTimes='0;0.475;1' dur='1.5s' repeatCount='indefinite'/%3E%3C/circle%3E%3C/g%3E%3C/svg%3E")}#memoizex-root :where(.menu li:empty){--tw-bg-opacity: 1;background-color:var(--fallback-bc,oklch(var(--bc)/var(--tw-bg-opacity)));opacity:.1;margin:8px 16px;height:1px}#memoizex-root .menu :where(li ul):before{position:absolute;bottom:12px;inset-inline-start:0px;top:12px;width:1px;--tw-bg-opacity: 1;background-color:var(--fallback-bc,oklch(var(--bc)/var(--tw-bg-opacity)));opacity:.1;content:""}#memoizex-root .menu :where(li:not(.menu-title)>*:not(ul,details,.menu-title,.btn)),#memoizex-root .menu :where(li:not(.menu-title)>details>summary:not(.menu-title)){border-radius:var(--rounded-btn, 8px);padding:8px 16px;text-align:start;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-timing-function:cubic-bezier(0,0,.2,1);transition-duration:.2s;text-wrap:balance}#memoizex-root :where(.menu li:not(.menu-title,.disabled)>*:not(ul,details,.menu-title)):not(summary,.active,.btn).focus,#memoizex-root :where(.menu li:not(.menu-title,.disabled)>*:not(ul,details,.menu-title)):not(summary,.active,.btn):focus,#memoizex-root :where(.menu li:not(.menu-title,.disabled)>*:not(ul,details,.menu-title)):is(summary):not(.active,.btn):focus-visible,#memoizex-root :where(.menu li:not(.menu-title,.disabled)>details>summary:not(.menu-title)):not(summary,.active,.btn).focus,#memoizex-root :where(.menu li:not(.menu-title,.disabled)>details>summary:not(.menu-title)):not(summary,.active,.btn):focus,#memoizex-root :where(.menu li:not(.menu-title,.disabled)>details>summary:not(.menu-title)):is(summary):not(.active,.btn):focus-visible{cursor:pointer;background-color:var(--fallback-bc,oklch(var(--bc)/.1));--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));outline:2px solid transparent;outline-offset:2px}#memoizex-root .menu li>*:not(ul,.menu-title,details,.btn):active,#memoizex-root .menu li>*:not(ul,.menu-title,details,.btn).active,#memoizex-root .menu li>details>summary:active{--tw-bg-opacity: 1;background-color:var(--fallback-n,oklch(var(--n)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-nc,oklch(var(--nc)/var(--tw-text-opacity)))}#memoizex-root .menu :where(li>details>summary)::-webkit-details-marker{display:none}#memoizex-root .menu :where(li>details>summary):after,#memoizex-root .menu :where(li>.menu-dropdown-toggle):after{justify-self:end;display:block;margin-top:-8px;height:8px;width:8px;transform:rotate(45deg);transition-property:transform,margin-top;transition-duration:.3s;transition-timing-function:cubic-bezier(.4,0,.2,1);content:"";transform-origin:75% 75%;box-shadow:2px 2px;pointer-events:none}#memoizex-root .menu :where(li>details[open]>summary):after,#memoizex-root .menu :where(li>.menu-dropdown-toggle.menu-dropdown-show):after{transform:rotate(225deg);margin-top:0}#memoizex-root .mockup-phone .display{overflow:hidden;border-radius:40px;margin-top:-25px}#memoizex-root .mockup-browser .mockup-browser-toolbar .input{position:relative;margin-left:auto;margin-right:auto;display:block;height:28px;width:384px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)));padding-left:32px;direction:ltr}#memoizex-root .mockup-browser .mockup-browser-toolbar .input:before{content:"";position:absolute;left:8px;top:50%;aspect-ratio:1 / 1;height:12px;--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));border-radius:9999px;border-width:2px;border-color:currentColor;opacity:.6}#memoizex-root .mockup-browser .mockup-browser-toolbar .input:after{content:"";position:absolute;left:20px;top:50%;height:8px;--tw-translate-y: 25%;--tw-rotate: -45deg;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));border-radius:9999px;border-width:1px;border-color:currentColor;opacity:.6}#memoizex-root .modal:not(dialog:not(.modal-open)),#memoizex-root .modal::backdrop{background-color:#0006;animation:modal-pop .2s ease-out}#memoizex-root .modal-backdrop{z-index:-1;grid-column-start:1;grid-row-start:1;display:grid;align-self:stretch;justify-self:stretch;color:transparent}#memoizex-root .modal-open .modal-box,#memoizex-root .modal-toggle:checked+.modal .modal-box,#memoizex-root .modal:target .modal-box,#memoizex-root .modal[open] .modal-box{--tw-translate-y: 0px;--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))}@keyframes modal-pop{0%{opacity:0}}#memoizex-root .progress::-moz-progress-bar{border-radius:var(--rounded-box, 16px);background-color:currentColor}#memoizex-root .progress-primary::-moz-progress-bar{border-radius:var(--rounded-box, 16px);--tw-bg-opacity: 1;background-color:var(--fallback-p,oklch(var(--p)/var(--tw-bg-opacity)))}#memoizex-root .progress-secondary::-moz-progress-bar{border-radius:var(--rounded-box, 16px);--tw-bg-opacity: 1;background-color:var(--fallback-s,oklch(var(--s)/var(--tw-bg-opacity)))}#memoizex-root .progress:indeterminate{--progress-color: var(--fallback-bc,oklch(var(--bc)/1));background-image:repeating-linear-gradient(90deg,var(--progress-color) -1%,var(--progress-color) 10%,transparent 10%,transparent 90%);background-size:200%;background-position-x:15%;animation:progress-loading 5s ease-in-out infinite}#memoizex-root .progress-primary:indeterminate{--progress-color: var(--fallback-p,oklch(var(--p)/1))}#memoizex-root .progress-secondary:indeterminate{--progress-color: var(--fallback-s,oklch(var(--s)/1))}#memoizex-root .progress::-webkit-progress-bar{border-radius:var(--rounded-box, 16px);background-color:transparent}#memoizex-root .progress::-webkit-progress-value{border-radius:var(--rounded-box, 16px);background-color:currentColor}#memoizex-root .progress-primary::-webkit-progress-value{--tw-bg-opacity: 1;background-color:var(--fallback-p,oklch(var(--p)/var(--tw-bg-opacity)))}#memoizex-root .progress-secondary::-webkit-progress-value{--tw-bg-opacity: 1;background-color:var(--fallback-s,oklch(var(--s)/var(--tw-bg-opacity)))}#memoizex-root .progress:indeterminate::-moz-progress-bar{background-color:transparent;background-image:repeating-linear-gradient(90deg,var(--progress-color) -1%,var(--progress-color) 10%,transparent 10%,transparent 90%);background-size:200%;background-position-x:15%;animation:progress-loading 5s ease-in-out infinite}@keyframes progress-loading{50%{background-position-x:-115%}}@keyframes radiomark{0%{box-shadow:0 0 0 12px var(--fallback-b1,oklch(var(--b1)/1)) inset,0 0 0 12px var(--fallback-b1,oklch(var(--b1)/1)) inset}50%{box-shadow:0 0 0 3px var(--fallback-b1,oklch(var(--b1)/1)) inset,0 0 0 3px var(--fallback-b1,oklch(var(--b1)/1)) inset}to{box-shadow:0 0 0 4px var(--fallback-b1,oklch(var(--b1)/1)) inset,0 0 0 4px var(--fallback-b1,oklch(var(--b1)/1)) inset}}@keyframes rating-pop{0%{transform:translateY(-.125em)}40%{transform:translateY(-.125em)}to{transform:translateY(0)}}#memoizex-root .select-bordered{border-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .select:focus{box-shadow:none;border-color:var(--fallback-bc,oklch(var(--bc)/.2));outline-style:solid;outline-width:2px;outline-offset:2px;outline-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .select-disabled,#memoizex-root .select:disabled,#memoizex-root .select[disabled]{cursor:not-allowed;--tw-border-opacity: 1;border-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)));color:var(--fallback-bc,oklch(var(--bc)/.4))}#memoizex-root .select-disabled::-moz-placeholder,#memoizex-root .select:disabled::-moz-placeholder,#memoizex-root .select[disabled]::-moz-placeholder{color:var(--fallback-bc,oklch(var(--bc)/var(--tw-placeholder-opacity)));--tw-placeholder-opacity: .2}#memoizex-root .select-disabled::placeholder,#memoizex-root .select:disabled::placeholder,#memoizex-root .select[disabled]::placeholder{color:var(--fallback-bc,oklch(var(--bc)/var(--tw-placeholder-opacity)));--tw-placeholder-opacity: .2}#memoizex-root .select-multiple,#memoizex-root .select[multiple],#memoizex-root .select[size].select:not([size="1"]){background-image:none;padding-right:16px}#memoizex-root [dir=rtl] .select{background-position:calc(0% + 12px) calc(1px + 50%),calc(0% + 16px) calc(1px + 50%)}@keyframes skeleton{0%{background-position:150%}to{background-position:-50%}}#memoizex-root :where(.stats)>:not([hidden])~:not([hidden]){--tw-divide-x-reverse: 0;border-right-width:calc(1px * var(--tw-divide-x-reverse));border-left-width:calc(1px * calc(1 - var(--tw-divide-x-reverse)));--tw-divide-y-reverse: 0;border-top-width:calc(0px * calc(1 - var(--tw-divide-y-reverse)));border-bottom-width:calc(0px * var(--tw-divide-y-reverse))}#memoizex-root [dir=rtl] .stats>*:not([hidden])~*:not([hidden]){--tw-divide-x-reverse: 1}#memoizex-root .\\!table:where([dir=rtl],[dir=rtl] *){text-align:right!important}#memoizex-root .table:where([dir=rtl],[dir=rtl] *){text-align:right}#memoizex-root .\\!table :where(th,td){padding:12px 16px!important;vertical-align:middle!important}#memoizex-root .table :where(th,td){padding:12px 16px;vertical-align:middle}#memoizex-root .table tr.active,#memoizex-root .table tr.active:nth-child(2n),#memoizex-root .table-zebra tbody tr:nth-child(2n){--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)))}#memoizex-root .\\!table tr.active,#memoizex-root .\\!table tr.active:nth-child(2n){--tw-bg-opacity: 1 !important;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)))!important}#memoizex-root .table-zebra tr.active,#memoizex-root .table-zebra tr.active:nth-child(2n),#memoizex-root .table-zebra-zebra tbody tr:nth-child(2n){--tw-bg-opacity: 1;background-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-bg-opacity)))}#memoizex-root .\\!table :where(thead tr,tbody tr:not(:last-child),tbody tr:first-child:last-child){border-bottom-width:1px!important;--tw-border-opacity: 1 !important;border-bottom-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)))!important}#memoizex-root .table :where(thead tr,tbody tr:not(:last-child),tbody tr:first-child:last-child){border-bottom-width:1px;--tw-border-opacity: 1;border-bottom-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)))}#memoizex-root .\\!table :where(thead,tfoot){white-space:nowrap!important;font-size:12px!important;line-height:16px!important;font-weight:700!important;color:var(--fallback-bc,oklch(var(--bc)/.6))!important}#memoizex-root .table :where(thead,tfoot){white-space:nowrap;font-size:12px;line-height:16px;font-weight:700;color:var(--fallback-bc,oklch(var(--bc)/.6))}#memoizex-root .\\!table :where(tfoot){border-top-width:1px!important;--tw-border-opacity: 1 !important;border-top-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)))!important}#memoizex-root .table :where(tfoot){border-top-width:1px;--tw-border-opacity: 1;border-top-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)))}#memoizex-root .textarea-bordered{border-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .textarea:focus{box-shadow:none;border-color:var(--fallback-bc,oklch(var(--bc)/.2));outline-style:solid;outline-width:2px;outline-offset:2px;outline-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .textarea-disabled,#memoizex-root .textarea:disabled,#memoizex-root .textarea[disabled]{cursor:not-allowed;--tw-border-opacity: 1;border-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-border-opacity)));--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity)));color:var(--fallback-bc,oklch(var(--bc)/.4))}#memoizex-root .textarea-disabled::-moz-placeholder,#memoizex-root .textarea:disabled::-moz-placeholder,#memoizex-root .textarea[disabled]::-moz-placeholder{color:var(--fallback-bc,oklch(var(--bc)/var(--tw-placeholder-opacity)));--tw-placeholder-opacity: .2}#memoizex-root .textarea-disabled::placeholder,#memoizex-root .textarea:disabled::placeholder,#memoizex-root .textarea[disabled]::placeholder{color:var(--fallback-bc,oklch(var(--bc)/var(--tw-placeholder-opacity)));--tw-placeholder-opacity: .2}#memoizex-root .timeline hr{height:4px}#memoizex-root :where(.timeline hr){--tw-bg-opacity: 1;background-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-bg-opacity)))}#memoizex-root :where(.timeline:has(.timeline-middle) hr):first-child{border-start-end-radius:var(--rounded-badge, 30.4px);border-end-end-radius:var(--rounded-badge, 30.4px);border-start-start-radius:0px;border-end-start-radius:0px}#memoizex-root :where(.timeline:has(.timeline-middle) hr):last-child{border-start-start-radius:var(--rounded-badge, 30.4px);border-end-start-radius:var(--rounded-badge, 30.4px);border-start-end-radius:0px;border-end-end-radius:0px}#memoizex-root :where(.timeline:not(:has(.timeline-middle)) :first-child hr:last-child){border-start-start-radius:var(--rounded-badge, 30.4px);border-end-start-radius:var(--rounded-badge, 30.4px);border-start-end-radius:0px;border-end-end-radius:0px}#memoizex-root :where(.timeline:not(:has(.timeline-middle)) :last-child hr:first-child){border-start-end-radius:var(--rounded-badge, 30.4px);border-end-end-radius:var(--rounded-badge, 30.4px);border-start-start-radius:0px;border-end-start-radius:0px}@keyframes toast-pop{0%{transform:scale(.9);opacity:0}to{transform:scale(1);opacity:1}}#memoizex-root [dir=rtl] .toggle{--handleoffsetcalculator: calc(var(--handleoffset) * 1)}#memoizex-root .toggle:focus-visible{outline-style:solid;outline-width:2px;outline-offset:2px;outline-color:var(--fallback-bc,oklch(var(--bc)/.2))}#memoizex-root .toggle:hover{background-color:currentColor}#memoizex-root .toggle:checked,#memoizex-root .toggle[aria-checked=true]{background-image:none;--handleoffsetcalculator: var(--handleoffset);--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)))}#memoizex-root [dir=rtl] .toggle:checked,#memoizex-root [dir=rtl] .toggle[aria-checked=true]{--handleoffsetcalculator: calc(var(--handleoffset) * -1)}#memoizex-root .toggle:indeterminate{--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity)));box-shadow:calc(var(--handleoffset) / 2) 0 0 2px var(--tglbg) inset,calc(var(--handleoffset) / -2) 0 0 2px var(--tglbg) inset,0 0 0 2px var(--tglbg) inset}#memoizex-root [dir=rtl] .toggle:indeterminate{box-shadow:calc(var(--handleoffset) / 2) 0 0 2px var(--tglbg) inset,calc(var(--handleoffset) / -2) 0 0 2px var(--tglbg) inset,0 0 0 2px var(--tglbg) inset}#memoizex-root .toggle-primary:focus-visible{outline-color:var(--fallback-p,oklch(var(--p)/1))}#memoizex-root .toggle-primary:checked,#memoizex-root .toggle-primary[aria-checked=true]{border-color:var(--fallback-p,oklch(var(--p)/var(--tw-border-opacity)));--tw-border-opacity: .1;--tw-bg-opacity: 1;background-color:var(--fallback-p,oklch(var(--p)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-pc,oklch(var(--pc)/var(--tw-text-opacity)))}#memoizex-root .toggle-secondary:focus-visible{outline-color:var(--fallback-s,oklch(var(--s)/1))}#memoizex-root .toggle-secondary:checked,#memoizex-root .toggle-secondary[aria-checked=true]{border-color:var(--fallback-s,oklch(var(--s)/var(--tw-border-opacity)));--tw-border-opacity: .1;--tw-bg-opacity: 1;background-color:var(--fallback-s,oklch(var(--s)/var(--tw-bg-opacity)));--tw-text-opacity: 1;color:var(--fallback-sc,oklch(var(--sc)/var(--tw-text-opacity)))}#memoizex-root .toggle:disabled{cursor:not-allowed;--tw-border-opacity: 1;border-color:var(--fallback-bc,oklch(var(--bc)/var(--tw-border-opacity)));background-color:transparent;opacity:.3;--togglehandleborder: 0 0 0 3px var(--fallback-bc,oklch(var(--bc)/1)) inset, var(--handleoffsetcalculator) 0 0 3px var(--fallback-bc,oklch(var(--bc)/1)) inset}#memoizex-root .badge-xs{height:12px;font-size:12px;line-height:12px;padding-left:5.008px;padding-right:5.008px}#memoizex-root .btm-nav-xs>*:where(.active){border-top-width:1px}#memoizex-root .btm-nav-sm>*:where(.active){border-top-width:2px}#memoizex-root .btm-nav-md>*:where(.active){border-top-width:2px}#memoizex-root .btm-nav-lg>*:where(.active){border-top-width:4px}#memoizex-root .btn-xs{height:24px;min-height:24px;padding-left:8px;padding-right:8px;font-size:12px}#memoizex-root .btn-sm{height:32px;min-height:32px;padding-left:12px;padding-right:12px;font-size:14px}#memoizex-root .btn-square:where(.btn-xs){height:24px;width:24px;padding:0}#memoizex-root .btn-square:where(.btn-sm){height:32px;width:32px;padding:0}#memoizex-root .btn-circle:where(.btn-xs){height:24px;width:24px;border-radius:9999px;padding:0}#memoizex-root .btn-circle:where(.btn-sm){height:32px;width:32px;border-radius:9999px;padding:0}#memoizex-root [type=checkbox].checkbox-xs{height:16px;width:16px}#memoizex-root [type=checkbox].checkbox-sm{height:20px;width:20px}#memoizex-root .input-xs{height:24px;padding-left:8px;padding-right:8px;font-size:12px;line-height:16px;line-height:1.625}#memoizex-root .input-sm{height:32px;padding-left:12px;padding-right:12px;font-size:14px;line-height:32px}#memoizex-root .join.join-vertical{flex-direction:column}#memoizex-root .join.join-vertical .join-item:first-child:not(:last-child),#memoizex-root .join.join-vertical *:first-child:not(:last-child) .join-item{border-end-start-radius:0;border-end-end-radius:0;border-start-start-radius:inherit;border-start-end-radius:inherit}#memoizex-root .join.join-vertical .join-item:last-child:not(:first-child),#memoizex-root .join.join-vertical *:last-child:not(:first-child) .join-item{border-start-start-radius:0;border-start-end-radius:0;border-end-start-radius:inherit;border-end-end-radius:inherit}#memoizex-root .join.join-horizontal{flex-direction:row}#memoizex-root .join.join-horizontal .join-item:first-child:not(:last-child),#memoizex-root .join.join-horizontal *:first-child:not(:last-child) .join-item{border-end-end-radius:0;border-start-end-radius:0;border-end-start-radius:inherit;border-start-start-radius:inherit}#memoizex-root .join.join-horizontal .join-item:last-child:not(:first-child),#memoizex-root .join.join-horizontal *:last-child:not(:first-child) .join-item{border-end-start-radius:0;border-start-start-radius:0;border-end-end-radius:inherit;border-start-end-radius:inherit}#memoizex-root .select-sm{height:32px;min-height:32px;padding-left:12px;padding-right:32px;font-size:14px;line-height:32px}#memoizex-root [dir=rtl] .select-sm{padding-left:32px;padding-right:12px}#memoizex-root .select-xs{height:24px;min-height:24px;padding-left:8px;padding-right:32px;font-size:12px;line-height:16px;line-height:1.625}#memoizex-root [dir=rtl] .select-xs{padding-left:32px;padding-right:8px}#memoizex-root .textarea-xs{padding:4px 8px;font-size:12px;line-height:16px;line-height:1.625}#memoizex-root .tooltip{position:relative;display:inline-block;--tooltip-offset: calc(100% + 1px + var(--tooltip-tail, 0px))}#memoizex-root .tooltip:before{position:absolute;pointer-events:none;z-index:1;content:var(--tw-content);--tw-content: attr(data-tip)}#memoizex-root .tooltip:before,#memoizex-root .tooltip-top:before{transform:translate(-50%);top:auto;left:50%;right:auto;bottom:var(--tooltip-offset)}#memoizex-root .tooltip-bottom:before{transform:translate(-50%);top:var(--tooltip-offset);left:50%;right:auto;bottom:auto}#memoizex-root .avatar.online:before{content:"";position:absolute;z-index:10;display:block;border-radius:9999px;--tw-bg-opacity: 1;background-color:var(--fallback-su,oklch(var(--su)/var(--tw-bg-opacity)));outline-style:solid;outline-width:2px;outline-color:var(--fallback-b1,oklch(var(--b1)/1));width:15%;height:15%;top:7%;right:7%}#memoizex-root .avatar.offline:before{content:"";position:absolute;z-index:10;display:block;border-radius:9999px;--tw-bg-opacity: 1;background-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-bg-opacity)));outline-style:solid;outline-width:2px;outline-color:var(--fallback-b1,oklch(var(--b1)/1));width:15%;height:15%;top:7%;right:7%}#memoizex-root .card-compact .card-body{padding:16px;font-size:14px;line-height:20px}#memoizex-root .card-compact .card-title{margin-bottom:4px}#memoizex-root .join.join-vertical>:where(*:not(:first-child)){margin-left:0;margin-right:0;margin-top:-1px}#memoizex-root .join.join-vertical>:where(*:not(:first-child)):is(.btn){margin-top:calc(var(--border-btn) * -1)}#memoizex-root .join.join-horizontal>:where(*:not(:first-child)){margin-top:0;margin-bottom:0;margin-inline-start:-1px}#memoizex-root .join.join-horizontal>:where(*:not(:first-child)):is(.btn){margin-inline-start:calc(var(--border-btn) * -1);margin-top:0}#memoizex-root .menu-sm :where(li:not(.menu-title)>*:not(ul,details,.menu-title)),#memoizex-root .menu-sm :where(li:not(.menu-title)>details>summary:not(.menu-title)){border-radius:var(--rounded-btn, 8px);padding:4px 12px;font-size:14px;line-height:20px}#memoizex-root .menu-sm .menu-title{padding:8px 12px}#memoizex-root .modal-top :where(.modal-box){width:100%;max-width:none;--tw-translate-y: -40px;--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));border-bottom-right-radius:var(--rounded-box, 16px);border-bottom-left-radius:var(--rounded-box, 16px);border-top-left-radius:0;border-top-right-radius:0}#memoizex-root .modal-middle :where(.modal-box){width:91.666667%;max-width:512px;--tw-translate-y: 0px;--tw-scale-x: .9;--tw-scale-y: .9;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));border-top-left-radius:var(--rounded-box, 16px);border-top-right-radius:var(--rounded-box, 16px);border-bottom-right-radius:var(--rounded-box, 16px);border-bottom-left-radius:var(--rounded-box, 16px)}#memoizex-root .modal-bottom :where(.modal-box){width:100%;max-width:none;--tw-translate-y: 40px;--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));border-top-left-radius:var(--rounded-box, 16px);border-top-right-radius:var(--rounded-box, 16px);border-bottom-right-radius:0;border-bottom-left-radius:0}#memoizex-root .table-xs :not(thead):not(tfoot) tr{font-size:12px;line-height:16px}#memoizex-root .table-xs :where(th,td){padding:4px 8px}#memoizex-root .tooltip{position:relative;display:inline-block;text-align:center;--tooltip-tail: 3px;--tooltip-color: var(--fallback-n,oklch(var(--n)/1));--tooltip-text-color: var(--fallback-nc,oklch(var(--nc)/1));--tooltip-tail-offset: calc(100% + 1px - var(--tooltip-tail))}#memoizex-root .tooltip:before,#memoizex-root .tooltip:after{opacity:0;transition-property:color,background-color,border-color,text-decoration-color,fill,stroke,opacity,box-shadow,transform,filter,backdrop-filter;transition-delay:.1s;transition-duration:.2s;transition-timing-function:cubic-bezier(.4,0,.2,1)}#memoizex-root .tooltip:after{position:absolute;content:"";border-style:solid;border-width:var(--tooltip-tail, 0);width:0;height:0;display:block}#memoizex-root .tooltip:before{max-width:320px;white-space:normal;border-radius:4px;padding:4px 8px;font-size:14px;line-height:20px;background-color:var(--tooltip-color);color:var(--tooltip-text-color);width:-moz-max-content;width:max-content}#memoizex-root .tooltip.tooltip-open:before{opacity:1;transition-delay:75ms}#memoizex-root .tooltip.tooltip-open:after{opacity:1;transition-delay:75ms}#memoizex-root .tooltip:hover:before{opacity:1;transition-delay:75ms}#memoizex-root .tooltip:hover:after{opacity:1;transition-delay:75ms}#memoizex-root .tooltip:has(:focus-visible):after,#memoizex-root .tooltip:has(:focus-visible):before{opacity:1;transition-delay:75ms}#memoizex-root .tooltip:not([data-tip]):hover:before,#memoizex-root .tooltip:not([data-tip]):hover:after{visibility:hidden;opacity:0}#memoizex-root .tooltip:after,#memoizex-root .tooltip-top:after{transform:translate(-50%);border-color:var(--tooltip-color) transparent transparent transparent;top:auto;left:50%;right:auto;bottom:var(--tooltip-tail-offset)}#memoizex-root .tooltip-bottom:after{transform:translate(-50%);border-color:transparent transparent var(--tooltip-color) transparent;top:var(--tooltip-tail-offset);left:50%;right:auto;bottom:auto}#memoizex-root .static{position:static}#memoizex-root .fixed{position:fixed}#memoizex-root .absolute{position:absolute}#memoizex-root .relative{position:relative}#memoizex-root .bottom-0\\.5{bottom:2px}#memoizex-root .left-0\\.5{left:2px}#memoizex-root .left-8{left:32px}#memoizex-root .left-\\[-20px\\]{left:-20px}#memoizex-root .right-3{right:12px}#memoizex-root .top-3{top:12px}#memoizex-root .top-8{top:32px}#memoizex-root .top-\\[60\\%\\]{top:60%}#memoizex-root .z-10{z-index:10}#memoizex-root .col-span-1{grid-column:span 1 / span 1}#memoizex-root .col-span-3{grid-column:span 3 / span 3}#memoizex-root .m-0{margin:0}#memoizex-root .my-3{margin-top:12px;margin-bottom:12px}#memoizex-root .my-\\[2px\\]{margin-top:2px;margin-bottom:2px}#memoizex-root .mb-0{margin-bottom:0}#memoizex-root .mb-1{margin-bottom:4px}#memoizex-root .mb-2{margin-bottom:8px}#memoizex-root .ml-0\\.5{margin-left:2px}#memoizex-root .ml-1{margin-left:4px}#memoizex-root .ml-4{margin-left:16px}#memoizex-root .mr-2{margin-right:8px}#memoizex-root .mr-3{margin-right:12px}#memoizex-root .mt-0{margin-top:0}#memoizex-root .mt-2{margin-top:8px}#memoizex-root .mt-3{margin-top:12px}#memoizex-root .mt-6{margin-top:24px}#memoizex-root .line-clamp-3{overflow:hidden;display:-webkit-box;-webkit-box-orient:vertical;-webkit-line-clamp:3}#memoizex-root .block{display:block}#memoizex-root .inline-block{display:inline-block}#memoizex-root .inline{display:inline}#memoizex-root .flex{display:flex}#memoizex-root .inline-flex{display:inline-flex}#memoizex-root .\\!table{display:table!important}#memoizex-root .table{display:table}#memoizex-root .contents{display:contents}#memoizex-root .h-10{height:40px}#memoizex-root .h-12{height:48px}#memoizex-root .h-28{height:112px}#memoizex-root .h-4{height:16px}#memoizex-root .h-6{height:24px}#memoizex-root .h-8{height:32px}#memoizex-root .h-9{height:36px}#memoizex-root .h-\\[320px\\]{height:320px}#memoizex-root .h-full{height:100%}#memoizex-root .max-h-44{max-height:176px}#memoizex-root .max-h-48{max-height:192px}#memoizex-root .max-h-\\[400px\\]{max-height:400px}#memoizex-root .max-h-\\[500px\\]{max-height:500px}#memoizex-root .max-h-full{max-height:100%}#memoizex-root .min-h-\\[512px\\]{min-height:512px}#memoizex-root .w-12{width:48px}#memoizex-root .w-16{width:64px}#memoizex-root .w-20{width:80px}#memoizex-root .w-24{width:96px}#memoizex-root .w-32{width:128px}#memoizex-root .w-36{width:144px}#memoizex-root .w-4{width:16px}#memoizex-root .w-40{width:160px}#memoizex-root .w-48{width:192px}#memoizex-root .w-52{width:208px}#memoizex-root .w-60{width:240px}#memoizex-root .w-80{width:320px}#memoizex-root .w-9{width:36px}#memoizex-root .w-auto{width:auto}#memoizex-root .w-full{width:100%}#memoizex-root .w-max{width:-moz-max-content;width:max-content}#memoizex-root .max-w-4xl{max-width:896px}#memoizex-root .max-w-\\[200px\\]{max-width:200px}#memoizex-root .max-w-\\[50\\%\\]{max-width:50%}#memoizex-root .max-w-\\[calc\\(100\\%-46px\\)\\]{max-width:calc(100% - 46px)}#memoizex-root .max-w-full{max-width:100%}#memoizex-root .max-w-lg{max-width:512px}#memoizex-root .max-w-sm{max-width:384px}#memoizex-root .max-w-xl{max-width:576px}#memoizex-root .flex-1{flex:1 1 0%}#memoizex-root .flex-shrink-0,#memoizex-root .shrink-0{flex-shrink:0}#memoizex-root .flex-grow,#memoizex-root .grow{flex-grow:1}#memoizex-root .origin-\\[bottom_center\\]{transform-origin:bottom center}#memoizex-root .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))}#memoizex-root .translate-x-\\[-500px\\]{--tw-translate-x: -500px;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))}#memoizex-root .transform-none{transform:none}@keyframes ping{75%,to{transform:scale(2);opacity:0}}#memoizex-root .animate-ping{animation:ping 1s cubic-bezier(0,0,.2,1) infinite}#memoizex-root .cursor-pointer{cursor:pointer}#memoizex-root .select-none{-webkit-user-select:none;-moz-user-select:none;user-select:none}#memoizex-root .resize{resize:both}#memoizex-root .grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}#memoizex-root .grid-cols-\\[auto_minmax\\(auto\\,1fr\\)\\]{grid-template-columns:auto minmax(auto,1fr)}#memoizex-root .flex-row{flex-direction:row}#memoizex-root .flex-col{flex-direction:column}#memoizex-root .flex-wrap{flex-wrap:wrap}#memoizex-root .items-start{align-items:flex-start}#memoizex-root .items-center{align-items:center}#memoizex-root .justify-start{justify-content:flex-start}#memoizex-root .justify-end{justify-content:flex-end}#memoizex-root .justify-center{justify-content:center}#memoizex-root .justify-between{justify-content:space-between}#memoizex-root .gap-1{gap:4px}#memoizex-root .gap-2{gap:8px}#memoizex-root .space-x-1>:not([hidden])~:not([hidden]){--tw-space-x-reverse: 0;margin-right:calc(4px * var(--tw-space-x-reverse));margin-left:calc(4px * calc(1 - var(--tw-space-x-reverse)))}#memoizex-root .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)))}#memoizex-root .overflow-hidden{overflow:hidden}#memoizex-root .overflow-scroll{overflow:scroll}#memoizex-root .overflow-x-scroll{overflow-x:scroll}#memoizex-root .overflow-y-scroll{overflow-y:scroll}#memoizex-root .overscroll-none{overscroll-behavior:none}#memoizex-root .truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}#memoizex-root .whitespace-nowrap{white-space:nowrap}#memoizex-root .whitespace-pre{white-space:pre}#memoizex-root .whitespace-pre-wrap{white-space:pre-wrap}#memoizex-root .break-words{overflow-wrap:break-word}#memoizex-root .break-all{word-break:break-all}#memoizex-root .rounded{border-radius:4px}#memoizex-root .rounded-box{border-radius:var(--rounded-box, 16px)}#memoizex-root .rounded-full{border-radius:9999px}#memoizex-root .rounded-md{border-radius:6px}#memoizex-root .border{border-width:1px}#memoizex-root .border-b{border-bottom-width:1px}#memoizex-root .border-solid{border-style:solid}#memoizex-root .border-base-300{--tw-border-opacity: 1;border-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-border-opacity, 1)))}#memoizex-root .border-neutral-content{--tw-border-opacity: 1;border-color:var(--fallback-nc,oklch(var(--nc)/var(--tw-border-opacity, 1)))}#memoizex-root .border-opacity-50{--tw-border-opacity: .5}#memoizex-root .bg-accent{--tw-bg-opacity: 1;background-color:var(--fallback-a,oklch(var(--a)/var(--tw-bg-opacity, 1)))}#memoizex-root .bg-base-100{--tw-bg-opacity: 1;background-color:var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity, 1)))}#memoizex-root .bg-base-200{--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity, 1)))}#memoizex-root .bg-base-300{--tw-bg-opacity: 1;background-color:var(--fallback-b3,oklch(var(--b3)/var(--tw-bg-opacity, 1)))}#memoizex-root .bg-black{--tw-bg-opacity: 1;background-color:rgb(0 0 0 / var(--tw-bg-opacity, 1))}#memoizex-root .bg-primary{--tw-bg-opacity: 1;background-color:var(--fallback-p,oklch(var(--p)/var(--tw-bg-opacity, 1)))}#memoizex-root .bg-secondary{--tw-bg-opacity: 1;background-color:var(--fallback-s,oklch(var(--s)/var(--tw-bg-opacity, 1)))}#memoizex-root .bg-transparent{background-color:transparent}#memoizex-root .bg-opacity-30{--tw-bg-opacity: .3}#memoizex-root .fill-base-content{fill:var(--fallback-bc,oklch(var(--bc)/1))}#memoizex-root .object-contain{-o-object-fit:contain;object-fit:contain}#memoizex-root .object-cover{-o-object-fit:cover;object-fit:cover}#memoizex-root .p-0{padding:0}#memoizex-root .p-2{padding:8px}#memoizex-root .p-3{padding:12px}#memoizex-root .px-0\\.5{padding-left:2px;padding-right:2px}#memoizex-root .px-1{padding-left:4px;padding-right:4px}#memoizex-root .px-2{padding-left:8px;padding-right:8px}#memoizex-root .px-4{padding-left:16px;padding-right:16px}#memoizex-root .py-1{padding-top:4px;padding-bottom:4px}#memoizex-root .py-2{padding-top:8px;padding-bottom:8px}#memoizex-root .py-2\\.5{padding-top:10px;padding-bottom:10px}#memoizex-root .py-3{padding-top:12px;padding-bottom:12px}#memoizex-root .pl-2{padding-left:8px}#memoizex-root .pr-2{padding-right:8px}#memoizex-root .text-center{text-align:center}#memoizex-root .align-top{vertical-align:top}#memoizex-root .align-middle{vertical-align:middle}#memoizex-root .font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}#memoizex-root .text-base{font-size:16px;line-height:24px}#memoizex-root .text-sm{font-size:14px;line-height:20px}#memoizex-root .text-xl{font-size:20px;line-height:28px}#memoizex-root .text-xs{font-size:12px;line-height:16px}#memoizex-root .font-bold{font-weight:700}#memoizex-root .font-medium{font-weight:500}#memoizex-root .font-semibold{font-weight:600}#memoizex-root .leading-4{line-height:16px}#memoizex-root .leading-5{line-height:20px}#memoizex-root .leading-6{line-height:24px}#memoizex-root .leading-8{line-height:32px}#memoizex-root .leading-\\[48px\\]{line-height:48px}#memoizex-root .leading-loose{line-height:2}#memoizex-root .leading-none{line-height:1}#memoizex-root .leading-normal{line-height:1.5}#memoizex-root .text-base-content{--tw-text-opacity: 1;color:var(--fallback-bc,oklch(var(--bc)/var(--tw-text-opacity, 1)))}#memoizex-root .text-base-content\\/60{color:var(--fallback-bc,oklch(var(--bc)/.6))}#memoizex-root .text-error{--tw-text-opacity: 1;color:var(--fallback-er,oklch(var(--er)/var(--tw-text-opacity, 1)))}#memoizex-root .text-success{--tw-text-opacity: 1;color:var(--fallback-su,oklch(var(--su)/var(--tw-text-opacity, 1)))}#memoizex-root .text-warning{--tw-text-opacity: 1;color:var(--fallback-wa,oklch(var(--wa)/var(--tw-text-opacity, 1)))}#memoizex-root .text-white{--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}#memoizex-root .text-opacity-50{--tw-text-opacity: .5}#memoizex-root .text-opacity-60{--tw-text-opacity: .6}#memoizex-root .text-opacity-70{--tw-text-opacity: .7}#memoizex-root .opacity-50{opacity:.5}#memoizex-root .opacity-75{opacity:.75}#memoizex-root .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)}#memoizex-root .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)}#memoizex-root .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)}#memoizex-root .transition-all{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}#memoizex-root .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}#memoizex-root .transition-transform{transition-property:transform;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}#memoizex-root .duration-200{transition-duration:.2s}#memoizex-root .duration-500{transition-duration:.5s}#memoizex-root .rounded-box-half{border-radius:calc(var(--rounded-box) / 2)}#memoizex-root .table-border-bc :where(thead,tbody) :where(tr:not(:last-child)),#memoizex-root .table-border-bc :where(thead,tbody) :where(tr:first-child:last-child){--tw-border-opacity: 20%;border-bottom-width:1px;border-bottom-color:var(--fallback-bc, oklch(var(--bc) / var(--tw-border-opacity)))}#memoizex-root .table-padding-sm :where(th,td){padding:8px 12px}#memoizex-root .no-scrollbar::-webkit-scrollbar{display:none}#memoizex-root .no-scrollbar{-ms-overflow-style:none;scrollbar-width:none}#memoizex-root .before\\:max-w-40:before{content:var(--tw-content);max-width:160px}#memoizex-root .before\\:max-w-max:before{content:var(--tw-content);max-width:-moz-max-content;max-width:max-content}#memoizex-root .before\\:whitespace-pre-line:before{content:var(--tw-content);white-space:pre-line}#memoizex-root .last\\:border-0:last-child{border-width:0px}#memoizex-root .hover\\:bg-base-200:hover{--tw-bg-opacity: 1;background-color:var(--fallback-b2,oklch(var(--b2)/var(--tw-bg-opacity, 1)))}#memoizex-root .group:hover .group-hover\\:translate-x-\\[5px\\]{--tw-translate-x: 5px;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))}#memoizex-root .group:hover .group-hover\\:rotate-\\[20deg\\]{--tw-rotate: 20deg;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))}#memoizex-root .group:hover .group-hover\\:opacity-90{opacity:.9}@media(min-width:640px){#memoizex-root .sm\\:grid{display:grid}#memoizex-root .sm\\:h-9{height:36px}#memoizex-root .sm\\:max-w-screen-sm{max-width:640px}#memoizex-root .sm\\:gap-2{gap:8px}#memoizex-root .sm\\:pl-2{padding-left:8px}}@media(min-width:768px){#memoizex-root .md\\:max-w-screen-md{max-width:768px}#memoizex-root .md\\:max-w-screen-sm{max-width:640px}}#memoizex-root .\\[\\&\\>path\\]\\:stroke-0>path{stroke-width:0} `);

(function (preact, hooks, signals, i18next, dayjs, Dexie, dexieExportImport, fileSaverEs, tableCore) {
  'use strict';

  var __defProp = Object.defineProperty;
  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 f = 0;
  function u(e, t, n, o, i, u2) {
    t || (t = {});
    var a, c, p = t;
    if ("ref" in p) for (c in p = {}, t) "ref" == c ? a = t[c] : p[c] = t[c];
    var l = { type: e, props: p, key: n, ref: a, __k: null, __: null, __b: 0, __e: null, __c: null, constructor: void 0, __v: --f, __i: -1, __u: 0, __source: i, __self: u2 };
    if ("function" == typeof e && (a = e.defaultProps)) for (c in a) void 0 === p[c] && (p[c] = a[c]);
    return preact.options.vnode && preact.options.vnode(l), l;
  }
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var defaultAttributes = {
    outline: {
      xmlns: "http://www.w3.org/2000/svg",
      width: 24,
      height: 24,
      viewBox: "0 0 24 24",
      fill: "none",
      stroke: "currentColor",
      "stroke-width": 2,
      "stroke-linecap": "round",
      "stroke-linejoin": "round"
    },
    filled: {
      xmlns: "http://www.w3.org/2000/svg",
      width: 24,
      height: 24,
      viewBox: "0 0 24 24",
      fill: "currentColor",
      stroke: "none"
    }
  };
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  const createPreactComponent = (type, iconName, iconNamePascal, iconNode) => {
    const Component2 = ({
      color = "currentColor",
      size = 24,
      stroke = 2,
      title,
      children,
      className = "",
      class: classes = "",
      style,
      ...rest
    }) => preact.h(
      "svg",
      {
        ...defaultAttributes[type],
        width: String(size),
        height: String(size),
        class: [`tabler-icon`, `tabler-icon-${iconName}`, classes, className].join(" "),
        ...type === "filled" ? {
          fill: color
        } : {
          "stroke-width": stroke,
          stroke: color
        },
        style,
        ...rest
      },
      [
        title && preact.h("title", {}, title),
        ...iconNode.map(([tag, attrs]) => preact.h(tag, attrs)),
        ...preact.toChildArray(children)
      ]
    );
    Component2.displayName = `${iconNamePascal}`;
    return Component2;
  };
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconArrowUpRight = createPreactComponent("outline", "arrow-up-right", "IconArrowUpRight", [["path", { "d": "M17 7l-10 10", "key": "svg-0" }], ["path", { "d": "M8 7l9 0l0 9", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconCheck = createPreactComponent("outline", "check", "IconCheck", [["path", { "d": "M5 12l5 5l10 -10", "key": "svg-0" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconChevronLeftPipe = createPreactComponent("outline", "chevron-left-pipe", "IconChevronLeftPipe", [["path", { "d": "M7 6v12", "key": "svg-0" }], ["path", { "d": "M18 6l-6 6l6 6", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconChevronLeft = createPreactComponent("outline", "chevron-left", "IconChevronLeft", [["path", { "d": "M15 6l-6 6l6 6", "key": "svg-0" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconChevronRightPipe = createPreactComponent("outline", "chevron-right-pipe", "IconChevronRightPipe", [["path", { "d": "M6 6l6 6l-6 6", "key": "svg-0" }], ["path", { "d": "M17 5v13", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconChevronRight = createPreactComponent("outline", "chevron-right", "IconChevronRight", [["path", { "d": "M9 6l6 6l-6 6", "key": "svg-0" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconCircleCheck = createPreactComponent("outline", "circle-check", "IconCircleCheck", [["path", { "d": "M12 12m-9 0a9 9 0 1 0 18 0a9 9 0 1 0 -18 0", "key": "svg-0" }], ["path", { "d": "M9 12l2 2l4 -4", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconCircleDashed = createPreactComponent("outline", "circle-dashed", "IconCircleDashed", [["path", { "d": "M8.56 3.69a9 9 0 0 0 -2.92 1.95", "key": "svg-0" }], ["path", { "d": "M3.69 8.56a9 9 0 0 0 -.69 3.44", "key": "svg-1" }], ["path", { "d": "M3.69 15.44a9 9 0 0 0 1.95 2.92", "key": "svg-2" }], ["path", { "d": "M8.56 20.31a9 9 0 0 0 3.44 .69", "key": "svg-3" }], ["path", { "d": "M15.44 20.31a9 9 0 0 0 2.92 -1.95", "key": "svg-4" }], ["path", { "d": "M20.31 15.44a9 9 0 0 0 .69 -3.44", "key": "svg-5" }], ["path", { "d": "M20.31 8.56a9 9 0 0 0 -1.95 -2.92", "key": "svg-6" }], ["path", { "d": "M15.44 3.69a9 9 0 0 0 -3.44 -.69", "key": "svg-7" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconDatabaseExport = createPreactComponent("outline", "database-export", "IconDatabaseExport", [["path", { "d": "M4 6c0 1.657 3.582 3 8 3s8 -1.343 8 -3s-3.582 -3 -8 -3s-8 1.343 -8 3", "key": "svg-0" }], ["path", { "d": "M4 6v6c0 1.657 3.582 3 8 3c1.118 0 2.183 -.086 3.15 -.241", "key": "svg-1" }], ["path", { "d": "M20 12v-6", "key": "svg-2" }], ["path", { "d": "M4 12v6c0 1.657 3.582 3 8 3c.157 0 .312 -.002 .466 -.005", "key": "svg-3" }], ["path", { "d": "M16 19h6", "key": "svg-4" }], ["path", { "d": "M19 16l3 3l-3 3", "key": "svg-5" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconEdit = createPreactComponent("outline", "edit", "IconEdit", [["path", { "d": "M7 7h-1a2 2 0 0 0 -2 2v9a2 2 0 0 0 2 2h9a2 2 0 0 0 2 -2v-1", "key": "svg-0" }], ["path", { "d": "M20.385 6.585a2.1 2.1 0 0 0 -2.97 -2.97l-8.415 8.385v3h3l8.385 -8.415z", "key": "svg-1" }], ["path", { "d": "M16 5l3 3", "key": "svg-2" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconExclamationCircle = createPreactComponent("outline", "exclamation-circle", "IconExclamationCircle", [["path", { "d": "M12 12m-9 0a9 9 0 1 0 18 0a9 9 0 1 0 -18 0", "key": "svg-0" }], ["path", { "d": "M12 9v4", "key": "svg-1" }], ["path", { "d": "M12 16v.01", "key": "svg-2" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconFileDownload = createPreactComponent("outline", "file-download", "IconFileDownload", [["path", { "d": "M14 3v4a1 1 0 0 0 1 1h4", "key": "svg-0" }], ["path", { "d": "M17 21h-10a2 2 0 0 1 -2 -2v-14a2 2 0 0 1 2 -2h7l5 5v11a2 2 0 0 1 -2 2z", "key": "svg-1" }], ["path", { "d": "M12 17v-6", "key": "svg-2" }], ["path", { "d": "M9.5 14.5l2.5 2.5l2.5 -2.5", "key": "svg-3" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconHelp = createPreactComponent("outline", "help", "IconHelp", [["path", { "d": "M12 12m-9 0a9 9 0 1 0 18 0a9 9 0 1 0 -18 0", "key": "svg-0" }], ["path", { "d": "M12 17l0 .01", "key": "svg-1" }], ["path", { "d": "M12 13.5a1.5 1.5 0 0 1 1 -1.5a2.6 2.6 0 1 0 -3 -4", "key": "svg-2" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconInfoCircle = createPreactComponent("outline", "info-circle", "IconInfoCircle", [["path", { "d": "M3 12a9 9 0 1 0 18 0a9 9 0 0 0 -18 0", "key": "svg-0" }], ["path", { "d": "M12 9h.01", "key": "svg-1" }], ["path", { "d": "M11 12h1v4h1", "key": "svg-2" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconLink = createPreactComponent("outline", "link", "IconLink", [["path", { "d": "M9 15l6 -6", "key": "svg-0" }], ["path", { "d": "M11 6l.463 -.536a5 5 0 0 1 7.071 7.072l-.534 .464", "key": "svg-1" }], ["path", { "d": "M13 18l-.397 .534a5.068 5.068 0 0 1 -7.127 0a4.972 4.972 0 0 1 0 -7.071l.524 -.463", "key": "svg-2" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconPlayerPlay = createPreactComponent("outline", "player-play", "IconPlayerPlay", [["path", { "d": "M7 4v16l13 -8z", "key": "svg-0" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconPlus = createPreactComponent("outline", "plus", "IconPlus", [["path", { "d": "M12 5l0 14", "key": "svg-0" }], ["path", { "d": "M5 12l14 0", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconReportAnalytics = createPreactComponent("outline", "report-analytics", "IconReportAnalytics", [["path", { "d": "M9 5h-2a2 2 0 0 0 -2 2v12a2 2 0 0 0 2 2h10a2 2 0 0 0 2 -2v-12a2 2 0 0 0 -2 -2h-2", "key": "svg-0" }], ["path", { "d": "M9 3m0 2a2 2 0 0 1 2 -2h2a2 2 0 0 1 2 2v0a2 2 0 0 1 -2 2h-2a2 2 0 0 1 -2 -2z", "key": "svg-1" }], ["path", { "d": "M9 17v-5", "key": "svg-2" }], ["path", { "d": "M12 17v-1", "key": "svg-3" }], ["path", { "d": "M15 17v-3", "key": "svg-4" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconSearch = createPreactComponent("outline", "search", "IconSearch", [["path", { "d": "M10 10m-7 0a7 7 0 1 0 14 0a7 7 0 1 0 -14 0", "key": "svg-0" }], ["path", { "d": "M21 21l-6 -6", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconSettings = createPreactComponent("outline", "settings", "IconSettings", [["path", { "d": "M10.325 4.317c.426 -1.756 2.924 -1.756 3.35 0a1.724 1.724 0 0 0 2.573 1.066c1.543 -.94 3.31 .826 2.37 2.37a1.724 1.724 0 0 0 1.065 2.572c1.756 .426 1.756 2.924 0 3.35a1.724 1.724 0 0 0 -1.066 2.573c.94 1.543 -.826 3.31 -2.37 2.37a1.724 1.724 0 0 0 -2.572 1.065c-.426 1.756 -2.924 1.756 -3.35 0a1.724 1.724 0 0 0 -2.573 -1.066c-1.543 .94 -3.31 -.826 -2.37 -2.37a1.724 1.724 0 0 0 -1.065 -2.572c-1.756 -.426 -1.756 -2.924 0 -3.35a1.724 1.724 0 0 0 1.066 -2.573c-.94 -1.543 .826 -3.31 2.37 -2.37c1 .608 2.296 .07 2.572 -1.065z", "key": "svg-0" }], ["path", { "d": "M9 12a3 3 0 1 0 6 0a3 3 0 0 0 -6 0", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconSortAscending = createPreactComponent("outline", "sort-ascending", "IconSortAscending", [["path", { "d": "M4 6l7 0", "key": "svg-0" }], ["path", { "d": "M4 12l7 0", "key": "svg-1" }], ["path", { "d": "M4 18l9 0", "key": "svg-2" }], ["path", { "d": "M15 9l3 -3l3 3", "key": "svg-3" }], ["path", { "d": "M18 6l0 12", "key": "svg-4" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconSortDescending = createPreactComponent("outline", "sort-descending", "IconSortDescending", [["path", { "d": "M4 6l9 0", "key": "svg-0" }], ["path", { "d": "M4 12l7 0", "key": "svg-1" }], ["path", { "d": "M4 18l7 0", "key": "svg-2" }], ["path", { "d": "M15 15l3 3l3 -3", "key": "svg-3" }], ["path", { "d": "M18 6l0 12", "key": "svg-4" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconTrashX = createPreactComponent("outline", "trash-x", "IconTrashX", [["path", { "d": "M4 7h16", "key": "svg-0" }], ["path", { "d": "M5 7l1 12a2 2 0 0 0 2 2h8a2 2 0 0 0 2 -2l1 -12", "key": "svg-1" }], ["path", { "d": "M9 7v-3a1 1 0 0 1 1 -1h4a1 1 0 0 1 1 1v3", "key": "svg-2" }], ["path", { "d": "M10 12l4 4m0 -4l-4 4", "key": "svg-3" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconWebhook = createPreactComponent("outline", "webhook", "IconWebhook", [["path", { "d": "M4.876 13.61a4 4 0 1 0 6.124 3.39h6", "key": "svg-0" }], ["path", { "d": "M15.066 20.502a4 4 0 1 0 1.934 -7.502c-.706 0 -1.424 .179 -2 .5l-3 -5.5", "key": "svg-1" }], ["path", { "d": "M16 8a4 4 0 1 0 -8 0c0 1.506 .77 2.818 2 3.5l-3 5.5", "key": "svg-2" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconX = createPreactComponent("outline", "x", "IconX", [["path", { "d": "M18 6l-12 12", "key": "svg-0" }], ["path", { "d": "M6 6l12 12", "key": "svg-1" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconBrandGithubFilled = createPreactComponent("filled", "brand-github-filled", "IconBrandGithubFilled", [["path", { "d": "M5.315 2.1c.791 -.113 1.9 .145 3.333 .966l.272 .161l.16 .1l.397 -.083a13.3 13.3 0 0 1 4.59 -.08l.456 .08l.396 .083l.161 -.1c1.385 -.84 2.487 -1.17 3.322 -1.148l.164 .008l.147 .017l.076 .014l.05 .011l.144 .047a1 1 0 0 1 .53 .514a5.2 5.2 0 0 1 .397 2.91l-.047 .267l-.046 .196l.123 .163c.574 .795 .93 1.728 1.03 2.707l.023 .295l.007 .272c0 3.855 -1.659 5.883 -4.644 6.68l-.245 .061l-.132 .029l.014 .161l.008 .157l.004 .365l-.002 .213l-.003 3.834a1 1 0 0 1 -.883 .993l-.117 .007h-6a1 1 0 0 1 -.993 -.883l-.007 -.117v-.734c-1.818 .26 -3.03 -.424 -4.11 -1.878l-.535 -.766c-.28 -.396 -.455 -.579 -.589 -.644l-.048 -.019a1 1 0 0 1 .564 -1.918c.642 .188 1.074 .568 1.57 1.239l.538 .769c.76 1.079 1.36 1.459 2.609 1.191l.001 -.678l-.018 -.168a5.03 5.03 0 0 1 -.021 -.824l.017 -.185l.019 -.12l-.108 -.024c-2.976 -.71 -4.703 -2.573 -4.875 -6.139l-.01 -.31l-.004 -.292a5.6 5.6 0 0 1 .908 -3.051l.152 -.222l.122 -.163l-.045 -.196a5.2 5.2 0 0 1 .145 -2.642l.1 -.282l.106 -.253a1 1 0 0 1 .529 -.514l.144 -.047l.154 -.03z", "key": "svg-0" }]]);
  /**
   * @license @tabler/icons-preact v3.31.0 - MIT
   *
   * This source code is licensed under the MIT license.
   * See the LICENSE file in the root directory of this source tree.
   */
  var IconBrandTwitterFilled = createPreactComponent("filled", "brand-twitter-filled", "IconBrandTwitterFilled", [["path", { "d": "M14.058 3.41c-1.807 .767 -2.995 2.453 -3.056 4.38l-.002 .182l-.243 -.023c-2.392 -.269 -4.498 -1.512 -5.944 -3.531a1 1 0 0 0 -1.685 .092l-.097 .186l-.049 .099c-.719 1.485 -1.19 3.29 -1.017 5.203l.03 .273c.283 2.263 1.5 4.215 3.779 5.679l.173 .107l-.081 .043c-1.315 .663 -2.518 .952 -3.827 .9c-1.056 -.04 -1.446 1.372 -.518 1.878c3.598 1.961 7.461 2.566 10.792 1.6c4.06 -1.18 7.152 -4.223 8.335 -8.433l.127 -.495c.238 -.993 .372 -2.006 .401 -3.024l.003 -.332l.393 -.779l.44 -.862l.214 -.434l.118 -.247c.265 -.565 .456 -1.033 .574 -1.43l.014 -.056l.008 -.018c.22 -.593 -.166 -1.358 -.941 -1.358l-.122 .007a.997 .997 0 0 0 -.231 .057l-.086 .038a7.46 7.46 0 0 1 -.88 .36l-.356 .115l-.271 .08l-.772 .214c-1.336 -1.118 -3.144 -1.254 -5.012 -.554l-.211 .084z", "key": "svg-0" }]]);
  var _GM = /* @__PURE__ */ (() => typeof GM != "undefined" ? GM : void 0)();
  var _GM_registerMenuCommand = /* @__PURE__ */ (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : void 0)();
  var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)();
  const en = { "exporter": { "ID": "ID", "Date": "Date", "Content": "Content", "Show Full Text": "Show Full Text", "Media": "Media", "Screen Name": "Screen Name", "Profile Name": "Profile Name", "Profile Image": "Profile Image", "User ID": "User ID", "Replying To": "Replying To", "RT Source": "RT Source", "Quote Source": "Quote Source", "Media Tags": "Media Tags", "Favorites": "Favorites", "Retweets": "Retweets", "Bookmarks": "Bookmarks", "Quotes": "Quotes", "Replies": "Replies", "Views": "Views", "Favorited": "Favorited", "Retweeted": "Retweeted", "Bookmarked": "Bookmarked", "URL": "URL", "Actions": "Actions", "Details": "Details", "Description": "Description", "Profile Banner": "Profile Banner", "Followers": "Followers", "FollowingCount": "Following Count", "Statuses": "Statuses", "Favourites": "Favourites", "Listed": "Listed", "Location": "Location", "Website": "Website", "Birthdate": "Birthdate", "Verified Type": "Verified Type", "Blue Verified": "Blue Verified", "Following": "Following", "Follows You": "Follows You", "Can DM": "Can DM", "Protected": "Protected", "Created At": "Created At", "Sender": "Sender", "Recipient": "Recipient", "Conversation ID": "Conversation ID", "Conversation Type": "Conversation Type", "Data": "Data", "Export captured data as JSON/HTML/CSV file. This may take a while depending on the amount of data. The exported file does not include media files such as images and videos but only the URLs.": "Export captured data as JSON/HTML/CSV file. This may take a while depending on the amount of data. The exported file does not include media files such as images and videos but only the URLs.", "Data length:": "Data length:", "Include all metadata:": "Include all metadata:", "Export as:": "Export as:", "No data selected.": "No data selected.", "Cancel": "Cancel", "Start Export": "Start Export", "Download and save media files from captured data. This may take a while depending on the amount of data. Media that will be downloaded includes: profile images, profile banners (for users), images, videos (for tweets).": "Download and save media files from captured data. This may take a while depending on the amount of data. Media that will be downloaded includes: profile images, profile banners (for users), images, videos (for tweets).", "For more than 100 media or large files, it is recommended to copy the URLs and download them with an external download manager such as aria2.": "For more than 100 media or large files, it is recommended to copy the URLs and download them with an external download manager such as aria2.", "Filename template:": "Filename template:", "Use aria2 format:": "Use aria2 format:", "Click for more information. Each URL will be on a new line, with its filename on the next line. This format is compatible with aria2.": "Click for more information. Each URL will be on a new line, with its filename on the next line. This format is compatible with aria2.", "Rate limit (ms):": "Rate limit (ms):", "Media Filter:": "Media Filter:", "File Name": "File Name", "Media Type": "Media Type", "Download URL": "Download URL", "No media selected.": "No media selected.", "Copied!": "Copied!", "Copy URLs": "Copy URLs", "The tweet ID": "The tweet ID", "The username of tweet author": "The username of tweet author", "The profile name of tweet author": "The profile name of tweet author", "The media index in tweet (start from 0)": "The media index in tweet (start from 0)", "The order of media in tweet (1/2/3/4)": "The order of media in tweet (1/2/3/4)", "The post date in YYYYMMDD format": "The post date in YYYYMMDD format", "The post time in HHmmss format": "The post time in HHmmss format", "The media type (photo/video/animated_gif)": "The media type (photo/video/animated_gif)", "The file extension of media (jpg/png/mp4)": "The file extension of media (jpg/png/mp4)", "Failed to export media. Open DevTools for more details.": "Failed to export media. Open DevTools for more details.", "Failed to copy media URLs. Open DevTools for more details.": "Failed to copy media URLs. Open DevTools for more details.", "filter.photo": "Photo", "filter.video": "Video", "filter.animated_gif": "GIF", "filter.retweet": "Include retweets", "Article Title": "Article Title", "Article Preview": "Article Preview", "Article URL": "Article URL", "Article Cover": "Article Cover", "Quoted Article Title": "Quoted Article Title", "Quoted Article Preview": "Quoted Article Preview", "Quoted Article URL": "Quoted Article URL", "Quoted Article Cover": "Quoted Article Cover" }, "common": { "Open Control Panel": "Open Control Panel", "Browse around to capture more data.": "Browse around to capture more data.", "Settings": "Settings", "General": "General", "Theme": "Theme", "Language": "Language", "Debug": "Debug", "Date Time Format": "Date Time Format", "Click for more information. This will take effect on both previewer and exported files.": "Click for more information. This will take effect on both previewer and exported files.", "Use dedicated DB for accounts": "Use dedicated DB for accounts", "This will create separate database for each Twitter account, which can help reduce the chance of data mixing when you use multiple accounts.": "This will create separate database for each Twitter account, which can help reduce the chance of data mixing when you use multiple accounts.", "Filter promoted tweets": "Filter promoted tweets", "Automatically filter out promoted/advertisement tweets from being saved to the database.": "Automatically filter out promoted/advertisement tweets from being saved to the database.", "Filter promotional tweets": "Filter promotional tweets", "Filter tweets from business accounts and amplify videos.": "Filter tweets from business accounts and amplify videos.", "Local Database": "Local Database", "Analyze DB": "Analyze", "Export DB": "Export", "Clear DB": "Clear", "Are you sure to clear all data in the database?": "Are you sure to clear all data in the database?", "Database cleared.": "Database cleared.", "Module": "Module", "Modules (Scroll to see more)": "Modules (Scroll to see more)", "About": "About", "Version": "Version", "Search...": "Search...", "Something went wrong.": "Something went wrong.", "Error:": "Error:", "Captured:": "Captured:", "Rows per page:": "Rows per page:", "A - B of N items": "{{from}} - {{to}} of {{total}} items", "No data available.": "No data available.", "Clear": "Clear", "Export Media": "Export Media", "Export Data": "Export Data", "JSON View": "JSON View", "Media View": "Media View", "Bookmarks": "Bookmarks", "CommunityMembers": "CommunityMembers", "CommunityTimeline": "CommunityTimeline", "DirectMessages": "DirectMessages", "Followers": "Followers", "Following": "Following", "HomeTimeline": "HomeTimeline", "Likes": "Likes", "ListMembers": "ListMembers", "ListSubscribers": "ListSubscribers", "ListTimeline": "ListTimeline", "RetweetersModule": "RetweetersModule", "RuntimeLogs": "RuntimeLogs", "SearchTimeline": "SearchTimeline", "TweetDetail": "TweetDetail", "UserDetail": "UserDetail", "UserMedia": "UserMedia", "UserTweets": "UserTweets", "BrowsingHistory": "Browsing History", "WebhookManager": "Webhook Manager", "Load More": "Load More", "Save": "Save", "Cancel": "Cancel", "Test": "Test", "Add": "Add", "Edit": "Edit", "Delete": "Delete", "Enabled": "Enabled", "Active": "Active", "Inactive": "Inactive", "Name": "Name", "URL": "URL", "Events": "Events", "Headers": "Headers", "Retry on failure": "Retry on failure", "Max retries": "Max retries", "Configs": "Configs", "Logs": "Logs", "No configs": "No webhook configurations yet.", "No logs": "No webhook logs yet.", "No browsing history": "No browsing history yet.", "Confirm delete webhook": "Are you sure you want to delete this webhook?", "Confirm clear logs": "Are you sure you want to clear all webhook logs?", "Confirm clear history": "Are you sure you want to clear all browsing history?", "Article Title": "Article Title", "Article Preview": "Article Preview", "Article URL": "Article URL", "Article Cover": "Article Cover", "Quoted Article Title": "Quoted Article Title", "Quoted Article Preview": "Quoted Article Preview", "Quoted Article URL": "Quoted Article URL", "Quoted Article Cover": "Quoted Article Cover" } };
  const zh_Hans = { "exporter": { "ID": "ID", "Date": "日期", "Content": "内容", "Show Full Text": "显示全文", "Media": "媒体", "Screen Name": "用户名", "Profile Name": "用户昵称", "Profile Image": "用户头像", "User ID": "用户 ID", "Replying To": "回复推文", "RT Source": "转推来源", "Quote Source": "引用来源", "Media Tags": "圈人", "Favorites": "喜欢数量", "Retweets": "转推数量", "Bookmarks": "书签数量", "Quotes": "引用数量", "Replies": "回复数量", "Views": "查看次数", "Favorited": "已喜欢", "Retweeted": "已转推", "Bookmarked": "已加书签", "URL": "URL", "Actions": "操作", "Details": "查看详情", "Description": "简介", "Profile Banner": "个人资料头图", "Followers": "关注者数量", "FollowingCount": "正在关注数量", "Statuses": "推文数量", "Favourites": "喜欢数量", "Listed": "被加入列表数", "Location": "位置", "Website": "网站", "Birthdate": "出生日期", "Verified Type": "认证类型", "Blue Verified": "蓝标认证", "Following": "正在关注", "Follows You": "关注你", "Can DM": "可私信", "Protected": "受保护", "Created At": "创建时间", "Sender": "发送者", "Recipient": "接收者", "Conversation ID": "对话 ID", "Conversation Type": "对话类型", "Data": "数据", "Export captured data as JSON/HTML/CSV file. This may take a while depending on the amount of data. The exported file does not include media files such as images and videos but only the URLs.": "将捕获的数据导出为 JSON/HTML/CSV 文件。这可能需要一些时间,具体取决于数据量。导出的文件不包括图片和视频等媒体文件,只包括它们的 URL。", "Data length:": "数据长度:", "Include all metadata:": "包括所有元数据:", "Export as:": "导出为:", "No data selected.": "未选择数据。", "Cancel": "取消", "Start Export": "开始导出", "Download and save media files from captured data. This may take a while depending on the amount of data. Media that will be downloaded includes: profile images, profile banners (for users), images, videos (for tweets).": "从捕获的数据中下载并保存媒体文件。这可能需要一些时间,具体取决于数据量。将下载的媒体包括:用户的个人资料图片、个人资料头图、图片、推文中的视频。", "For more than 100 media or large files, it is recommended to copy the URLs and download them with an external download manager such as aria2.": "对于超过 100 个媒体或大文件,建议复制 URL 并使用外部下载管理器(如 aria2)下载。", "Filename template:": "文件名模板:", "Use aria2 format:": "使用 aria2 格式:", "Click for more information. Each URL will be on a new line, with its filename on the next line. This format is compatible with aria2.": "点击获取更多信息。每个 URL 将在新行中显示,其文件名在下一行。此格式与 aria2 兼容。", "Rate limit (ms):": "速率限制(毫秒):", "Media Filter:": "媒体过滤器:", "File Name": "文件名", "Media Type": "媒体类型", "Download URL": "下载地址", "No media selected.": "未选择媒体。", "Copied!": "已复制!", "Copy URLs": "复制 URL", "The tweet ID": "推文 ID", "The username of tweet author": "推文作者的用户名", "The profile name of tweet author": "推文作者的用户昵称", "The media index in tweet (start from 0)": "推文中的媒体索引(从 0 开始)", "The order of media in tweet (1/2/3/4)": "推文中的媒体顺序(1/2/3/4)", "The post date in YYYYMMDD format": "发布日期(YYYYMMDD 格式)", "The post time in HHmmss format": "发布时间(HHmmss 格式)", "The media type (photo/video/animated_gif)": "媒体类型(photo/video/animated_gif)", "The file extension of media (jpg/png/mp4)": "媒体文件扩展名(jpg/png/mp4)", "Failed to export media. Open DevTools for more details.": "导出媒体失败。打开 DevTools 以获取更多详细信息。", "Failed to copy media URLs. Open DevTools for more details.": "复制媒体 URL 失败。打开 DevTools 以获取更多详细信息。", "filter.photo": "图片", "filter.video": "视频", "filter.animated_gif": "GIF", "filter.retweet": "包括转推", "Article Title": "文章标题", "Article Preview": "文章预览", "Article URL": "文章链接", "Article Cover": "文章封面", "Quoted Article Title": "引用文章标题", "Quoted Article Preview": "引用文章预览", "Quoted Article URL": "引用文章链接", "Quoted Article Cover": "引用文章封面" }, "common": { "Open Control Panel": "打开控制面板", "Browse around to capture more data.": "浏览页面以捕获更多数据。", "Settings": "设置", "General": "通用", "Theme": "主题", "Language": "语言", "Debug": "调试开关", "Date Time Format": "日期时间格式", "Click for more information. This will take effect on both previewer and exported files.": "点击查看详细信息。此选项会影响预览和导出的文件。", "Use dedicated DB for accounts": "为每个账户使用专用数据库", "This will create separate database for each Twitter account, which can help reduce the chance of data mixing when you use multiple accounts.": "这将为每个 Twitter 账户创建单独的数据库,这有助于避免使用多个账户时数据混淆。", "Filter promoted tweets": "过滤广告推文", "Automatically filter out promoted/advertisement tweets from being saved to the database.": "自动过滤广告推文,不将其保存到数据库中。", "Filter promotional tweets": "过滤推广内容", "Filter tweets from business accounts and amplify videos.": "过滤企业认证账号和付费推广视频的推文。", "Local Database": "本地数据库", "Analyze DB": "统计", "Export DB": "导出", "Clear DB": "清空", "Are you sure to clear all data in the database?": "确定要清空数据库中的所有数据吗?", "Database cleared.": "数据库已清空。", "Module": "模块", "Modules (Scroll to see more)": "模块列表(滑动以查看完整列表)", "About": "关于", "Version": "版本", "Search...": "搜索...", "Something went wrong.": "出错了。", "Error:": "错误:", "Captured:": "已捕获:", "Rows per page:": "每页显示行数:", "A - B of N items": "第 {{from}} - {{to}} 项,共 {{total}} 项", "No data available.": "没有数据。", "Clear": "清除", "Export Media": "导出媒体文件", "Export Data": "导出数据", "JSON View": "JSON 数据预览", "Media View": "媒体预览", "Bookmarks": "书签", "CommunityMembers": "社群成员", "CommunityTimeline": "社群时间线", "DirectMessages": "私信", "Followers": "关注者", "Following": "正在关注", "HomeTimeline": "主页时间线", "Likes": "喜欢", "ListMembers": "列表成员", "ListSubscribers": "列表关注者", "ListTimeline": "列表时间线", "RetweetersModule": "转推者", "RuntimeLogs": "运行时日志", "SearchTimeline": "搜索结果", "TweetDetail": "推文详情", "UserDetail": "用户详情", "UserMedia": "用户媒体", "UserTweets": "用户推文", "BrowsingHistory": "浏览历史", "WebhookManager": "Webhook 管理", "Load More": "加载更多", "Save": "保存", "Cancel": "取消", "Test": "测试", "Add": "添加", "Edit": "编辑", "Delete": "删除", "Enabled": "启用", "Active": "活跃", "Inactive": "未启用", "Name": "名称", "URL": "地址", "Events": "事件", "Headers": "请求头", "Retry on failure": "失败时重试", "Max retries": "最大重试次数", "Configs": "配置", "Logs": "日志", "No configs": "暂无 Webhook 配置", "No logs": "暂无 Webhook 日志", "No browsing history": "暂无浏览历史", "Confirm delete webhook": "确定要删除此 Webhook 吗?", "Confirm clear logs": "确定要清空所有 Webhook 日志吗?", "Confirm clear history": "确定要清空所有浏览历史吗?", "Source": "来源", "Viewed At": "浏览时间", "Tweet ID": "推文 ID", "Refresh": "刷新", "Export": "导出", "records": "条记录", "Article Title": "文章标题", "Article Preview": "文章预览", "Article URL": "文章链接", "Article Cover": "文章封面", "Quoted Article Title": "引用文章标题", "Quoted Article Preview": "引用文章预览", "Quoted Article URL": "引用文章链接", "Quoted Article Cover": "引用文章封面" } };
  const resources = {
    "en": en,
    "zh-Hans": zh_Hans
  };
  const logLinesSignal = signals.signal([]);
  class Logger {
    constructor() {
      __publicField(this, "index", 0);
      __publicField(this, "buffer", []);
      __publicField(this, "bufferTimer", null);
    }
    info(line, ...args) {
      console.info("[twitter-web-exporter]", line, ...args);
      this.writeBuffer({ type: "info", line, index: this.index++ });
    }
    warn(line, ...args) {
      console.warn("[twitter-web-exporter]", line, ...args);
      this.writeBuffer({ type: "warn", line, index: this.index++ });
    }
    error(line, ...args) {
      console.error("[twitter-web-exporter]", line, ...args);
      this.writeBuffer({ type: "error", line, index: this.index++ });
    }
    errorWithBanner(msg, err, ...args) {
      this.error(
        `${msg} (Message: ${(err == null ? void 0 : err.message) ?? "none"})
  This may be a problem caused by Twitter updates.
  Please file an issue on GitHub:
  https://github.com/prinsss/twitter-web-exporter/issues`,
        ...args
      );
    }
    debug(...args) {
      console.debug("[twitter-web-exporter]", ...args);
    }
    /**
     * Buffer log lines to reduce the number of signal and DOM updates.
     */
    writeBuffer(log) {
      this.buffer.push(log);
      if (this.bufferTimer) {
        clearTimeout(this.bufferTimer);
      }
      this.bufferTimer = window.setTimeout(() => {
        this.bufferTimer = null;
        this.flushBuffer();
      }, 0);
    }
    /**
     * Flush buffered log lines and update the UI.
     */
    flushBuffer() {
      logLinesSignal.value = [...logLinesSignal.value, ...this.buffer];
      this.buffer = [];
    }
  }
  const logger = new Logger();
  function safeJSONParse(text) {
    try {
      return JSON.parse(text);
    } catch (e) {
      logger.error(e.message);
      return null;
    }
  }
  function useSignalState(value) {
    const signal2 = signals.useSignal(value);
    const updateSignal = (newValue) => {
      signal2.value = newValue;
    };
    return [signal2.value, updateSignal, signal2];
  }
  function useToggle(defaultValue = false) {
    const signal2 = signals.useSignal(defaultValue);
    const toggle = () => {
      signal2.value = !signal2.value;
    };
    return [signal2.value, toggle, signal2];
  }
  function cx(...classNames) {
    return classNames.filter(Boolean).join(" ");
  }
  function isEqual(obj1, obj2) {
    return JSON.stringify(obj1) === JSON.stringify(obj2);
  }
  function capitalizeFirstLetter(str) {
    return str.charAt(0).toUpperCase() + str.slice(1);
  }
  function xssFilter(str) {
    return str.replace(/"/g, "&quot;").replace(/</g, "&lt;").replace(/>/g, "&gt;");
  }
  function strEntitiesToHTML(str, urls) {
    let temp = str;
    if (!(urls == null ? void 0 : urls.length)) {
      return temp;
    }
    for (const { url, display_url, expanded_url } of urls) {
      temp = temp.replaceAll(
        url,
        `<a class="link" target="_blank" href="${xssFilter(expanded_url ?? url)}">${xssFilter(
        display_url ?? url
      )}</a>`
      );
    }
    return temp;
  }
  function parseTwitterDateTime(str) {
    if (!str) {
      return dayjs(0);
    }
    const trimmed = str.replace(/^\w+ (.*)$/, "$1");
    return dayjs(trimmed, "MMM DD HH:mm:ss ZZ YYYY", "en");
  }
  function formatDateTime(date, format) {
    if (typeof date === "number" || typeof date === "string") {
      date = dayjs(date);
    }
    return date.format(format);
  }
  function formatTwitterBirthdate(arg) {
    if (!arg) {
      return null;
    }
    const { day, month, year } = arg;
    const date = dayjs().set("year", year ?? 0).set("month", month - 1).set("date", day);
    return year ? date.format("MMM DD, YYYY") : date.format("MMM DD");
  }
  function formatVideoDuration(durationInMs) {
    if (typeof durationInMs !== "number" || Number.isNaN(durationInMs)) {
      return "N/A";
    }
    const durationInSeconds = Math.floor(durationInMs / 1e3);
    const minutes = Math.floor(durationInSeconds / 60);
    const seconds = durationInSeconds % 60;
    return `${minutes}:${seconds.toString().padStart(2, "0")}`;
  }
  const name = "memoizex";
  const version = "1.0.3";
  const homepage = "https://github.com/memoizex/memoizex";
  const packageJson = {
    name,
    version,
    homepage
  };
  const DEFAULT_APP_OPTIONS = {
    theme: "system",
    debug: false,
    showControlPanel: true,
    disabledExtensions: [
      "HomeTimelineModule",
      "ListTimelineModule",
      "ListSubscribersModule",
      "ListMembersModule",
      "CommunityMembersModule",
      "CommunityTimelineModule",
      "RetweetersModule"
    ],
    dateTimeFormat: "YYYY-MM-DD HH:mm:ss Z",
    filenamePattern: "{screen_name}_{id}_{type}_{num}_{date}.{ext}",
    language: "",
    dedicatedDbForAccounts: false,
    filterPromotedTweets: true,
    filterPromotionalTweets: true,
    version: packageJson.version
  };
  const THEMES = [
    "system",
    "cupcake",
    "dark",
    "emerald",
    "cyberpunk",
    "valentine",
    "lofi",
    "dracula",
    "cmyk",
    "business",
    "winter"
  ];
  const LOCAL_STORAGE_KEY = packageJson.name;
  class AppOptionsManager {
    constructor() {
      __publicField(this, "appOptions", { ...DEFAULT_APP_OPTIONS });
      __publicField(this, "previous", { ...DEFAULT_APP_OPTIONS });
      /**
       * Signal for subscribing to option changes.
       */
      __publicField(this, "signal", new signals.Signal(0));
      this.loadAppOptions();
    }
    get(key, defaultValue) {
      return this.appOptions[key] ?? defaultValue;
    }
    set(key, value) {
      this.appOptions[key] = value;
      this.saveAppOptions();
    }
    /**
     * Read app options from local storage.
     */
    loadAppOptions() {
      this.appOptions = {
        ...this.appOptions,
        ...safeJSONParse(localStorage.getItem(LOCAL_STORAGE_KEY) || "{}")
      };
      const oldVersion = this.appOptions.version ?? "";
      const newVersion = DEFAULT_APP_OPTIONS.version;
      if (newVersion.startsWith("1.1") && oldVersion.startsWith("1.0")) {
        this.appOptions.disabledExtensions = [
          ...this.appOptions.disabledExtensions ?? [],
          "HomeTimelineModule",
          "ListTimelineModule"
        ];
        logger.info(`App options migrated from v${oldVersion} to v${newVersion}`);
        setTimeout(() => this.saveAppOptions(), 0);
      }
      this.previous = { ...this.appOptions };
      logger.info("App options loaded", this.appOptions);
      this.signal.value++;
    }
    /**
     * Write app options to local storage.
     */
    saveAppOptions() {
      const oldValue = this.previous;
      const newValue = {
        ...this.appOptions,
        version: packageJson.version
      };
      if (isEqual(oldValue, newValue)) {
        return;
      }
      this.appOptions = newValue;
      localStorage.setItem(LOCAL_STORAGE_KEY, JSON.stringify(this.appOptions));
      this.previous = { ...this.appOptions };
      logger.debug("App options saved", this.appOptions);
      this.signal.value++;
    }
  }
  const appOptionsManager = new AppOptionsManager();
  const LANGUAGES_CONFIG = {
    en: {
      name: "English",
      nameEn: "English",
      test: (code) => /^en/.test(code)
    },
    "zh-Hans": {
      name: "简体中文",
      nameEn: "Simplified Chinese",
      test: (code) => /^zh/.test(code)
    }
  };
  function detectBrowserLanguage() {
    const language = window.navigator.language || "en";
    for (const [langTag, langConf] of Object.entries(LANGUAGES_CONFIG)) {
      if (langConf.test(language)) {
        return langTag;
      }
    }
    return language;
  }
  const languageDetector = {
    type: "languageDetector",
    detect: function() {
      return appOptionsManager.get("language") || detectBrowserLanguage();
    }
  };
  function initI18n() {
    if (i18next.isInitialized) {
      return i18next;
    }
    i18next.on("languageChanged", (lng) => {
      if (!appOptionsManager.get("language")) {
        appOptionsManager.set("language", lng);
      }
    });
    i18next.use(languageDetector).init({
      initImmediate: true,
      defaultNS: "common",
      fallbackLng: "en",
      nsSeparator: "::",
      debug: appOptionsManager.get("debug"),
      resources
    });
    return i18next;
  }
  function useTranslation(ns) {
    const i18n = initI18n();
    const [t, setT] = hooks.useState(() => i18n.getFixedT(null, ns ?? null));
    const isMountedRef = hooks.useRef(true);
    const previousNamespaceRef = hooks.useRef(ns);
    hooks.useEffect(() => {
      isMountedRef.current = true;
      if (previousNamespaceRef.current !== ns) {
        previousNamespaceRef.current = ns;
        setT(() => i18n.getFixedT(null, ns ?? null));
      }
      function boundReset() {
        if (isMountedRef.current) {
          setT(() => i18n.getFixedT(null, ns ?? null));
        }
      }
      i18n.on("languageChanged", boundReset);
      return () => {
        isMountedRef.current = false;
        i18n.off("languageChanged", boundReset);
      };
    }, [ns]);
    return { t, i18n };
  }
  function Trans({ i18nKey, ns = "exporter" }) {
    const { t } = useTranslation(ns);
    return /* @__PURE__ */ u("span", { children: t(i18nKey) });
  }
  class ErrorBoundary extends preact.Component {
    constructor() {
      super(...arguments);
      __publicField(this, "state", { error: null });
    }
    static getDerivedStateFromError(err) {
      return { error: err.message };
    }
    componentDidCatch(err) {
      logger.error(err.message, err);
      this.setState({ error: err.message });
    }
    render() {
      if (this.state.error) {
        return /* @__PURE__ */ u("div", { class: "alert alert-error p-2", children: [
          /* @__PURE__ */ u(IconExclamationCircle, {}),
          /* @__PURE__ */ u("div", { children: [
            /* @__PURE__ */ u("h3", { class: "font-bold leading-normal", children: /* @__PURE__ */ u(Trans, { ns: "common", i18nKey: "Something went wrong." }) }),
            /* @__PURE__ */ u("p", { class: "text-xs", children: [
              /* @__PURE__ */ u(Trans, { ns: "common", i18nKey: "Error:" }),
              " ",
              this.state.error
            ] })
          ] })
        ] });
      }
      return this.props.children;
    }
  }
  function extractDataFromResponse(response, extractInstructionsFromJson, extractDataFromTimelineEntry) {
    const json = JSON.parse(response.responseText);
    const instructions = extractInstructionsFromJson(json);
    const timelineAddEntriesInstruction = instructions.find(
      (i) => i.type === "TimelineAddEntries"
    );
    const timelineAddEntriesInstructionEntries = (timelineAddEntriesInstruction == null ? void 0 : timelineAddEntriesInstruction.entries) ?? [];
    const newData = [];
    for (const entry of timelineAddEntriesInstructionEntries) {
      if (isTimelineEntryItem(entry)) {
        const data = extractDataFromTimelineEntry(entry);
        if (data) {
          newData.push(data);
        }
      }
    }
    return newData;
  }
  function extractTimelineTweet(itemContent) {
    const tweetUnion = itemContent.tweet_results.result;
    if (!tweetUnion) {
      logger.warn(
        "TimelineTweet is empty. This could happen when the tweet's visibility is limited by Twitter.",
        itemContent
      );
      return null;
    }
    return extractTweetUnion(tweetUnion);
  }
  function extractTimelineUser(itemContent) {
    const user = itemContent.user_results.result;
    if (!user || user.__typename !== "User") {
      logger.warn(
        "TimelineUser is empty. This could happen when the user's account is suspended or deleted.",
        itemContent
      );
      return null;
    }
    return user;
  }
  function isTimelineEntryItem(entry) {
    return entry.content.entryType === "TimelineTimelineItem";
  }
  function isTimelineEntryTweet(entry) {
    return isTimelineEntryItem(entry) && entry.entryId.startsWith("tweet-") && entry.content.itemContent.__typename === "TimelineTweet";
  }
  function isTimelineEntryUser(entry) {
    return isTimelineEntryItem(entry) && entry.entryId.startsWith("user-") && entry.content.itemContent.__typename === "TimelineUser";
  }
  function isTimelineEntryModule(entry) {
    return entry.content.entryType === "TimelineTimelineModule";
  }
  function isTimelineEntryConversationThread(entry) {
    return isTimelineEntryModule(entry) && entry.entryId.startsWith("conversationthread-") && Array.isArray(entry.content.items);
  }
  function isTimelineEntryProfileConversation(entry) {
    return isTimelineEntryModule(entry) && entry.entryId.startsWith("profile-conversation-") && Array.isArray(entry.content.items);
  }
  function isTimelineEntryProfileGrid(entry) {
    return isTimelineEntryModule(entry) && entry.entryId.startsWith("profile-grid-") && Array.isArray(entry.content.items);
  }
  function isTimelineEntrySearchGrid(entry) {
    return isTimelineEntryModule(entry) && entry.entryId.startsWith("search-grid-") && Array.isArray(entry.content.items);
  }
  function isTimelineEntryListSearch(entry) {
    return isTimelineEntryModule(entry) && entry.entryId.startsWith("list-search-") && Array.isArray(entry.content.items);
  }
  function isTimelineEntryCommunitiesGrid(entry) {
    return isTimelineEntryModule(entry) && entry.entryId.startsWith("communities-grid-") && Array.isArray(entry.content.items);
  }
  function extractTweetUnion(tweet) {
    var _a, _b;
    try {
      let result = null;
      if (tweet.__typename === "Tweet") {
        result = filterEmptyTweet(tweet);
      } else if (tweet.__typename === "TweetWithVisibilityResults") {
        result = filterEmptyTweet(tweet.tweet);
      }
      if (result && appOptionsManager.get("filterPromotedTweets", true)) {
        result = filterPromotedTweet(result);
      }
      if (result && appOptionsManager.get("filterPromotionalTweets", true)) {
        result = filterPromotionalTweet(result);
      }
      if (result) {
        return result;
      }
      if (tweet.__typename === "TweetTombstone") {
        logger.warn(`TweetTombstone received (Reason: ${(_b = (_a = tweet.tombstone) == null ? void 0 : _a.text) == null ? void 0 : _b.text})`, tweet);
        return null;
      }
      if (tweet.__typename === "TweetUnavailable") {
        logger.warn("TweetUnavailable received (Reason: unknown)", tweet);
        return null;
      }
      logger.debug(tweet);
      logger.errorWithBanner("Unknown tweet type received");
    } catch (err) {
      logger.debug(tweet);
      logger.errorWithBanner("Failed to extract tweet", err);
    }
    return null;
  }
  function extractRetweetedTweet(tweet) {
    var _a;
    if ((_a = tweet.legacy.retweeted_status_result) == null ? void 0 : _a.result) {
      return extractTweetUnion(tweet.legacy.retweeted_status_result.result);
    }
    return null;
  }
  function extractQuotedTweet(tweet) {
    var _a;
    if ((_a = tweet.quoted_status_result) == null ? void 0 : _a.result) {
      return extractTweetUnion(tweet.quoted_status_result.result);
    }
    return null;
  }
  function extractTweetUserScreenName(tweet) {
    return tweet.core.user_results.result.core.screen_name;
  }
  function extractTweetMedia(tweet) {
    var _a;
    const realTweet = extractRetweetedTweet(tweet) ?? tweet;
    if ((_a = realTweet.legacy.extended_entities) == null ? void 0 : _a.media) {
      return realTweet.legacy.extended_entities.media;
    }
    return realTweet.legacy.entities.media ?? [];
  }
  function extractTweetMediaTags(tweet) {
    const media = extractTweetMedia(tweet);
    const dedupedTags = [];
    for (const item of media) {
      const tags = getMediaTags(item);
      for (const tag of tags) {
        if (dedupedTags.some((t) => t.user_id === tag.user_id)) {
          continue;
        }
        dedupedTags.push(tag);
      }
    }
    return dedupedTags;
  }
  function extractTweetFullText(tweet) {
    var _a;
    return ((_a = tweet.note_tweet) == null ? void 0 : _a.note_tweet_results.result.text) ?? tweet.legacy.full_text;
  }
  function filterEmptyTweet(tweet) {
    if (!tweet.legacy) {
      logger.warn("Empty tweet received", tweet);
      return null;
    }
    return tweet;
  }
  function isPromotedTweet(tweet) {
    var _a;
    const adSourcePatterns = ["twads", "ads-api", "advertiser-interface", "promoted"];
    const sourceLower = ((_a = tweet.source) == null ? void 0 : _a.toLowerCase()) ?? "";
    if (adSourcePatterns.some((pattern) => sourceLower.includes(pattern))) {
      return true;
    }
    if (tweet.legacy.scopes !== void 0) {
      return true;
    }
    return false;
  }
  function isPromotionalTweet(tweet) {
    var _a, _b, _c, _d, _e, _f, _g;
    const media = ((_a = tweet.legacy.extended_entities) == null ? void 0 : _a.media) ?? ((_b = tweet.legacy.entities) == null ? void 0 : _b.media) ?? [];
    for (const item of media) {
      const mediaUrl = item.media_url_https ?? "";
      const videoUrls = ((_d = (_c = item.video_info) == null ? void 0 : _c.variants) == null ? void 0 : _d.map((v) => v.url)) ?? [];
      if (mediaUrl.includes("amplify_video")) {
        return true;
      }
      if (videoUrls.some((url) => url.includes("amplify_video"))) {
        return true;
      }
    }
    const user = (_f = (_e = tweet.core) == null ? void 0 : _e.user_results) == null ? void 0 : _f.result;
    if (((_g = user == null ? void 0 : user.verification) == null ? void 0 : _g.verified_type) === "Business") {
      return true;
    }
    return false;
  }
  function filterPromotedTweet(tweet) {
    if (isPromotedTweet(tweet)) {
      logger.debug("Filtered promoted tweet", tweet.rest_id);
      return null;
    }
    return tweet;
  }
  function filterPromotionalTweet(tweet) {
    if (isPromotionalTweet(tweet)) {
      logger.debug("Filtered promotional tweet", tweet.rest_id);
      return null;
    }
    return tweet;
  }
  function getMediaIndex(tweet, media) {
    const key = media.media_key;
    return extractTweetMedia(tweet).findIndex((value) => value.media_key === key);
  }
  function getMediaOriginalUrl(media) {
    var _a;
    if (media.type === "video" || media.type === "animated_gif") {
      const variants = ((_a = media.video_info) == null ? void 0 : _a.variants) ?? [];
      let maxBitrateVariant = variants[0];
      for (const variant of variants) {
        if (variant.bitrate && variant.bitrate > ((maxBitrateVariant == null ? void 0 : maxBitrateVariant.bitrate) ?? 0)) {
          maxBitrateVariant = variant;
        }
      }
      return (maxBitrateVariant == null ? void 0 : maxBitrateVariant.url) ?? media.media_url_https;
    }
    return formatTwitterImage(media.media_url_https, "orig");
  }
  function getMediaTags(media) {
    var _a, _b;
    return ((_b = (_a = media.features) == null ? void 0 : _a.all) == null ? void 0 : _b.tags) ?? [];
  }
  function formatTwitterImage(imgUrl, name2 = "medium") {
    const regex = /^(https?:\/\/pbs\.twimg\.com\/media\/.+)\.(\w+)$/;
    const match = imgUrl.match(regex);
    if (!match) {
      return `${imgUrl}?name=${name2}`;
    }
    const [, url, ext] = match;
    return `${url}?format=${ext}&name=${name2}`;
  }
  function getProfileImageOriginalUrl(url) {
    return url.replace(/_normal\.(jpe?g|png|gif)$/, ".$1");
  }
  function getFileExtensionFromUrl(url) {
    const regex = /format=(\w+)|\.(\w+)$|\.(\w+)\?.+$/;
    const match = regex.exec(url);
    return (match == null ? void 0 : match[1]) ?? (match == null ? void 0 : match[2]) ?? (match == null ? void 0 : match[3]) ?? "jpg";
  }
  function getTweetURL(tweet) {
    return `https://twitter.com/${extractTweetUserScreenName(tweet)}/status/${tweet.legacy.id_str}`;
  }
  function getUserURL(user) {
    return `https://twitter.com/${typeof user === "string" ? user : user.core.screen_name}`;
  }
  function getInReplyToTweetURL(tweet) {
    return `https://twitter.com/${tweet.legacy.in_reply_to_screen_name}/status/${tweet.legacy.in_reply_to_status_id_str}`;
  }
  function ExtensionPanel({
    title,
    description,
    children,
    onClick,
    active,
    indicatorColor = "bg-secondary"
  }) {
    return /* @__PURE__ */ u("section", { class: "module-panel", children: [
      /* @__PURE__ */ u("div", { class: "h-12 flex items-center justify-start", children: [
        /* @__PURE__ */ u("div", { class: "relative flex h-4 w-4 mr-3 shrink-0", children: [
          active && /* @__PURE__ */ u(
            "span",
            {
              class: cx(
                "animate-ping absolute inline-flex h-full w-full rounded-full opacity-75",
                indicatorColor
              )
            }
          ),
          /* @__PURE__ */ u("span", { class: cx("relative inline-flex rounded-full h-4 w-4", indicatorColor) })
        ] }),
        /* @__PURE__ */ u("div", { class: "flex flex-col flex-grow", children: [
          /* @__PURE__ */ u("p", { class: "text-base m-0 font-medium leading-none", children: title }),
          /* @__PURE__ */ u("p", { class: "text-sm text-base-content leading-5 text-opacity-70 m-0", children: description })
        ] }),
        /* @__PURE__ */ u("button", { class: "btn btn-sm p-0 w-9 h-9", onClick, children: /* @__PURE__ */ u(IconArrowUpRight, {}) })
      ] }),
      children
    ] });
  }
  function Modal({ show, onClose, title, children, class: className }) {
    if (!show) {
      return /* @__PURE__ */ u("dialog", { class: "modal" });
    }
    return /* @__PURE__ */ u("dialog", { class: "modal modal-open", open: true, children: [
      /* @__PURE__ */ u("div", { class: cx("modal-box p-3 flex flex-col", className), children: [
        /* @__PURE__ */ u("header", { class: "flex items-center h-9 mb-2", children: [
          /* @__PURE__ */ u(
            "div",
            {
              onClick: onClose,
              class: "w-9 h-9 mr-2 cursor-pointer flex justify-center items-center transition-colors duration-200 rounded-full hover:bg-base-200",
              children: /* @__PURE__ */ u(IconX, {})
            }
          ),
          /* @__PURE__ */ u("h2", { class: "leading-none text-xl m-0 font-semibold", children: title })
        ] }),
        /* @__PURE__ */ u(ErrorBoundary, { children })
      ] }),
      /* @__PURE__ */ u("form", { method: "dialog", class: "modal-backdrop", children: /* @__PURE__ */ u("div", { onClick: onClose }) })
    ] });
  }
  function SearchArea({ defaultValue, onChange }) {
    const { t } = useTranslation();
    const inputRef = hooks.useRef(null);
    return /* @__PURE__ */ u("div", { class: "join justify-end my-[2px] w-full max-w-[50%] absolute top-3 right-3", children: [
      /* @__PURE__ */ u(
        "input",
        {
          ref: inputRef,
          type: "text",
          class: "input input-bordered input-sm join-item max-w-[calc(100%-46px)]",
          placeholder: t("Search..."),
          defaultValue,
          onKeyDown: (e) => {
            var _a;
            if (e.key === "Enter") {
              onChange(((_a = inputRef.current) == null ? void 0 : _a.value) ?? "");
            }
          }
        }
      ),
      /* @__PURE__ */ u("button", { class: "btn btn-sm join-item", onClick: () => {
        var _a;
        return onChange(((_a = inputRef.current) == null ? void 0 : _a.value) ?? "");
      }, children: /* @__PURE__ */ u(IconSearch, { size: 20 }) })
    ] });
  }
  function MultiSelect(props) {
    const { options: options2, selected, onChange } = props;
    const onInputChange = (e) => {
      const target = e.target;
      if (target.checked) {
        onChange([.../* @__PURE__ */ new Set([...selected, target.value])]);
      } else {
        onChange(selected.filter((value) => value !== target.value));
      }
    };
    return /* @__PURE__ */ u("div", { class: cx("dropdown", props.class), children: [
      /* @__PURE__ */ u(
        "div",
        {
          tabIndex: 0,
          class: "input input-bordered input-sm flex flex-row items-center space-x-1 cursor-pointer",
          children: options2.filter((option) => selected.includes(option.value)).map((option) => /* @__PURE__ */ u("div", { class: "badge badge-accent select-none", children: option.label }, option.value))
        }
      ),
      /* @__PURE__ */ u(
        "ul",
        {
          tabIndex: 0,
          class: "dropdown-content menu menu-sm z-10 w-full rounded-box bg-base-100 p-2 shadow",
          children: options2.map((option) => /* @__PURE__ */ u("li", { children: /* @__PURE__ */ u("label", { class: "label cursor-pointer justify-start", children: [
            /* @__PURE__ */ u(
              "input",
              {
                type: "checkbox",
                class: "checkbox checkbox-accent checkbox-sm",
                value: option.value,
                checked: selected.includes(option.value),
                onChange: onInputChange
              }
            ),
            /* @__PURE__ */ u("span", { class: "label-text ml-1", children: option.label })
          ] }) }, option.value))
        }
      )
    ] });
  }
  function MediaDisplayColumn({ data, onClick }) {
    return /* @__PURE__ */ u("div", { class: "flex flex-row items-start space-x-1 w-max", children: [
      data.map((media) => {
        var _a;
        return /* @__PURE__ */ u(
          "div",
          {
            class: "flex-shrink-0 block cursor-pointer relative w-12 h-12 rounded bg-base-300 overflow-hidden",
            onClick: () => onClick(media),
            children: [
              /* @__PURE__ */ u(
                "img",
                {
                  class: "w-full h-full object-cover",
                  src: formatTwitterImage(media.media_url_https, "thumb"),
                  alt: media.ext_alt_text || "",
                  title: media.ext_alt_text || ""
                }
              ),
              media.type !== "photo" && /* @__PURE__ */ u("div", { class: "absolute bottom-0.5 left-0.5 h-4 w-max px-0.5 text-xs text-white bg-black bg-opacity-30 leading-4 text-center rounded", children: media.type === "video" ? formatVideoDuration((_a = media.video_info) == null ? void 0 : _a.duration_millis) : "GIF" }),
              media.type === "photo" && media.ext_alt_text && /* @__PURE__ */ u("div", { class: "absolute bottom-0.5 left-0.5 h-4 w-max px-0.5 text-xs text-white bg-black bg-opacity-30 leading-4 text-center rounded", children: "ALT" })
            ]
          },
          media.media_key ?? media.id_str
        );
      }),
      data.length ? null : "N/A"
    ] });
  }
  /**
   * @license
   * Credit: https://icooon-mono.com/12776-%e7%8c%ab%e3%81%ae%e7%84%a1%e6%96%99%e3%82%a2%e3%82%a4%e3%82%b3%e3%83%b32/
   */
  const CatIcon = () => /* @__PURE__ */ u("svg", { xmlns: "http://www.w3.org/2000/svg", viewBox: "0 0 512 512", class: "w-full h-full select-none", children: /* @__PURE__ */ u("g", { children: /* @__PURE__ */ u("path", { d: "M461.814,197.514c-2.999-11.335-14.624-18.093-25.958-15.094c-1.866,0.553-13.477,3.649-26.042,14.341 c-6.234,5.349-12.633,12.751-17.361,22.454c-4.748,9.69-7.685,21.577-7.657,35.033c0.013,16.345,4.133,34.895,13.442,56.257 c6.282,14.403,9.144,29.697,9.144,44.846c0.062,25.627-8.438,50.756-21.121,68.283c-6.296,8.777-13.546,15.606-20.816,20.022 c-2.986,1.81-5.943,3.131-8.888,4.181l0.989-5.854c-0.055-17.03-4.05-34.84-13.021-50.528 c-28.356-49.643-66.223-134.741-66.223-134.741l-1.527-4.879c29.47-7.796,58.579-23.408,73.148-54.985 c38.931-84.344-41.08-142.73-41.08-142.73s-25.958-56.222-38.924-54.06c-12.978,2.164-41.094,38.931-41.094,38.931h-23.788h-23.788 c0,0-28.108-36.767-41.08-38.931c-12.979-2.163-38.924,54.06-38.924,54.06s-80.018,58.386-41.087,142.73 c13.822,29.953,40.741,45.572,68.634,53.748l-2.951,9.662c0,0-31.908,81.552-60.279,131.195C37.198,441.092,58.478,512,97.477,512 c29.47,0,79.14,0,101.692,0c7.292,0,11.763,0,11.763,0c22.544,0,72.222,0,101.691,0c12.654,0,23.38-7.547,31.204-19.324 c15.826-0.013,30.81-4.872,43.707-12.758c19.455-11.915,34.708-30.32,45.434-51.896c10.685-21.618,16.856-46.636,16.878-72.672 c0-20.484-3.885-41.619-12.682-61.813c-7.561-17.34-9.918-30.216-9.904-39.29c0.028-7.526,1.5-12.544,3.359-16.414 c1.417-2.889,3.124-5.17,4.983-7.091c2.771-2.868,5.964-4.879,8.349-6.054c1.182-0.595,2.135-0.968,2.674-1.162l0.449-0.152 l-0.007-0.028C458.179,220.189,464.779,208.724,461.814,197.514z" }) }) });
  const globalObject = _unsafeWindow ?? window ?? globalThis;
  const xhrOpen = globalObject.XMLHttpRequest.prototype.open;
  const xhrSend = globalObject.XMLHttpRequest.prototype.send;
  const xhrBodyMap = /* @__PURE__ */ new WeakMap();
  class ExtensionManager {
    constructor() {
      __publicField(this, "extensions", /* @__PURE__ */ new Map());
      __publicField(this, "disabledExtensions", /* @__PURE__ */ new Set());
      __publicField(this, "debugEnabled", false);
      /**
       * Signal for subscribing to extension changes.
       */
      __publicField(this, "signal", new signals.Signal(1));
      this.installHttpHooks();
      this.disabledExtensions = new Set(appOptionsManager.get("disabledExtensions", []));
      if (appOptionsManager.get("debug")) {
        this.debugEnabled = true;
        logger.info("Debug mode enabled");
      }
    }
    /**
     * Register and instantiate a new extension.
     *
     * @param ctor Extension constructor.
     */
    add(ctor) {
      try {
        logger.debug(`Register new extension: ${ctor.name}`);
        const instance = new ctor(this);
        this.extensions.set(instance.name, instance);
      } catch (err) {
        logger.error(`Failed to register extension: ${ctor.name}`, err);
      }
    }
    /**
     * Set up all enabled extensions.
     */
    start() {
      for (const ext of this.extensions.values()) {
        if (this.disabledExtensions.has(ext.name)) {
          this.disable(ext.name);
        } else {
          this.enable(ext.name);
        }
      }
    }
    enable(name2) {
      try {
        this.disabledExtensions.delete(name2);
        appOptionsManager.set("disabledExtensions", [...this.disabledExtensions]);
        const ext = this.extensions.get(name2);
        ext.enabled = true;
        ext.setup();
        logger.debug(`Enabled extension: ${name2}`);
        this.signal.value++;
      } catch (err) {
        logger.error(`Failed to enable extension: ${name2}`, err);
      }
    }
    disable(name2) {
      try {
        this.disabledExtensions.add(name2);
        appOptionsManager.set("disabledExtensions", [...this.disabledExtensions]);
        const ext = this.extensions.get(name2);
        ext.enabled = false;
        ext.dispose();
        logger.debug(`Disabled extension: ${name2}`);
        this.signal.value++;
      } catch (err) {
        logger.error(`Failed to disable extension: ${name2}`, err);
      }
    }
    getExtensions() {
      return [...this.extensions.values()];
    }
    /**
     * Here we hooks the browser's XHR method to intercept Twitter's Web API calls.
     * This need to be done before any XHR request is made.
     */
    installHttpHooks() {
      const manager = this;
      globalObject.XMLHttpRequest.prototype.send = function(body) {
        if (body && typeof body === "string") {
          xhrBodyMap.set(this, body);
        }
        return xhrSend.apply(this, arguments);
      };
      globalObject.XMLHttpRequest.prototype.open = function(method, url) {
        if (manager.debugEnabled) {
          logger.debug(`XHR initialized`, { method, url });
        }
        this.addEventListener("load", () => {
          const requestBody = xhrBodyMap.get(this) || null;
          if (manager.debugEnabled) {
            logger.debug(`XHR finished`, { method, url, hasBody: !!requestBody });
          }
          manager.getExtensions().filter((ext) => ext.enabled).forEach((ext) => {
            const func = ext.intercept();
            if (func) {
              func({ method, url, body: requestBody }, this, ext);
            }
          });
        });
        xhrOpen.apply(this, arguments);
      };
      logger.info("Hooked into XMLHttpRequest");
      setTimeout(() => {
        if (!("webpackChunk_twitter_responsive_web" in globalObject)) {
          logger.error(
            'Error: Wrong execution context detected.\n  This script needs to be injected into "page" context rather than "content" context.\n  The XMLHttpRequest hook will not work properly.\n  See: https://github.com/prinsss/twitter-web-exporter/issues/19'
          );
        }
      }, 1e3);
    }
  }
  var ExtensionType = /* @__PURE__ */ ((ExtensionType2) => {
    ExtensionType2["TWEET"] = "tweet";
    ExtensionType2["USER"] = "user";
    ExtensionType2["CUSTOM"] = "custom";
    ExtensionType2["NONE"] = "none";
    return ExtensionType2;
  })(ExtensionType || {});
  class Extension {
    constructor(manager) {
      __publicField(this, "name", "");
      __publicField(this, "enabled", true);
      __publicField(this, "type", "none");
      __publicField(this, "manager");
      this.manager = manager;
    }
    /**
     * Optionally run side effects when enabled.
     */
    setup() {
    }
    /**
     * Optionally clear side effects when disabled.
     */
    dispose() {
    }
    /**
     * Intercept HTTP responses.
     */
    intercept() {
      return null;
    }
    /**
     * Render extension UI.
     */
    render() {
      return null;
    }
  }
  const extensionManager = new ExtensionManager();
  const EXPORT_FORMAT = {
    JSON: "JSON",
    HTML: "HTML",
    CSV: "CSV"
  };
  function csvEscapeStr(str) {
    return `"${str.replace(/"/g, '""').replace(/\n/g, "\\n").replace(/\r/g, "\\r")}"`;
  }
  function saveFile(filename, content, prependBOM = false) {
    const link = document.createElement("a");
    let blob;
    if (content instanceof Blob) {
      blob = content;
    } else {
      blob = new Blob(prependBOM ? [new Uint8Array([239, 187, 191]), content] : [content], {
        type: "text/plain;charset=utf-8"
      });
    }
    const url = URL.createObjectURL(blob);
    link.href = url;
    link.download = filename;
    link.click();
    URL.revokeObjectURL(url);
  }
  async function exportData(data, format, filename, translations) {
    try {
      let content = "";
      let prependBOM = false;
      logger.info(`Exporting to ${format} file: ${filename}`);
      switch (format) {
        case EXPORT_FORMAT.JSON:
          content = await jsonExporter(data);
          break;
        case EXPORT_FORMAT.HTML:
          content = await htmlExporter(data, translations);
          break;
        case EXPORT_FORMAT.CSV:
          prependBOM = true;
          content = await csvExporter(data);
          break;
      }
      saveFile(filename, content, prependBOM);
    } catch (err) {
      logger.errorWithBanner("Failed to export file", err);
    }
  }
  async function jsonExporter(data) {
    return JSON.stringify(data, void 0, "  ");
  }
  async function htmlExporter(data, translations) {
    const table = document.createElement("table");
    const thead = document.createElement("thead");
    const tbody = document.createElement("tbody");
    const exportKeys = Object.keys(data[0] ?? {});
    const headerRow = document.createElement("tr");
    for (const exportKey of exportKeys) {
      const th = document.createElement("th");
      th.textContent = translations[exportKey] ?? exportKey;
      headerRow.appendChild(th);
    }
    thead.appendChild(headerRow);
    table.appendChild(thead);
    table.className = "table table-striped";
    for (const row of data) {
      const tr = document.createElement("tr");
      for (const exportKey of exportKeys) {
        const td = document.createElement("td");
        const value = row[exportKey];
        if (exportKey === "profile_image_url" || exportKey === "profile_banner_url") {
          const img = document.createElement("img");
          img.src = value;
          img.width = 50;
          td.innerHTML = "";
          td.appendChild(img);
        } else if (exportKey === "media") {
          if ((value == null ? void 0 : value.length) > 0) {
            for (const media of value) {
              const img = document.createElement("img");
              img.src = media.thumbnail;
              img.width = 50;
              img.alt = media.ext_alt_text || "";
              img.title = media.ext_alt_text || "";
              const link = document.createElement("a");
              link.href = media.original;
              link.target = "_blank";
              link.style.marginRight = "0.5em";
              link.appendChild(img);
              td.appendChild(link);
            }
          }
        } else if (exportKey === "full_text" || exportKey === "description") {
          const p = document.createElement("p");
          p.innerHTML = value;
          p.style.whiteSpace = "pre-wrap";
          p.style.maxWidth = "640px";
          td.appendChild(p);
        } else if (exportKey === "metadata") {
          const details = document.createElement("details");
          const summary = document.createElement("summary");
          summary.textContent = "Expand";
          details.appendChild(summary);
          const pre = document.createElement("pre");
          pre.textContent = JSON.stringify(value, void 0, "  ");
          details.appendChild(pre);
          td.appendChild(details);
        } else if (exportKey === "url") {
          const link = document.createElement("a");
          link.href = value;
          link.target = "_blank";
          link.textContent = value;
          td.appendChild(link);
        } else {
          td.textContent = typeof value === "string" ? value : JSON.stringify(row[exportKey]);
        }
        tr.appendChild(td);
      }
      tbody.appendChild(tr);
    }
    table.appendChild(tbody);
    return `
    <html>
      <head>
        <meta charset="utf-8">
        <title>Exported Data ${(/* @__PURE__ */ new Date()).toISOString()}</title>
        <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css">
      </head>
      <body>
        ${table.outerHTML}
      </body>
    </html>
  `;
  }
  async function csvExporter(data) {
    const headers = Object.keys(data[0] ?? {});
    let content = headers.join(",") + "\n";
    for (const row of data) {
      const values = headers.map((header) => {
        const value = row[header];
        if (typeof value === "string") {
          return csvEscapeStr(value);
        }
        if (typeof value === "object") {
          return csvEscapeStr(JSON.stringify(value));
        }
        return value;
      });
      content += values.join(",");
      content += "\n";
    }
    return content;
  }
  function formatWebhookPayload(tweet, eventType) {
    return {
      event: eventType,
      timestamp: Date.now(),
      data: formatTweetData(tweet)
    };
  }
  function extractTweetFromUnion(tweetUnion) {
    if (!tweetUnion) return void 0;
    if (tweetUnion.__typename === "Tweet") return tweetUnion;
    if (tweetUnion.__typename === "TweetWithVisibilityResults") return tweetUnion.tweet;
    return void 0;
  }
  function formatTweetData(tweet) {
    var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l;
    const author = (_b = (_a = tweet.core) == null ? void 0 : _a.user_results) == null ? void 0 : _b.result;
    const legacy = tweet.legacy;
    const screenName = ((_c = author == null ? void 0 : author.core) == null ? void 0 : _c.screen_name) || "unknown";
    const isMinimalTweet = !legacy && !author;
    if (isMinimalTweet) {
      return {
        id: tweet.rest_id,
        text: "",
        author: { id: "", screen_name: "unknown", name: "" },
        url: `https://x.com/i/status/${tweet.rest_id}`,
        created_at: "",
        stats: { likes: 0, retweets: 0, replies: 0, quotes: 0, bookmarks: 0 },
        media: []
      };
    }
    const fullText = ((_f = (_e = (_d = tweet.note_tweet) == null ? void 0 : _d.note_tweet_results) == null ? void 0 : _e.result) == null ? void 0 : _f.text) || (legacy == null ? void 0 : legacy.full_text) || "";
    const result = {
      id: tweet.rest_id,
      text: fullText,
      author: {
        id: (author == null ? void 0 : author.rest_id) || "",
        screen_name: screenName,
        name: ((_g = author == null ? void 0 : author.core) == null ? void 0 : _g.name) || ""
      },
      url: `https://x.com/${screenName}/status/${tweet.rest_id}`,
      created_at: (legacy == null ? void 0 : legacy.created_at) || "",
      stats: {
        likes: (legacy == null ? void 0 : legacy.favorite_count) || 0,
        retweets: (legacy == null ? void 0 : legacy.retweet_count) || 0,
        replies: (legacy == null ? void 0 : legacy.reply_count) || 0,
        quotes: (legacy == null ? void 0 : legacy.quote_count) || 0,
        bookmarks: (legacy == null ? void 0 : legacy.bookmark_count) || 0
      },
      media: formatMediaItems(tweet)
    };
    const retweetedTweet = extractTweetFromUnion((_h = legacy == null ? void 0 : legacy.retweeted_status_result) == null ? void 0 : _h.result);
    if (retweetedTweet) {
      result.retweeted_tweet = formatRetweetedTweet(retweetedTweet);
    }
    const quotedTweet = extractTweetFromUnion((_i = tweet.quoted_status_result) == null ? void 0 : _i.result);
    if (quotedTweet) {
      result.quoted_tweet = formatQuotedTweet(quotedTweet);
    }
    const urls = (_j = legacy == null ? void 0 : legacy.entities) == null ? void 0 : _j.urls;
    if (urls && urls.length > 0) {
      result.urls = formatUrls(urls);
    }
    const article = (_l = (_k = tweet.article) == null ? void 0 : _k.article_results) == null ? void 0 : _l.result;
    if (article) {
      result.article = formatArticle(article, urls);
    }
    return result;
  }
  function formatRetweetedTweet(tweet) {
    var _a, _b, _c, _d, _e, _f, _g;
    const author = (_b = (_a = tweet.core) == null ? void 0 : _a.user_results) == null ? void 0 : _b.result;
    const legacy = tweet.legacy;
    const screenName = ((_c = author == null ? void 0 : author.core) == null ? void 0 : _c.screen_name) || "unknown";
    const fullText = ((_f = (_e = (_d = tweet.note_tweet) == null ? void 0 : _d.note_tweet_results) == null ? void 0 : _e.result) == null ? void 0 : _f.text) || (legacy == null ? void 0 : legacy.full_text) || "";
    return {
      id: tweet.rest_id,
      text: fullText,
      author: {
        id: (author == null ? void 0 : author.rest_id) || "",
        screen_name: screenName,
        name: ((_g = author == null ? void 0 : author.core) == null ? void 0 : _g.name) || ""
      },
      url: `https://x.com/${screenName}/status/${tweet.rest_id}`,
      created_at: (legacy == null ? void 0 : legacy.created_at) || "",
      media: formatMediaItems(tweet)
    };
  }
  function formatQuotedTweet(tweet) {
    var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j;
    const author = (_b = (_a = tweet.core) == null ? void 0 : _a.user_results) == null ? void 0 : _b.result;
    const legacy = tweet.legacy;
    const screenName = ((_c = author == null ? void 0 : author.core) == null ? void 0 : _c.screen_name) || "unknown";
    const fullText = ((_f = (_e = (_d = tweet.note_tweet) == null ? void 0 : _d.note_tweet_results) == null ? void 0 : _e.result) == null ? void 0 : _f.text) || (legacy == null ? void 0 : legacy.full_text) || "";
    const result = {
      id: tweet.rest_id,
      text: fullText,
      author: {
        id: (author == null ? void 0 : author.rest_id) || "",
        screen_name: screenName,
        name: ((_g = author == null ? void 0 : author.core) == null ? void 0 : _g.name) || ""
      },
      url: `https://x.com/${screenName}/status/${tweet.rest_id}`,
      created_at: (legacy == null ? void 0 : legacy.created_at) || ""
    };
    const article = (_i = (_h = tweet.article) == null ? void 0 : _h.article_results) == null ? void 0 : _i.result;
    if (article) {
      result.article = formatArticle(article, (_j = legacy == null ? void 0 : legacy.entities) == null ? void 0 : _j.urls);
    }
    return result;
  }
  function formatUrls(urls) {
    return urls.filter((u2) => !u2.expanded_url.includes("twitter.com/") && !u2.expanded_url.includes("x.com/")).map((u2) => ({
      url: u2.url,
      expanded_url: u2.expanded_url,
      display_url: u2.display_url
    }));
  }
  function formatArticle(article, urls) {
    var _a, _b, _c;
    const articleUrl = ((_a = urls == null ? void 0 : urls.find((u2) => u2.expanded_url.includes("/article/"))) == null ? void 0 : _a.expanded_url) || `https://x.com/i/article/${article.rest_id}`;
    return {
      id: article.rest_id,
      title: article.title,
      preview_text: article.preview_text,
      url: articleUrl,
      cover_image_url: (_c = (_b = article.cover_media) == null ? void 0 : _b.media_info) == null ? void 0 : _c.original_img_url
    };
  }
  function formatMediaItems(tweet) {
    const mediaItems = extractTweetMedia(tweet);
    return mediaItems.map((item) => ({
      type: item.type,
      url: item.url,
      thumbnail: formatTwitterImage(item.media_url_https, "thumb"),
      original: getMediaOriginalUrl(item)
    }));
  }
  function generateWebhookLogId(eventType, tweetId, timestamp) {
    return `webhook-${eventType}-${tweetId}-${timestamp}`;
  }
  function generateWebhookConfigId() {
    return `webhook-config-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
  }
  const DEFAULT_DEDUP_WINDOW_MS = 5 * 60 * 1e3;
  const MAX_BROWSING_HISTORY_COUNT = 1e4;
  function generateBrowsingHistoryId(tweetId, timestamp) {
    return `browsing-${tweetId}-${timestamp}`;
  }
  async function migration_20250609(tx) {
    logger.info("Running migration_20250609: Migrating legacy user data format");
    let userCount = 0;
    await tx.table("users").toCollection().modify((user) => {
      if (user.core && user.avatar) {
        return;
      }
      migrateFromLegacyUser(user);
      logger.debug(`Migrated user: ${user.rest_id}`);
      user.twe_private_fields.migrated_at = Date.now();
      userCount++;
    });
    let tweetCount = 0;
    await tx.table("tweets").toCollection().modify((tweet) => {
      var _a, _b, _c, _d, _e, _f;
      const user = (_b = (_a = tweet.core) == null ? void 0 : _a.user_results) == null ? void 0 : _b.result;
      if (user && user.core && user.avatar) {
        return;
      }
      migrateFromLegacyUser(user);
      logger.debug(`Migrated tweet user: ${tweet.rest_id} `);
      const rtSource = extractRetweetedTweet(tweet);
      if ((_d = (_c = rtSource == null ? void 0 : rtSource.core) == null ? void 0 : _c.user_results) == null ? void 0 : _d.result) {
        migrateFromLegacyUser(rtSource.core.user_results.result);
        logger.debug(`Migrated retweeted user: ${rtSource.rest_id}`);
      }
      const qtSource = extractQuotedTweet(tweet);
      if ((_f = (_e = qtSource == null ? void 0 : qtSource.core) == null ? void 0 : _e.user_results) == null ? void 0 : _f.result) {
        migrateFromLegacyUser(qtSource.core.user_results.result);
        logger.debug(`Migrated quoted user: ${qtSource.rest_id}`);
      }
      tweet.twe_private_fields.migrated_at = Date.now();
      tweetCount++;
    });
    logger.info(`Migration completed: ${userCount} users and ${tweetCount} tweets updated.`);
  }
  function migrateFromLegacyUser(user) {
    const ul = user;
    if (!(ul == null ? void 0 : ul.legacy)) {
      logger.debug("migrateFromLegacyUser", user);
      logger.warn(`User ${user == null ? void 0 : user.rest_id} has no legacy data to migrate.`);
      return;
    }
    user.core ?? (user.core = {
      created_at: ul.legacy.created_at,
      name: ul.legacy.name,
      screen_name: ul.legacy.screen_name
    });
    user.avatar ?? (user.avatar = {
      image_url: ul.legacy.profile_image_url_https
    });
    user.verification ?? (user.verification = {
      verified: ul.legacy.verified,
      verified_type: ul.legacy.verified_type
    });
    user.relationship_perspectives ?? (user.relationship_perspectives = {
      following: ul.legacy.following,
      followed_by: ul.legacy.followed_by
    });
    user.dm_permissions ?? (user.dm_permissions = {
      can_dm: ul.legacy.can_dm
    });
    user.privacy ?? (user.privacy = {
      protected: ul.legacy.protected
    });
    user.location ?? (user.location = {
      location: ul.legacy.location
    });
  }
  const DB_NAME = packageJson.name;
  const DB_VERSION = 3;
  class DatabaseManager {
    constructor() {
      __publicField(this, "db");
      var _a;
      const globalObject2 = _unsafeWindow ?? window ?? globalThis;
      const userId = ((_a = globalObject2.__META_DATA__) == null ? void 0 : _a.userId) ?? "unknown";
      const suffix = appOptionsManager.get("dedicatedDbForAccounts") ? `_${userId}` : "";
      logger.debug(`Using database: ${DB_NAME}${suffix} for userId: ${userId}`);
      this.db = new Dexie(`${DB_NAME}${suffix}`);
      this.init();
    }
    /*
    |--------------------------------------------------------------------------
    | Type-Safe Table Accessors
    |--------------------------------------------------------------------------
    */
    tweets() {
      return this.db.table("tweets");
    }
    users() {
      return this.db.table("users");
    }
    captures() {
      return this.db.table("captures");
    }
    browsingHistory() {
      return this.db.table("browsing_history");
    }
    webhookConfigs() {
      return this.db.table("webhook_configs");
    }
    webhookLogs() {
      return this.db.table("webhook_logs");
    }
    /*
    |--------------------------------------------------------------------------
    | Read Methods for Extensions
    |--------------------------------------------------------------------------
    */
    async extGetCaptures(extName) {
      return this.captures().where("extension").equals(extName).toArray().catch(this.logError);
    }
    async extGetCaptureCount(extName) {
      return this.captures().where("extension").equals(extName).count().catch(this.logError);
    }
    async extGetCapturedTweets(extName) {
      const captures = await this.extGetCaptures(extName);
      if (!captures) {
        return [];
      }
      const tweetIds = captures.map((capture) => capture.data_key);
      return this.tweets().where("rest_id").anyOf(tweetIds).filter((t) => this.filterEmptyData(t)).toArray().catch(this.logError);
    }
    async extGetCapturedUsers(extName) {
      const captures = await this.extGetCaptures(extName);
      if (!captures) {
        return [];
      }
      const userIds = captures.map((capture) => capture.data_key);
      return this.users().where("rest_id").anyOf(userIds).filter((t) => this.filterEmptyData(t)).toArray().catch(this.logError);
    }
    /*
    |--------------------------------------------------------------------------
    | Write Methods for Extensions
    |--------------------------------------------------------------------------
    */
    async extAddTweets(extName, tweets) {
      await this.upsertTweets(tweets);
      await this.upsertCaptures(
        tweets.map((tweet) => ({
          id: `${extName}-${tweet.rest_id}`,
          extension: extName,
          type: ExtensionType.TWEET,
          data_key: tweet.rest_id,
          created_at: Date.now()
        }))
      );
    }
    async extAddUsers(extName, users) {
      await this.upsertUsers(users);
      await this.upsertCaptures(
        users.map((user) => ({
          id: `${extName}-${user.rest_id}`,
          extension: extName,
          type: ExtensionType.USER,
          data_key: user.rest_id,
          created_at: Date.now()
        }))
      );
    }
    /*
    |--------------------------------------------------------------------------
    | Delete Methods for Extensions
    |--------------------------------------------------------------------------
    */
    async extClearCaptures(extName) {
      const captures = await this.extGetCaptures(extName);
      if (!captures) {
        return;
      }
      return this.captures().bulkDelete(captures.map((capture) => capture.id));
    }
    /*
    |--------------------------------------------------------------------------
    | Tweet Lookup Methods
    |--------------------------------------------------------------------------
    */
    /**
     * Get a tweet by its rest_id.
     */
    async getTweetById(tweetId) {
      try {
        return await this.tweets().where("rest_id").equals(tweetId).first();
      } catch (err) {
        this.logError(err);
        return void 0;
      }
    }
    /*
    |--------------------------------------------------------------------------
    | Browsing History Methods
    |--------------------------------------------------------------------------
    */
    async addBrowsingHistories(histories) {
      if (histories.length === 0) return;
      return this.db.transaction("rw", this.browsingHistory(), async () => {
        await this.browsingHistory().bulkPut(histories);
        const count = await this.browsingHistory().count();
        if (count > MAX_BROWSING_HISTORY_COUNT) {
          const excess = count - MAX_BROWSING_HISTORY_COUNT;
          const oldestRecords = await this.browsingHistory().orderBy("viewed_at").limit(excess).toArray();
          await this.browsingHistory().bulkDelete(oldestRecords.map((r) => r.id));
          logger.debug(`Cleaned up ${excess} old browsing history records`);
        }
      }).catch(this.logError);
    }
    async getBrowsingHistories(limit = 100, offset = 0) {
      return this.browsingHistory().orderBy("viewed_at").reverse().offset(offset).limit(limit).toArray().catch(this.logError);
    }
    async getBrowsingHistoryCount() {
      return this.browsingHistory().count().catch(this.logError);
    }
    async getRecentBrowsingHistories(windowMs) {
      const since = Date.now() - windowMs;
      return this.browsingHistory().where("viewed_at").above(since).toArray().catch(this.logError);
    }
    async clearBrowsingHistory() {
      return this.browsingHistory().clear().catch(this.logError);
    }
    async getBrowsingHistoriesByTweetIds(tweetIds) {
      return this.browsingHistory().where("tweet_id").anyOf(tweetIds).toArray().catch(this.logError);
    }
    /*
    |--------------------------------------------------------------------------
    | Webhook Config Methods
    |--------------------------------------------------------------------------
    */
    async addWebhookConfig(config) {
      return this.webhookConfigs().put(config).catch(this.logError);
    }
    async updateWebhookConfig(id, updates) {
      return this.webhookConfigs().update(id, { ...updates, updated_at: Date.now() }).catch(this.logError);
    }
    async deleteWebhookConfig(id) {
      return this.webhookConfigs().delete(id).catch(this.logError);
    }
    async getWebhookConfigs() {
      return this.webhookConfigs().toArray().catch(this.logError);
    }
    async getWebhookConfigById(id) {
      return this.webhookConfigs().get(id).catch(this.logError);
    }
    async getEnabledWebhookConfigs() {
      return this.webhookConfigs().where("enabled").equals(1).toArray().catch(this.logError);
    }
    /*
    |--------------------------------------------------------------------------
    | Webhook Log Methods
    |--------------------------------------------------------------------------
    */
    async addWebhookLog(log) {
      return this.webhookLogs().put(log).catch(this.logError);
    }
    async updateWebhookLog(id, updates) {
      return this.webhookLogs().update(id, updates).catch(this.logError);
    }
    async getWebhookLogs(limit = 100, offset = 0) {
      return this.webhookLogs().orderBy("created_at").reverse().offset(offset).limit(limit).toArray().catch(this.logError);
    }
    async getWebhookLogCount() {
      return this.webhookLogs().count().catch(this.logError);
    }
    async getPendingWebhookLogs() {
      return this.webhookLogs().where("status").equals("pending").toArray().catch(this.logError);
    }
    async clearWebhookLogs() {
      return this.webhookLogs().clear().catch(this.logError);
    }
    /*
    |--------------------------------------------------------------------------
    | Export and Import Methods
    |--------------------------------------------------------------------------
    */
    async export() {
      return dexieExportImport.exportDB(this.db).catch(this.logError);
    }
    async import(data) {
      return dexieExportImport.importInto(this.db, data).catch(this.logError);
    }
    async clear() {
      await this.deleteAllCaptures();
      await this.deleteAllTweets();
      await this.deleteAllUsers();
      await this.clearBrowsingHistory();
      await this.clearWebhookLogs();
      logger.info("Database cleared");
    }
    async count() {
      try {
        return {
          tweets: await this.tweets().count(),
          users: await this.users().count(),
          captures: await this.captures().count(),
          browsingHistory: await this.browsingHistory().count(),
          webhookConfigs: await this.webhookConfigs().count(),
          webhookLogs: await this.webhookLogs().count()
        };
      } catch (error) {
        this.logError(error);
        return null;
      }
    }
    /*
    |--------------------------------------------------------------------------
    | Common Methods
    |--------------------------------------------------------------------------
    */
    async upsertTweets(tweets) {
      return this.db.transaction("rw", this.tweets(), () => {
        const data = tweets.map((tweet) => ({
          ...tweet,
          twe_private_fields: {
            created_at: +parseTwitterDateTime(tweet.legacy.created_at),
            updated_at: Date.now(),
            media_count: extractTweetMedia(tweet).length
          }
        }));
        return this.tweets().bulkPut(data);
      }).catch(this.logError);
    }
    async upsertUsers(users) {
      return this.db.transaction("rw", this.users(), () => {
        const data = users.map((user) => ({
          ...user,
          twe_private_fields: {
            created_at: +parseTwitterDateTime(user.core.created_at),
            updated_at: Date.now()
          }
        }));
        return this.users().bulkPut(data);
      }).catch(this.logError);
    }
    async upsertCaptures(captures) {
      return this.db.transaction("rw", this.captures(), () => {
        return this.captures().bulkPut(captures).catch(this.logError);
      }).catch(this.logError);
    }
    async getTweetsByIds(tweetIds) {
      return this.tweets().where("rest_id").anyOf(tweetIds).filter((t) => this.filterEmptyData(t)).toArray().catch(this.logError);
    }
    async deleteAllTweets() {
      return this.tweets().clear().catch(this.logError);
    }
    async deleteAllUsers() {
      return this.users().clear().catch(this.logError);
    }
    async deleteAllCaptures() {
      return this.captures().clear().catch(this.logError);
    }
    filterEmptyData(data) {
      if (!(data == null ? void 0 : data.legacy)) {
        logger.warn("Empty data found in DB", data);
        return false;
      }
      return true;
    }
    /*
    |--------------------------------------------------------------------------
    | Migrations
    |--------------------------------------------------------------------------
    */
    async init() {
      const tweetIndexPaths = [
        "rest_id",
        "twe_private_fields.created_at",
        "twe_private_fields.updated_at",
        "twe_private_fields.media_count",
        "core.user_results.result.core.screen_name",
        "legacy.favorite_count",
        "legacy.retweet_count",
        "legacy.bookmark_count",
        "legacy.quote_count",
        "legacy.reply_count",
        "views.count",
        "legacy.favorited",
        "legacy.retweeted",
        "legacy.bookmarked"
      ];
      const userIndexPaths = [
        "rest_id",
        "twe_private_fields.created_at",
        "twe_private_fields.updated_at",
        "core.screen_name",
        "legacy.followers_count",
        "legacy.friends_count",
        "legacy.statuses_count",
        "legacy.favourites_count",
        "legacy.listed_count",
        "verification.verified_type",
        "is_blue_verified",
        "relationship_perspectives.following",
        "relationship_perspectives.followed_by"
      ];
      const captureIndexPaths = ["id", "extension", "type", "created_at"];
      const browsingHistoryIndexPaths = [
        "id",
        "tweet_id",
        "viewed_at",
        "source_page"
      ];
      const webhookConfigIndexPaths = [
        "id",
        "enabled",
        "created_at",
        "updated_at"
      ];
      const webhookLogIndexPaths = [
        "id",
        "event_type",
        "tweet_id",
        "status",
        "created_at"
      ];
      try {
        this.db.version(2).stores({
          tweets: tweetIndexPaths.join(","),
          users: userIndexPaths.join(","),
          captures: captureIndexPaths.join(",")
        }).upgrade(async (tx) => {
          logger.info("Upgrading database schema to v2...");
          await migration_20250609(tx);
          logger.info("Database upgraded to v2");
        });
        this.db.version(DB_VERSION).stores({
          tweets: tweetIndexPaths.join(","),
          users: userIndexPaths.join(","),
          captures: captureIndexPaths.join(","),
          browsing_history: browsingHistoryIndexPaths.join(","),
          webhook_configs: webhookConfigIndexPaths.join(","),
          webhook_logs: webhookLogIndexPaths.join(",")
        });
        await this.db.open();
        logger.info(`Database connected: ${this.db.name}`);
      } catch (error) {
        this.logError(error);
      }
    }
    /*
    |--------------------------------------------------------------------------
    | Loggers
    |--------------------------------------------------------------------------
    */
    logError(error) {
      logger.error(`Database Error: ${error.message}`, error);
    }
  }
  const databaseManager = new DatabaseManager();
  const DEFAULT_TIMEOUT = 1e4;
  class WebhookSender {
    /**
     * Sends a webhook POST request.
     *
     * @param payload - The webhook payload to send
     * @param options - Send options including URL and headers
     * @returns Promise resolving to the send result
     */
    async send(payload, options2) {
      const { url, headers = {}, timeout = DEFAULT_TIMEOUT } = options2;
      try {
        const result = await this.makeRequest(url, payload, headers, timeout);
        return result;
      } catch (error) {
        logger.error("Webhook send error:", error);
        return {
          success: false,
          error: error.message || "Unknown error"
        };
      }
    }
    /**
     * Makes the actual HTTP request using GM_xmlhttpRequest.
     */
    makeRequest(url, payload, headers, timeout) {
      return new Promise((resolve) => {
        try {
          _GM.xmlHttpRequest({
            method: "POST",
            url,
            headers: {
              "Content-Type": "application/json",
              ...headers
            },
            data: JSON.stringify(payload),
            timeout,
            onload: (response) => {
              const success = response.status >= 200 && response.status < 300;
              resolve({
                success,
                status: response.status,
                responseText: response.responseText,
                error: success ? void 0 : `HTTP ${response.status}: ${response.statusText}`
              });
            },
            onerror: (error) => {
              resolve({
                success: false,
                error: `Network error: ${error.statusText || "Request failed"}`
              });
            },
            ontimeout: () => {
              resolve({
                success: false,
                error: `Request timeout after ${timeout}ms`
              });
            }
          });
        } catch (error) {
          resolve({
            success: false,
            error: error.message || "Failed to send request"
          });
        }
      });
    }
    /**
     * Tests a webhook URL with a test payload.
     *
     * @param url - Webhook URL to test
     * @param headers - Custom headers
     * @returns Promise resolving to the test result
     */
    async test(url, headers = {}) {
      const testPayload = {
        event: "like",
        timestamp: Date.now(),
        data: {
          id: "test-tweet-id",
          text: "This is a test webhook from MemoizeX",
          author: {
            id: "test-author-id",
            screen_name: "test_user",
            name: "Test User"
          },
          url: "https://x.com/test_user/status/test-tweet-id",
          created_at: (/* @__PURE__ */ new Date()).toISOString(),
          stats: {
            likes: 0,
            retweets: 0,
            replies: 0,
            quotes: 0,
            bookmarks: 0
          },
          media: []
        }
      };
      return this.send(testPayload, { url, headers, timeout: 5e3 });
    }
  }
  const BASE_RETRY_DELAY = 1e3;
  const MAX_RETRY_DELAY = 5 * 60 * 1e3;
  class WebhookQueue {
    constructor(sender) {
      __publicField(this, "queue", []);
      __publicField(this, "sender");
      __publicField(this, "isProcessing", false);
      __publicField(this, "timer", null);
      this.sender = sender;
    }
    /**
     * Adds an item to the retry queue.
     *
     * @param item - The retry queue item
     */
    add(item) {
      const delay = this.calculateDelay(item.retryCount);
      const nextRetryAt = Date.now() + delay;
      this.queue.push({
        ...item,
        nextRetryAt
      });
      logger.debug(`Added webhook to retry queue, next retry in ${delay}ms`);
      this.scheduleProcessing();
    }
    /**
     * Starts processing the retry queue.
     */
    start() {
      this.scheduleProcessing();
    }
    /**
     * Stops processing the retry queue.
     */
    stop() {
      if (this.timer) {
        clearTimeout(this.timer);
        this.timer = null;
      }
      this.isProcessing = false;
    }
    /**
     * Gets the current queue size.
     */
    get size() {
      return this.queue.length;
    }
    /**
     * Calculates the delay for a retry attempt using exponential backoff.
     *
     * @param retryCount - Current retry count
     * @returns Delay in milliseconds
     */
    calculateDelay(retryCount) {
      const delay = BASE_RETRY_DELAY * Math.pow(2, retryCount);
      return Math.min(delay, MAX_RETRY_DELAY);
    }
    /**
     * Schedules the next queue processing.
     */
    scheduleProcessing() {
      if (this.isProcessing || this.queue.length === 0) {
        return;
      }
      const now = Date.now();
      const nextItem = this.queue.reduce((earliest, item) => {
        if (!earliest || item.nextRetryAt < earliest.nextRetryAt) {
          return item;
        }
        return earliest;
      }, null);
      if (!nextItem) {
        return;
      }
      const delay = Math.max(0, nextItem.nextRetryAt - now);
      if (this.timer) {
        clearTimeout(this.timer);
      }
      this.timer = setTimeout(() => {
        this.processQueue();
      }, delay);
    }
    /**
     * Processes items in the queue that are ready for retry.
     */
    async processQueue() {
      if (this.isProcessing) {
        return;
      }
      this.isProcessing = true;
      const now = Date.now();
      const readyItems = this.queue.filter((item) => item.nextRetryAt <= now);
      for (const item of readyItems) {
        const index = this.queue.indexOf(item);
        if (index > -1) {
          this.queue.splice(index, 1);
        }
        const result = await this.sender.send(item.payload, {
          url: item.url,
          headers: item.headers
        });
        if (result.success) {
          await databaseManager.updateWebhookLog(item.logId, {
            status: "success",
            response_status: result.status,
            retry_count: item.retryCount + 1
          });
          logger.info(`Webhook retry succeeded for ${item.logId}`);
        } else {
          if (item.retryCount + 1 < item.maxRetries) {
            this.add({
              ...item,
              retryCount: item.retryCount + 1
            });
          } else {
            await databaseManager.updateWebhookLog(item.logId, {
              status: "failed",
              error_message: result.error,
              retry_count: item.retryCount + 1
            });
            logger.warn(`Webhook failed after ${item.maxRetries} retries: ${item.logId}`);
          }
        }
      }
      this.isProcessing = false;
      this.scheduleProcessing();
    }
  }
  class WebhookManager {
    constructor() {
      __publicField(this, "sender");
      __publicField(this, "queue");
      __publicField(this, "configs", []);
      /**
       * Signal for subscribing to webhook config changes.
       */
      __publicField(this, "signal", new signals.Signal(0));
      this.sender = new WebhookSender();
      this.queue = new WebhookQueue(this.sender);
      this.loadConfigs();
    }
    /**
     * Loads webhook configurations from the database.
     */
    async loadConfigs() {
      const configs = await databaseManager.getWebhookConfigs();
      this.configs = configs || [];
      this.signal.value++;
      logger.debug(`Loaded ${this.configs.length} webhook configs`);
    }
    /**
     * Gets all webhook configurations.
     */
    getConfigs() {
      return this.configs;
    }
    /**
     * Gets enabled webhook configurations for a specific event type.
     *
     * @param eventType - The event type to filter by
     * @returns Array of matching webhook configs
     */
    getConfigsForEvent(eventType) {
      return this.configs.filter(
        (config) => config.enabled && config.events.includes(eventType)
      );
    }
    /**
     * Adds a new webhook configuration.
     *
     * @param config - The webhook configuration to add
     */
    async addConfig(config) {
      await databaseManager.addWebhookConfig(config);
      await this.loadConfigs();
      logger.info(`Added webhook config: ${config.name}`);
    }
    /**
     * Updates a webhook configuration.
     *
     * @param id - Config ID
     * @param updates - Partial updates to apply
     */
    async updateConfig(id, updates) {
      await databaseManager.updateWebhookConfig(id, updates);
      await this.loadConfigs();
      logger.info(`Updated webhook config: ${id}`);
    }
    /**
     * Deletes a webhook configuration.
     *
     * @param id - Config ID to delete
     */
    async deleteConfig(id) {
      await databaseManager.deleteWebhookConfig(id);
      await this.loadConfigs();
      logger.info(`Deleted webhook config: ${id}`);
    }
    /**
     * Tests a webhook configuration.
     *
     * @param url - Webhook URL to test
     * @param headers - Custom headers
     * @returns Test result
     */
    async testWebhook(url, headers = {}) {
      const result = await this.sender.test(url, headers);
      return {
        success: result.success,
        message: result.success ? `Webhook test successful (HTTP ${result.status})` : result.error || "Unknown error"
      };
    }
    /**
     * Triggers webhooks for a specific event and tweet.
     *
     * @param eventType - The event type
     * @param tweet - The tweet data
     */
    async triggerWebhooks(eventType, tweet) {
      const configs = this.getConfigsForEvent(eventType);
      if (configs.length === 0) {
        return;
      }
      const payload = formatWebhookPayload(tweet, eventType);
      for (const config of configs) {
        await this.sendWebhook(config, payload, tweet.rest_id, eventType);
      }
    }
    /**
     * Triggers webhooks for multiple tweets (batch operation).
     *
     * @param eventType - The event type
     * @param tweets - Array of tweet data
     */
    async triggerWebhooksBatch(eventType, tweets) {
      const configs = this.getConfigsForEvent(eventType);
      if (configs.length === 0) {
        return;
      }
      for (const tweet of tweets) {
        const payload = formatWebhookPayload(tweet, eventType);
        for (const config of configs) {
          await this.sendWebhook(config, payload, tweet.rest_id, eventType);
        }
      }
    }
    /**
     * Sends a webhook and logs the result.
     */
    async sendWebhook(config, payload, tweetId, eventType) {
      const timestamp = Date.now();
      const logId = generateWebhookLogId(eventType, tweetId, timestamp);
      const log = {
        id: logId,
        event_type: eventType,
        tweet_id: tweetId,
        webhook_url: config.url,
        status: "pending",
        request_payload: JSON.stringify(payload),
        created_at: timestamp,
        retry_count: 0
      };
      await databaseManager.addWebhookLog(log);
      const result = await this.sender.send(payload, {
        url: config.url,
        headers: config.headers
      });
      if (result.success) {
        await databaseManager.updateWebhookLog(logId, {
          status: "success",
          response_status: result.status
        });
        logger.debug(`Webhook sent successfully: ${eventType} for tweet ${tweetId}`);
      } else {
        if (config.retry_on_failure && config.max_retries > 0) {
          this.queue.add({
            logId,
            configId: config.id,
            url: config.url,
            headers: config.headers,
            payload,
            retryCount: 0,
            maxRetries: config.max_retries
          });
          logger.debug(`Webhook failed, added to retry queue: ${logId}`);
        } else {
          await databaseManager.updateWebhookLog(logId, {
            status: "failed",
            error_message: result.error
          });
          logger.warn(`Webhook failed: ${result.error}`);
        }
      }
    }
    /**
     * Starts the retry queue processor.
     */
    startQueue() {
      this.queue.start();
    }
    /**
     * Stops the retry queue processor.
     */
    stopQueue() {
      this.queue.stop();
    }
  }
  const webhookManager = new WebhookManager();
  function Settings() {
    const { t, i18n } = useTranslation();
    const currentTheme = signals.useSignal(appOptionsManager.get("theme"));
    const [showSettings, toggleSettings] = useToggle(false);
    const webhookConfigs = signals.useSignal([]);
    const showWebhookForm = signals.useSignal(false);
    const editingWebhook = signals.useSignal(null);
    const testResult = signals.useSignal(null);
    const formName = signals.useSignal("");
    const formUrl = signals.useSignal("");
    const formEnabled = signals.useSignal(true);
    const formEvents = signals.useSignal(["like", "bookmark"]);
    const formHeaders = signals.useSignal("");
    const formRetryOnFailure = signals.useSignal(true);
    const formMaxRetries = signals.useSignal(3);
    const styles = {
      subtitle: "mb-2 text-base-content ml-4 opacity-50 font-semibold text-xs",
      block: "text-sm mb-2 w-full flex px-4 py-2 text-base-content bg-base-200 rounded-box justify-between",
      item: "label cursor-pointer flex justify-between h-8 items-center p-0"
    };
    const loadWebhooks = async () => {
      await webhookManager.loadConfigs();
      webhookConfigs.value = webhookManager.getConfigs();
    };
    const resetWebhookForm = () => {
      formName.value = "";
      formUrl.value = "";
      formEnabled.value = true;
      formEvents.value = ["like", "bookmark"];
      formHeaders.value = "";
      formRetryOnFailure.value = true;
      formMaxRetries.value = 3;
      editingWebhook.value = null;
      testResult.value = null;
    };
    const openAddWebhook = () => {
      resetWebhookForm();
      showWebhookForm.value = true;
    };
    const openEditWebhook = (config) => {
      formName.value = config.name;
      formUrl.value = config.url;
      formEnabled.value = config.enabled;
      formEvents.value = [...config.events];
      formHeaders.value = Object.keys(config.headers).length > 0 ? JSON.stringify(config.headers, null, 2) : "";
      formRetryOnFailure.value = config.retry_on_failure;
      formMaxRetries.value = config.max_retries;
      editingWebhook.value = config;
      showWebhookForm.value = true;
    };
    const parseHeaders = () => {
      try {
        return formHeaders.value ? JSON.parse(formHeaders.value) : {};
      } catch {
        return {};
      }
    };
    const saveWebhook = async () => {
      const headers = parseHeaders();
      const now = Date.now();
      if (editingWebhook.value) {
        await webhookManager.updateConfig(editingWebhook.value.id, {
          name: formName.value,
          url: formUrl.value,
          enabled: formEnabled.value,
          events: formEvents.value,
          headers,
          retry_on_failure: formRetryOnFailure.value,
          max_retries: formMaxRetries.value,
          updated_at: now
        });
      } else {
        const config = {
          id: generateWebhookConfigId(),
          name: formName.value,
          url: formUrl.value,
          enabled: formEnabled.value,
          events: formEvents.value,
          headers,
          retry_on_failure: formRetryOnFailure.value,
          max_retries: formMaxRetries.value,
          created_at: now,
          updated_at: now
        };
        await webhookManager.addConfig(config);
      }
      showWebhookForm.value = false;
      resetWebhookForm();
      await loadWebhooks();
    };
    const deleteWebhook = async (id) => {
      if (confirm("Are you sure you want to delete this webhook?")) {
        await webhookManager.deleteConfig(id);
        await loadWebhooks();
      }
    };
    const toggleEvent = (event) => {
      const current = formEvents.value;
      if (current.includes(event)) {
        formEvents.value = current.filter((e) => e !== event);
      } else {
        formEvents.value = [...current, event];
      }
    };
    const testWebhook = async () => {
      testResult.value = null;
      const result = await webhookManager.testWebhook(formUrl.value, parseHeaders());
      testResult.value = result;
    };
    hooks.useEffect(() => {
      _GM_registerMenuCommand(`${t("Version")} ${packageJson.version}`, () => {
        window.open(packageJson.homepage, "_blank");
      });
      loadWebhooks();
    }, []);
    return /* @__PURE__ */ u(preact.Fragment, { children: [
      /* @__PURE__ */ u(
        "div",
        {
          onClick: toggleSettings,
          class: "w-9 h-9 mr-2 cursor-pointer flex justify-center items-center transition-colors duration-200 rounded-full hover:bg-base-200",
          children: /* @__PURE__ */ u(IconSettings, {})
        }
      ),
      /* @__PURE__ */ u(Modal, { title: t("Settings"), show: showSettings, onClose: toggleSettings, class: "max-w-lg", children: [
        /* @__PURE__ */ u("p", { class: styles.subtitle, children: t("General") }),
        /* @__PURE__ */ u("div", { class: cx(styles.block, "flex-col"), children: [
          /* @__PURE__ */ u("label", { class: styles.item, children: [
            /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Theme") }),
            /* @__PURE__ */ u(
              "select",
              {
                class: "select select-xs",
                onChange: (e) => {
                  var _a;
                  currentTheme.value = ((_a = e.target) == null ? void 0 : _a.value) ?? DEFAULT_APP_OPTIONS.theme;
                  appOptionsManager.set("theme", currentTheme.value);
                },
                children: THEMES.map((theme) => /* @__PURE__ */ u("option", { value: theme, selected: currentTheme.value === theme, children: capitalizeFirstLetter(theme) }, theme))
              }
            )
          ] }),
          /* @__PURE__ */ u("label", { class: styles.item, children: [
            /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Language") }),
            /* @__PURE__ */ u(
              "select",
              {
                class: "select select-xs",
                onChange: (e) => {
                  var _a;
                  const language = ((_a = e.target) == null ? void 0 : _a.value) ?? detectBrowserLanguage();
                  i18n.changeLanguage(language);
                  appOptionsManager.set("language", language);
                },
                children: Object.entries(LANGUAGES_CONFIG).map(([langTag, langConf]) => /* @__PURE__ */ u(
                  "option",
                  {
                    value: langTag,
                    selected: appOptionsManager.get("language") === langTag,
                    children: [
                      langConf.nameEn,
                      " - ",
                      langConf.name
                    ]
                  },
                  langTag
                ))
              }
            )
          ] }),
          /* @__PURE__ */ u("label", { class: styles.item, children: [
            /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Debug") }),
            /* @__PURE__ */ u(
              "input",
              {
                type: "checkbox",
                class: "toggle toggle-primary",
                checked: appOptionsManager.get("debug"),
                onChange: (e) => {
                  var _a;
                  appOptionsManager.set("debug", (_a = e.target) == null ? void 0 : _a.checked);
                }
              }
            )
          ] }),
          /* @__PURE__ */ u("label", { class: styles.item, children: [
            /* @__PURE__ */ u("div", { class: "flex items-center", children: [
              /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Date Time Format") }),
              /* @__PURE__ */ u(
                "a",
                {
                  href: "https://day.js.org/docs/en/display/format",
                  target: "_blank",
                  rel: "noopener noreferrer",
                  class: "tooltip tooltip-bottom ml-0.5 before:max-w-40",
                  "data-tip": t(
                    "Click for more information. This will take effect on both previewer and exported files."
                  ),
                  children: /* @__PURE__ */ u(IconHelp, { size: 20 })
                }
              )
            ] }),
            /* @__PURE__ */ u(
              "input",
              {
                type: "text",
                class: "input input-bordered input-xs w-48",
                value: appOptionsManager.get("dateTimeFormat"),
                onChange: (e) => {
                  var _a;
                  appOptionsManager.set("dateTimeFormat", (_a = e.target) == null ? void 0 : _a.value);
                }
              }
            )
          ] }),
          /* @__PURE__ */ u("label", { class: styles.item, children: [
            /* @__PURE__ */ u("div", { class: "flex items-center", children: [
              /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Use dedicated DB for accounts") }),
              /* @__PURE__ */ u(
                "a",
                {
                  class: "tooltip tooltip-bottom ml-0.5 before:max-w-40",
                  "data-tip": t(
                    "This will create separate database for each Twitter account, which can help reduce the chance of data mixing when you use multiple accounts."
                  ),
                  children: /* @__PURE__ */ u(IconHelp, { size: 20 })
                }
              )
            ] }),
            /* @__PURE__ */ u(
              "input",
              {
                type: "checkbox",
                class: "toggle toggle-primary",
                checked: appOptionsManager.get("dedicatedDbForAccounts"),
                onChange: (e) => {
                  var _a;
                  appOptionsManager.set("dedicatedDbForAccounts", (_a = e.target) == null ? void 0 : _a.checked);
                }
              }
            )
          ] }),
          /* @__PURE__ */ u("label", { class: styles.item, children: [
            /* @__PURE__ */ u("div", { class: "flex items-center", children: [
              /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Filter promoted tweets") }),
              /* @__PURE__ */ u(
                "a",
                {
                  class: "tooltip tooltip-bottom ml-0.5 before:max-w-40",
                  "data-tip": t(
                    "Automatically filter out promoted/advertisement tweets from being saved to the database."
                  ),
                  children: /* @__PURE__ */ u(IconHelp, { size: 20 })
                }
              )
            ] }),
            /* @__PURE__ */ u(
              "input",
              {
                type: "checkbox",
                class: "toggle toggle-primary",
                checked: appOptionsManager.get("filterPromotedTweets"),
                onChange: (e) => {
                  var _a;
                  appOptionsManager.set("filterPromotedTweets", (_a = e.target) == null ? void 0 : _a.checked);
                }
              }
            )
          ] }),
          /* @__PURE__ */ u("label", { class: styles.item, children: [
            /* @__PURE__ */ u("div", { class: "flex items-center", children: [
              /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Filter promotional tweets") }),
              /* @__PURE__ */ u(
                "a",
                {
                  class: "tooltip tooltip-bottom ml-0.5 before:max-w-40",
                  "data-tip": t(
                    "Filter tweets from business accounts and amplify videos."
                  ),
                  children: /* @__PURE__ */ u(IconHelp, { size: 20 })
                }
              )
            ] }),
            /* @__PURE__ */ u(
              "input",
              {
                type: "checkbox",
                class: "toggle toggle-primary",
                checked: appOptionsManager.get("filterPromotionalTweets"),
                onChange: (e) => {
                  var _a;
                  appOptionsManager.set("filterPromotionalTweets", (_a = e.target) == null ? void 0 : _a.checked);
                }
              }
            )
          ] }),
          /* @__PURE__ */ u("div", { class: styles.item, children: [
            /* @__PURE__ */ u("div", { class: "flex items-center", children: /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: t("Local Database") }) }),
            /* @__PURE__ */ u("div", { class: "flex", children: [
              /* @__PURE__ */ u(
                "button",
                {
                  class: "btn btn-xs btn-neutral mr-2",
                  onClick: async () => {
                    let storageUsageText = "Storage usage: N/A";
                    if (typeof navigator.storage.estimate === "function") {
                      const { quota = 1, usage = 0 } = await navigator.storage.estimate();
                      const usageMB = (usage / 1024 / 1024).toFixed(2);
                      const quotaMB = (quota / 1024 / 1024).toFixed(2);
                      storageUsageText = `Storage usage: ${usageMB}MB / ${quotaMB}MB`;
                    }
                    const count = await databaseManager.count();
                    alert(
                      storageUsageText + "\n\nIndexedDB tables count:\n" + JSON.stringify(count, void 0, "  ")
                    );
                  },
                  children: [
                    /* @__PURE__ */ u(IconReportAnalytics, { size: 20 }),
                    t("Analyze DB")
                  ]
                }
              ),
              /* @__PURE__ */ u(
                "button",
                {
                  class: "btn btn-xs btn-primary mr-2",
                  onClick: async () => {
                    const blob = await databaseManager.export();
                    if (blob) {
                      saveFile(`memoizex-${Date.now()}.json`, blob);
                    }
                  },
                  children: [
                    /* @__PURE__ */ u(IconDatabaseExport, { size: 20 }),
                    t("Export DB")
                  ]
                }
              ),
              /* @__PURE__ */ u(
                "button",
                {
                  class: "btn btn-xs btn-warning",
                  onClick: async () => {
                    if (confirm(t("Are you sure to clear all data in the database?"))) {
                      await databaseManager.clear();
                    }
                  },
                  children: [
                    /* @__PURE__ */ u(IconTrashX, { size: 20 }),
                    t("Clear DB")
                  ]
                }
              )
            ] })
          ] })
        ] }),
        /* @__PURE__ */ u("p", { class: styles.subtitle, children: /* @__PURE__ */ u("span", { class: "flex items-center gap-2", children: [
          /* @__PURE__ */ u(IconWebhook, { size: 16 }),
          "Webhook Configurations"
        ] }) }),
        /* @__PURE__ */ u("div", { class: cx(styles.block, "flex-col"), children: [
          webhookConfigs.value.length === 0 && !showWebhookForm.value && /* @__PURE__ */ u("div", { class: "text-center text-base-content/60 py-2 text-xs", children: "No webhooks configured" }),
          !showWebhookForm.value && webhookConfigs.value.map((config) => /* @__PURE__ */ u("div", { class: "flex items-center justify-between py-1 border-b border-base-300 last:border-0", children: [
            /* @__PURE__ */ u("div", { class: "flex flex-col", children: [
              /* @__PURE__ */ u("div", { class: "flex items-center gap-2", children: [
                /* @__PURE__ */ u("span", { class: "text-sm font-medium", children: config.name }),
                /* @__PURE__ */ u("span", { class: `badge badge-xs ${config.enabled ? "badge-success" : "badge-ghost"}`, children: config.enabled ? "On" : "Off" })
              ] }),
              /* @__PURE__ */ u("span", { class: "text-xs text-base-content/60 font-mono truncate max-w-[200px]", children: config.url })
            ] }),
            /* @__PURE__ */ u("div", { class: "flex gap-1", children: [
              /* @__PURE__ */ u("button", { class: "btn btn-xs btn-ghost", onClick: () => openEditWebhook(config), children: /* @__PURE__ */ u(IconEdit, { size: 14 }) }),
              /* @__PURE__ */ u("button", { class: "btn btn-xs btn-ghost text-error", onClick: () => deleteWebhook(config.id), children: /* @__PURE__ */ u(IconTrashX, { size: 14 }) })
            ] })
          ] }, config.id)),
          showWebhookForm.value ? /* @__PURE__ */ u("div", { class: "flex flex-col gap-2 py-2", children: [
            /* @__PURE__ */ u(
              "input",
              {
                type: "text",
                class: "input input-xs input-bordered w-full",
                placeholder: "Webhook name",
                value: formName.value,
                onInput: (e) => formName.value = e.target.value
              }
            ),
            /* @__PURE__ */ u("div", { class: "flex gap-1", children: [
              /* @__PURE__ */ u(
                "input",
                {
                  type: "url",
                  class: "input input-xs input-bordered flex-1",
                  placeholder: "https://example.com/webhook",
                  value: formUrl.value,
                  onInput: (e) => formUrl.value = e.target.value
                }
              ),
              /* @__PURE__ */ u(
                "button",
                {
                  class: "btn btn-xs btn-ghost",
                  onClick: testWebhook,
                  disabled: !formUrl.value,
                  children: /* @__PURE__ */ u(IconPlayerPlay, { size: 14 })
                }
              )
            ] }),
            testResult.value && /* @__PURE__ */ u("div", { class: `text-xs flex items-center gap-1 ${testResult.value.success ? "text-success" : "text-error"}`, children: [
              testResult.value.success ? /* @__PURE__ */ u(IconCheck, { size: 12 }) : /* @__PURE__ */ u(IconX, { size: 12 }),
              testResult.value.message
            ] }),
            /* @__PURE__ */ u("div", { class: "flex gap-2 flex-wrap", children: [
              ["like", "bookmark", "view"].map((event) => /* @__PURE__ */ u("label", { class: "label cursor-pointer gap-1 p-0", children: [
                /* @__PURE__ */ u(
                  "input",
                  {
                    type: "checkbox",
                    class: "checkbox checkbox-xs",
                    checked: formEvents.value.includes(event),
                    onChange: () => toggleEvent(event)
                  }
                ),
                /* @__PURE__ */ u("span", { class: "label-text text-xs", children: event })
              ] }, event)),
              /* @__PURE__ */ u("label", { class: "label cursor-pointer gap-1 p-0", children: [
                /* @__PURE__ */ u(
                  "input",
                  {
                    type: "checkbox",
                    class: "checkbox checkbox-xs",
                    checked: formEnabled.value,
                    onChange: (e) => formEnabled.value = e.target.checked
                  }
                ),
                /* @__PURE__ */ u("span", { class: "label-text text-xs", children: "enabled" })
              ] })
            ] }),
            /* @__PURE__ */ u(
              "textarea",
              {
                class: "textarea textarea-bordered textarea-xs w-full",
                rows: 2,
                placeholder: 'Headers JSON: {"Authorization": "Bearer ..."}',
                value: formHeaders.value,
                onInput: (e) => formHeaders.value = e.target.value
              }
            ),
            /* @__PURE__ */ u("div", { class: "flex justify-end gap-2", children: [
              /* @__PURE__ */ u("button", { class: "btn btn-xs btn-ghost", onClick: () => {
                showWebhookForm.value = false;
                resetWebhookForm();
              }, children: "Cancel" }),
              /* @__PURE__ */ u(
                "button",
                {
                  class: "btn btn-xs btn-primary",
                  onClick: saveWebhook,
                  disabled: !formName.value || !formUrl.value,
                  children: "Save"
                }
              )
            ] })
          ] }) : /* @__PURE__ */ u("button", { class: "btn btn-xs btn-ghost mt-2", onClick: openAddWebhook, children: [
            /* @__PURE__ */ u(IconPlus, { size: 14 }),
            "Add Webhook"
          ] })
        ] }),
        /* @__PURE__ */ u("p", { class: styles.subtitle, children: t("Modules (Scroll to see more)") }),
        /* @__PURE__ */ u("div", { class: cx(styles.block, "flex-col", "max-h-44 overflow-scroll"), children: extensionManager.getExtensions().map((extension) => /* @__PURE__ */ u("label", { class: cx(styles.item, "flex-shrink-0"), children: [
          /* @__PURE__ */ u("span", { children: [
            t(extension.name.replace("Module", "")),
            " ",
            t("Module")
          ] }),
          /* @__PURE__ */ u(
            "input",
            {
              type: "checkbox",
              class: "toggle toggle-secondary",
              checked: extension.enabled,
              onChange: () => {
                if (extension.enabled) {
                  extensionManager.disable(extension.name);
                } else {
                  extensionManager.enable(extension.name);
                }
              }
            }
          )
        ] }, extension.name)) }),
        /* @__PURE__ */ u("p", { class: styles.subtitle, children: t("About") }),
        /* @__PURE__ */ u("div", { class: styles.block, children: [
          /* @__PURE__ */ u("span", { class: "label-text whitespace-nowrap", children: [
            t("Version"),
            " ",
            packageJson.version
          ] }),
          /* @__PURE__ */ u("a", { class: "btn btn-xs btn-ghost", target: "_blank", href: packageJson.homepage, children: [
            /* @__PURE__ */ u(IconBrandGithubFilled, { class: "[&>path]:stroke-0" }),
            "GitHub"
          ] })
        ] })
      ] })
    ] });
  }
  function App() {
    const { t } = useTranslation();
    const extensions = signals.useSignal([]);
    const currentTheme = signals.useSignal(appOptionsManager.get("theme"));
    const showControlPanel = signals.useSignal(appOptionsManager.get("showControlPanel"));
    const toggleControlPanel = () => {
      showControlPanel.value = !showControlPanel.value;
      appOptionsManager.set("showControlPanel", showControlPanel.value);
    };
    hooks.useEffect(() => {
      extensionManager.signal.subscribe(() => {
        extensions.value = extensionManager.getExtensions();
      });
      appOptionsManager.signal.subscribe(() => {
        currentTheme.value = appOptionsManager.get("theme");
      });
      _GM_registerMenuCommand(t("Open Control Panel"), toggleControlPanel);
      logger.debug("App useEffect executed");
    }, []);
    return /* @__PURE__ */ u(preact.Fragment, { children: [
      /* @__PURE__ */ u(
        "div",
        {
          onClick: toggleControlPanel,
          "data-theme": currentTheme.value,
          class: "group w-12 h-12 fixed top-[60%] left-[-20px] cursor-pointer bg-transparent fill-base-content",
          children: /* @__PURE__ */ u("div", { class: "w-full h-full origin origin-[bottom_center] transition-all duration-200 group-hover:translate-x-[5px] group-hover:rotate-[20deg] opacity-50 group-hover:opacity-90", children: /* @__PURE__ */ u(CatIcon, {}) })
        }
      ),
      /* @__PURE__ */ u(
        "section",
        {
          "data-theme": currentTheme.value,
          class: cx(
            "card card-compact bg-base-100 fixed border shadow-xl w-80 leading-loose text-base-content px-4 py-3 rounded-box border-solid border-neutral-content border-opacity-50 left-8 top-8 transition-transform duration-500",
            showControlPanel.value ? "translate-x-0 transform-none" : "translate-x-[-500px]"
          ),
          children: [
            /* @__PURE__ */ u("header", { class: "flex items-center h-9", children: [
              /* @__PURE__ */ u(IconBrandTwitterFilled, { class: "mr-2" }),
              /* @__PURE__ */ u("h2", { class: "font-semibold leading-none text-xl m-0 flex-grow", children: "MemoizeX" }),
              /* @__PURE__ */ u(ErrorBoundary, { children: /* @__PURE__ */ u(Settings, {}) }),
              /* @__PURE__ */ u(
                "div",
                {
                  onClick: toggleControlPanel,
                  class: "w-9 h-9 cursor-pointer flex justify-center items-center transition-colors duration-200 rounded-full hover:bg-base-200",
                  children: /* @__PURE__ */ u(IconX, {})
                }
              )
            ] }),
            /* @__PURE__ */ u("p", { class: "text-sm text-base-content text-opacity-70 mb-1 leading-none", children: t("Browse around to capture more data.") }),
            /* @__PURE__ */ u("div", { class: "divider mt-0 mb-0" }),
            /* @__PURE__ */ u("main", { children: extensions.value.map((ext) => {
              const Component2 = ext.render();
              if (ext.enabled && Component2) {
                return /* @__PURE__ */ u(ErrorBoundary, { children: /* @__PURE__ */ u(Component2, { extension: ext }, ext.name) });
              }
              return null;
            }) })
          ]
        }
      )
    ] });
  }
  /**
   * This file zip-stream.ts is copied from StreamSaver.js.
   *
   * @see https://github.com/jimmywarting/StreamSaver.js/blob/master/examples/zip-stream.js
   * @license MIT
   */
  class Crc32 {
    constructor() {
      this.crc = -1;
    }
    append(data) {
      var crc = this.crc | 0;
      var table = this.table;
      for (var offset = 0, len = data.length | 0; offset < len; offset++) {
        crc = crc >>> 8 ^ table[(crc ^ data[offset]) & 255];
      }
      this.crc = crc;
    }
    get() {
      return ~this.crc;
    }
  }
  Crc32.prototype.table = (() => {
    var i;
    var j;
    var t;
    var table = [];
    for (i = 0; i < 256; i++) {
      t = i;
      for (j = 0; j < 8; j++) {
        t = t & 1 ? t >>> 1 ^ 3988292384 : t >>> 1;
      }
      table[i] = t;
    }
    return table;
  })();
  const getDataHelper = (byteLength) => {
    var uint8 = new Uint8Array(byteLength);
    return {
      array: uint8,
      view: new DataView(uint8.buffer)
    };
  };
  const pump = (zipObj) => zipObj.reader.read().then((chunk) => {
    if (chunk.done) return zipObj.writeFooter();
    const outputData = chunk.value;
    zipObj.crc.append(outputData);
    zipObj.uncompressedLength += outputData.length;
    zipObj.compressedLength += outputData.length;
    zipObj.ctrl.enqueue(outputData);
  });
  function createWriter(underlyingSource) {
    const files = /* @__PURE__ */ Object.create(null);
    const filenames = [];
    const encoder = new TextEncoder();
    let offset = 0;
    let activeZipIndex = 0;
    let ctrl;
    let activeZipObject, closed;
    function next() {
      activeZipIndex++;
      activeZipObject = files[filenames[activeZipIndex]];
      if (activeZipObject) processNextChunk();
      else if (closed) closeZip();
    }
    var zipWriter = {
      enqueue(fileLike) {
        if (closed)
          throw new TypeError(
            "Cannot enqueue a chunk into a readable stream that is closed or has been requested to be closed"
          );
        let name2 = fileLike.name.trim();
        const date = new Date(
          typeof fileLike.lastModified === "undefined" ? Date.now() : fileLike.lastModified
        );
        if (fileLike.directory && !name2.endsWith("/")) name2 += "/";
        if (files[name2]) throw new Error("File already exists.");
        const nameBuf = encoder.encode(name2);
        filenames.push(name2);
        const zipObject = files[name2] = {
          level: 0,
          ctrl,
          directory: !!fileLike.directory,
          nameBuf,
          comment: encoder.encode(fileLike.comment || ""),
          compressedLength: 0,
          uncompressedLength: 0,
          writeHeader() {
            var header = getDataHelper(26);
            var data = getDataHelper(30 + nameBuf.length);
            zipObject.offset = offset;
            zipObject.header = header;
            if (zipObject.level !== 0 && !zipObject.directory) {
              header.view.setUint16(4, 2048);
            }
            header.view.setUint32(0, 335546376);
            header.view.setUint16(
              6,
              (date.getHours() << 6 | date.getMinutes()) << 5 | date.getSeconds() / 2,
              true
            );
            header.view.setUint16(
              8,
              (date.getFullYear() - 1980 << 4 | date.getMonth() + 1) << 5 | date.getDate(),
              true
            );
            header.view.setUint16(22, nameBuf.length, true);
            data.view.setUint32(0, 1347093252);
            data.array.set(header.array, 4);
            data.array.set(nameBuf, 30);
            offset += data.array.length;
            ctrl.enqueue(data.array);
          },
          writeFooter() {
            var footer = getDataHelper(16);
            footer.view.setUint32(0, 1347094280);
            if (zipObject.crc) {
              zipObject.header.view.setUint32(10, zipObject.crc.get(), true);
              zipObject.header.view.setUint32(14, zipObject.compressedLength, true);
              zipObject.header.view.setUint32(18, zipObject.uncompressedLength, true);
              footer.view.setUint32(4, zipObject.crc.get(), true);
              footer.view.setUint32(8, zipObject.compressedLength, true);
              footer.view.setUint32(12, zipObject.uncompressedLength, true);
            }
            ctrl.enqueue(footer.array);
            offset += zipObject.compressedLength + 16;
            next();
          },
          fileLike
        };
        if (!activeZipObject) {
          activeZipObject = zipObject;
          processNextChunk();
        }
      },
      close() {
        if (closed)
          throw new TypeError(
            "Cannot close a readable stream that has already been requested to be closed"
          );
        if (!activeZipObject) closeZip();
        closed = true;
      }
    };
    function closeZip() {
      var length = 0;
      var index = 0;
      var indexFilename, file;
      for (indexFilename = 0; indexFilename < filenames.length; indexFilename++) {
        file = files[filenames[indexFilename]];
        length += 46 + file.nameBuf.length + file.comment.length;
      }
      const data = getDataHelper(length + 22);
      for (indexFilename = 0; indexFilename < filenames.length; indexFilename++) {
        file = files[filenames[indexFilename]];
        data.view.setUint32(index, 1347092738);
        data.view.setUint16(index + 4, 5120);
        data.array.set(file.header.array, index + 6);
        data.view.setUint16(index + 32, file.comment.length, true);
        if (file.directory) {
          data.view.setUint8(index + 38, 16);
        }
        data.view.setUint32(index + 42, file.offset, true);
        data.array.set(file.nameBuf, index + 46);
        data.array.set(file.comment, index + 46 + file.nameBuf.length);
        index += 46 + file.nameBuf.length + file.comment.length;
      }
      data.view.setUint32(index, 1347093766);
      data.view.setUint16(index + 8, filenames.length, true);
      data.view.setUint16(index + 10, filenames.length, true);
      data.view.setUint32(index + 12, length, true);
      data.view.setUint32(index + 16, offset, true);
      ctrl.enqueue(data.array);
      ctrl.close();
    }
    function processNextChunk() {
      if (!activeZipObject) return;
      if (activeZipObject.directory)
        return activeZipObject.writeFooter(activeZipObject.writeHeader());
      if (activeZipObject.reader) return pump(activeZipObject);
      if (activeZipObject.fileLike.stream) {
        activeZipObject.crc = new Crc32();
        activeZipObject.reader = activeZipObject.fileLike.stream().getReader();
        activeZipObject.writeHeader();
      } else next();
    }
    return new ReadableStream({
      start: (c) => {
        ctrl = c;
        if (underlyingSource.start) Promise.resolve(underlyingSource.start(zipWriter));
      },
      pull() {
        return processNextChunk() || underlyingSource.pull && Promise.resolve(underlyingSource.pull(zipWriter));
      }
    });
  }
  async function zipStreamDownload(zipFilename, files, onProgress, rateLimit = 1e3) {
    let current = 0;
    const total = files.length;
    const fileIterator = files.values();
    const readableZipStream = createWriter({
      // eslint-disable-next-line @typescript-eslint/no-explicit-any
      async pull(ctrl) {
        const fileInfo = fileIterator.next();
        if (fileInfo.done) {
          ctrl.close();
        } else {
          const { filename, url } = fileInfo.value;
          const start = Date.now();
          logger.debug(`Start downloading ${filename} from ${url}`);
          return fetch(url).then((res) => {
            ctrl.enqueue({
              name: filename,
              stream: () => res.body
            });
            onProgress == null ? void 0 : onProgress(++current, total, fileInfo.value);
            logger.debug(`Finished downloading ${filename} in ${Date.now() - start}ms`);
          }).then(() => {
            return new Promise((resolve) => setTimeout(resolve, rateLimit));
          });
        }
      }
    });
    const chunks = [];
    const writableOutputStream = new WritableStream({
      write(chunk) {
        chunks.push(chunk);
      },
      close() {
        logger.info("Zip stream closed.");
      }
    });
    logger.info(`Exporting to ZIP file: ${zipFilename}`);
    await readableZipStream.pipeTo(writableOutputStream);
    const arrayBuffer = await new Blob(chunks).arrayBuffer();
    const blob = new Blob([arrayBuffer]);
    fileSaverEs.saveAs(blob, zipFilename);
  }
  const patterns = {
    id: {
      description: "The tweet ID",
      extractor: (tweet) => tweet.rest_id
    },
    screen_name: {
      description: "The username of tweet author",
      extractor: (tweet) => tweet.core.user_results.result.core.screen_name
    },
    name: {
      description: "The profile name of tweet author",
      extractor: (tweet) => tweet.core.user_results.result.core.name
    },
    index: {
      description: "The media index in tweet (start from 0)",
      extractor: (tweet, media) => String(getMediaIndex(tweet, media))
    },
    num: {
      description: "The order of media in tweet (1/2/3/4)",
      extractor: (tweet, media) => String(getMediaIndex(tweet, media) + 1)
    },
    date: {
      description: "The post date in YYYYMMDD format",
      extractor: (tweet) => parseTwitterDateTime(tweet.legacy.created_at).format("YYYYMMDD")
    },
    time: {
      description: "The post time in HHmmss format",
      extractor: (tweet) => parseTwitterDateTime(tweet.legacy.created_at).format("HHmmss")
    },
    type: {
      description: "The media type (photo/video/animated_gif)",
      extractor: (tweet, media) => media.type
    },
    ext: {
      description: "The file extension of media (jpg/png/mp4)",
      extractor: (tweet, media) => getFileExtensionFromUrl(getMediaOriginalUrl(media))
    }
  };
  const DEFAULT_MEDIA_TYPES = ["photo", "video", "animated_gif"];
  function extractMedia(data, includeRetweets, filenamePattern) {
    const gallery = /* @__PURE__ */ new Map();
    for (const item of data) {
      if (item.__typename === "Tweet" || typeof item.__typename === "undefined" && "core" in item) {
        if (!includeRetweets && item.legacy.retweeted_status_result) {
          continue;
        }
        const tweetMedia = extractTweetMedia(item).map((media) => {
          let filename = filenamePattern;
          for (const [key, value] of Object.entries(patterns)) {
            filename = filename.replace(`{${key}}`, value.extractor(item, media));
          }
          return { filename, type: media.type, url: getMediaOriginalUrl(media) };
        });
        for (const media of tweetMedia) {
          gallery.set(media.filename, media);
        }
      }
      if (item.__typename === "User") {
        if (item.avatar.image_url) {
          const ext = getFileExtensionFromUrl(item.avatar.image_url);
          const filename = `${item.core.screen_name}_profile_image.${ext}`;
          gallery.set(filename, {
            filename,
            type: "photo",
            url: getProfileImageOriginalUrl(item.avatar.image_url)
          });
        }
        if (item.legacy.profile_banner_url) {
          const ext = getFileExtensionFromUrl(item.legacy.profile_banner_url);
          const filename = `${item.core.screen_name}_profile_banner.${ext}`;
          gallery.set(filename, {
            filename,
            type: "photo",
            url: item.legacy.profile_banner_url
          });
        }
      }
    }
    return Array.from(gallery.values());
  }
  function ExportMediaModal({
    title,
    table,
    isTweet,
    show,
    onClose
  }) {
    const { t } = useTranslation("exporter");
    const [loading, setLoading] = useSignalState(false);
    const [copied, setCopied] = useSignalState(false);
    const [useAria2Format, toggleUseAria2Format] = useToggle(false);
    const [rateLimit, setRateLimit] = useSignalState(1e3);
    const [filenamePattern, setFilenamePattern] = useSignalState(appOptionsManager.get("filenamePattern"));
    const [currentProgress, setCurrentProgress] = useSignalState(0);
    const [totalProgress, setTotalProgress] = useSignalState(0);
    const taskStatusSignal = signals.useSignal({});
    const [filters, setFilters] = useSignalState([
      ...DEFAULT_MEDIA_TYPES,
      ...isTweet ? ["retweet"] : []
    ]);
    const includeRetweets = filters.includes("retweet");
    const mediaList = extractMedia(
      table.getSelectedRowModel().rows.map((row) => row.original),
      includeRetweets,
      filenamePattern ?? ""
    ).filter((media) => filters.includes(media.type));
    const onProgress = (current, total, value) => {
      setCurrentProgress(current);
      setTotalProgress(total);
      if (value == null ? void 0 : value.filename) {
        const updated = { ...taskStatusSignal.value, [value.filename]: 100 };
        taskStatusSignal.value = updated;
      }
    };
    const onExport = async () => {
      try {
        setLoading(true);
        await zipStreamDownload(`twitter-${title}-${Date.now()}-media.zip`, mediaList, onProgress);
        setLoading(false);
      } catch (err) {
        setLoading(false);
        logger.error(t("Failed to export media. Open DevTools for more details."), err);
      }
    };
    const onCopy = (saveAsFile = false) => {
      const text = mediaList.map((media) => useAria2Format ? `${media.url}
  out=${media.filename}` : media.url).join("\n");
      try {
        if (saveAsFile) {
          fileSaverEs.saveAs(new Blob([text], { type: "text/plain;charset=utf-8" }), "media-urls.txt");
          return;
        }
        navigator.clipboard.writeText(text);
        setCopied(true);
        setTimeout(() => setCopied(false), 2e3);
      } catch (err) {
        logger.error(t("Failed to copy media URLs. Open DevTools for more details."), err);
      }
    };
    return /* @__PURE__ */ u(
      Modal,
      {
        class: "max-w-sm md:max-w-screen-sm sm:max-w-screen-sm max-h-full",
        title: `${title} ${t("Media")}`,
        show,
        onClose,
        children: [
          /* @__PURE__ */ u("div", { class: "px-4 text-base overflow-y-scroll overscroll-none", children: [
            /* @__PURE__ */ u("p", { class: "text-base-content text-opacity-60 leading-5 text-sm", children: t(
              "Download and save media files from captured data. This may take a while depending on the amount of data. Media that will be downloaded includes: profile images, profile banners (for users), images, videos (for tweets)."
            ) }),
            /* @__PURE__ */ u("div", { role: "alert", class: "alert text-sm py-2 mt-2 mb-2 grid-cols-[auto_minmax(auto,1fr)]", children: [
              /* @__PURE__ */ u(IconInfoCircle, { size: 24 }),
              /* @__PURE__ */ u("span", { children: t(
                "For more than 100 media or large files, it is recommended to copy the URLs and download them with an external download manager such as aria2."
              ) })
            ] }),
            isTweet && /* @__PURE__ */ u("div", { class: "flex flex-wrap sm:grid grid-cols-4 sm:gap-2 items-center sm:h-9", children: [
              /* @__PURE__ */ u("p", { class: "leading-8", children: t("Filename template:") }),
              /* @__PURE__ */ u(
                "div",
                {
                  class: "tooltip tooltip-bottom col-span-3 before:whitespace-pre-line before:max-w-max",
                  "data-tip": Object.entries(patterns).map(([key, value]) => `{${key}} - ${t(value.description)}`).reduce((acc, cur) => acc + cur + "\n", ""),
                  children: /* @__PURE__ */ u(
                    "input",
                    {
                      type: "text",
                      class: "input input-bordered input-sm w-full",
                      value: filenamePattern,
                      onChange: (e) => {
                        var _a;
                        const value = (_a = e == null ? void 0 : e.target) == null ? void 0 : _a.value;
                        setFilenamePattern(value);
                        appOptionsManager.set("filenamePattern", value);
                      }
                    }
                  )
                }
              )
            ] }),
            /* @__PURE__ */ u("div", { class: "flex flex-wrap sm:grid grid-cols-4 sm:gap-2 items-center sm:h-9", children: [
              /* @__PURE__ */ u("p", { class: "leading-8 col-span-1 whitespace-nowrap", children: t("Rate limit (ms):") }),
              /* @__PURE__ */ u(
                "input",
                {
                  type: "number",
                  class: "input input-bordered input-sm col-span-1",
                  value: rateLimit,
                  onChange: (e) => {
                    var _a;
                    const value = parseInt((_a = e == null ? void 0 : e.target) == null ? void 0 : _a.value);
                    setRateLimit(value || 0);
                  }
                }
              ),
              /* @__PURE__ */ u("p", { class: "leading-8 col-span-1 whitespace-nowrap sm:pl-2", children: t("Use aria2 format:") }),
              /* @__PURE__ */ u("div", { class: "col-span-1 flex items-center", children: [
                /* @__PURE__ */ u(
                  "input",
                  {
                    type: "checkbox",
                    class: "toggle toggle-primary",
                    checked: useAria2Format,
                    onChange: toggleUseAria2Format
                  }
                ),
                /* @__PURE__ */ u(
                  "a",
                  {
                    href: "https://aria2.github.io/manual/en/html/aria2c.html#input-file",
                    target: "_blank",
                    rel: "noopener noreferrer",
                    class: "tooltip tooltip-bottom before:max-w-40 ml-1",
                    "data-tip": t(
                      "Click for more information. Each URL will be on a new line, with its filename on the next line. This format is compatible with aria2."
                    ),
                    children: /* @__PURE__ */ u(IconHelp, { size: 20 })
                  }
                )
              ] })
            ] }),
            /* @__PURE__ */ u("div", { class: "flex flex-wrap sm:grid grid-cols-4 sm:gap-2 items-center sm:h-9", children: [
              /* @__PURE__ */ u("p", { class: "leading-8", children: t("Media Filter:") }),
              /* @__PURE__ */ u(
                MultiSelect,
                {
                  class: "col-span-3",
                  options: [
                    { label: t("filter.photo"), value: "photo" },
                    { label: t("filter.video"), value: "video" },
                    { label: t("filter.animated_gif"), value: "animated_gif" },
                    ...isTweet ? [{ label: t("filter.retweet"), value: "retweet" }] : []
                  ],
                  selected: filters,
                  onChange: setFilters
                }
              )
            ] }),
            /* @__PURE__ */ u("div", { class: "my-3 overflow-x-scroll", children: [
              /* @__PURE__ */ u("table", { class: "table table-xs table-zebra", children: [
                /* @__PURE__ */ u("thead", { children: /* @__PURE__ */ u("tr", { children: [
                  /* @__PURE__ */ u("th", {}),
                  /* @__PURE__ */ u("th", { children: "#" }),
                  /* @__PURE__ */ u("th", { children: t("File Name") }),
                  /* @__PURE__ */ u("th", { children: t("Media Type") }),
                  /* @__PURE__ */ u("th", { children: t("Download URL") })
                ] }) }),
                /* @__PURE__ */ u("tbody", { children: mediaList.map((media, index) => /* @__PURE__ */ u("tr", { children: [
                  /* @__PURE__ */ u("td", { children: taskStatusSignal.value[media.filename] ? /* @__PURE__ */ u(IconCircleCheck, { class: "text-success", size: 14 }) : /* @__PURE__ */ u(IconCircleDashed, { size: 14 }) }),
                  /* @__PURE__ */ u("th", { children: index + 1 }),
                  /* @__PURE__ */ u("td", { children: media.filename }),
                  /* @__PURE__ */ u("td", { children: t(`filter.${media.type}`) }),
                  /* @__PURE__ */ u("td", { children: /* @__PURE__ */ u(
                    "a",
                    {
                      class: "link whitespace-nowrap",
                      href: media.url,
                      target: "_blank",
                      rel: "noopener noreferrer",
                      children: media.url
                    }
                  ) })
                ] })) })
              ] }),
              mediaList.length > 0 ? null : /* @__PURE__ */ u("div", { class: "flex items-center justify-center h-28 w-full", children: /* @__PURE__ */ u("p", { class: "text-base-content text-opacity-50", children: t("No media selected.") }) })
            ] }),
            /* @__PURE__ */ u("div", { class: "flex flex-col mt-6", children: [
              /* @__PURE__ */ u(
                "progress",
                {
                  class: "progress progress-secondary w-full",
                  value: currentProgress / (totalProgress || 1) * 100,
                  max: "100"
                }
              ),
              /* @__PURE__ */ u("span", { class: "text-sm h-4 leading-none mt-2 text-base-content text-opacity-60", children: `${currentProgress}/${mediaList.length}` })
            ] })
          ] }),
          /* @__PURE__ */ u("div", { class: "flex space-x-2 mt-2", children: [
            /* @__PURE__ */ u("span", { class: "flex-grow" }),
            /* @__PURE__ */ u("button", { class: "btn", onClick: onClose, children: t("Cancel") }),
            /* @__PURE__ */ u("div", { class: "join", children: [
              /* @__PURE__ */ u("button", { class: "btn join-item pr-2", onClick: () => onCopy(), children: copied ? t("Copied!") : t("Copy URLs") }),
              /* @__PURE__ */ u("button", { class: "btn join-item pl-2", onClick: () => onCopy(true), children: /* @__PURE__ */ u(IconFileDownload, {}) })
            ] }),
            /* @__PURE__ */ u("button", { class: cx("btn btn-secondary", loading && "btn-disabled"), onClick: onExport, children: [
              loading && /* @__PURE__ */ u("span", { class: "loading loading-spinner" }),
              t("Start Export")
            ] })
          ] })
        ]
      }
    );
  }
  /**
   * Modified from `dexie-react-hooks` with some lines removed. The modified version
   * is specifically designed for `Observable` and `liveQuery` from Dexie.js.
   *
   * @license Apache-2.0
   * @see https://dexie.org/docs/dexie-react-hooks/useObservable()
   * @see https://github.com/dexie/Dexie.js/blob/v4.0.4/libs/dexie-react-hooks/src/useObservable.ts
   * @param observableFactory Function that returns an observable.
   * @param deps The observableFactory function will be re-executed if deps change.
   * @param defaultResult Result returned on initial render.
   * @returns The current result of the observable.
   */
  function useObservable(observableFactory, deps, defaultResult) {
    const monitor = hooks.useRef({
      hasResult: false,
      result: defaultResult,
      error: null
    });
    const [, triggerUpdate] = hooks.useReducer((x) => x + 1, 0);
    const observable = hooks.useMemo(() => {
      const observable2 = observableFactory();
      if (!observable2 || typeof observable2.subscribe !== "function") {
        throw new TypeError(
          `Observable factory given to useObservable() did not return a valid observable.`
        );
      }
      if (monitor.current.hasResult) {
        return observable2;
      }
      if (typeof observable2.hasValue !== "function" || observable2.hasValue()) {
        if (typeof observable2.getValue === "function") {
          monitor.current.result = observable2.getValue();
          monitor.current.hasResult = true;
        }
      }
      return observable2;
    }, deps);
    hooks.useEffect(() => {
      const subscription = observable.subscribe(
        (val) => {
          const state = monitor.current;
          if (state.error !== null || state.result !== val) {
            state.error = null;
            state.result = val;
            state.hasResult = true;
            triggerUpdate(1);
          }
        },
        (err) => {
          if (monitor.current.error !== err) {
            monitor.current.error = err;
            triggerUpdate(1);
          }
        }
      );
      return subscription.unsubscribe.bind(subscription);
    }, deps);
    if (monitor.current.error) {
      throw monitor.current.error;
    }
    return monitor.current.result;
  }
  /**
   * A hook that subscribes to a live query and returns the current result.
   * Copied from `dexie-react-hooks` with some function overloads removed.
   *
   * @license Apache-2.0
   * @see https://dexie.org/docs/dexie-react-hooks/useLiveQuery()
   * @see https://github.com/dexie/Dexie.js/blob/v4.0.4/libs/dexie-react-hooks/src/useLiveQuery.ts
   * @see https://github.com/dexie/Dexie.js/blob/v4.0.4/src/live-query/live-query.ts
   * @param querier Function that returns a final result (Promise).
   * @param deps Variables that querier is dependent on.
   * @param defaultResult Result returned on initial render.
   * @returns The current result of the live query.
   */
  function useLiveQuery(querier, deps, defaultResult) {
    return useObservable(() => Dexie.liveQuery(querier), deps || [], defaultResult);
  }
  function useCaptureCount(extName) {
    return useLiveQuery(() => databaseManager.extGetCaptureCount(extName), [extName], 0);
  }
  function useCapturedRecords(extName, type) {
    return useLiveQuery(
      () => {
        logger.debug("useCapturedRecords liveQuery re-run", extName);
        if (type === ExtensionType.USER) {
          return databaseManager.extGetCapturedUsers(extName);
        }
        if (type === ExtensionType.TWEET) {
          return databaseManager.extGetCapturedTweets(extName);
        }
      },
      [extName],
      []
    );
  }
  function useClearCaptures(extName) {
    return async () => {
      logger.debug("Clearing captures for extension:", extName);
      return databaseManager.extClearCaptures(extName);
    };
  }
  function flexRender(Comp, props) {
    return !Comp ? null : isComponent(Comp) ? /* @__PURE__ */ u(Comp, { ...props }) : Comp;
  }
  function isComponent(component) {
    return typeof component === "function";
  }
  /**
   * @license MIT
   * https://github.com/TanStack/table/blob/main/packages/react-table/src/index.tsx
   */
  function useReactTable(options2) {
    const resolvedOptions = {
      state: {},
      // Dummy state
      onStateChange: () => {
      },
      // noop
      renderFallbackValue: null,
      ...options2
    };
    const [tableRef] = hooks.useState(() => ({
      current: tableCore.createTable(resolvedOptions)
    }));
    const [state, setState] = hooks.useState(() => tableRef.current.initialState);
    tableRef.current.setOptions((prev) => ({
      ...prev,
      ...options2,
      state: {
        ...state,
        ...options2.state
      },
      // Similarly, we'll maintain both our internal state and any user-provided
      // state.
      onStateChange: (updater) => {
        var _a;
        setState(updater);
        (_a = options2.onStateChange) == null ? void 0 : _a.call(options2, updater);
      }
    }));
    return tableRef.current;
  }
  const Pagination = ({ table }) => {
    const { t } = useTranslation();
    const state = table.getState().pagination;
    return /* @__PURE__ */ u("div", { className: "mt-3 flex items-center gap-2 flex-wrap", children: [
      /* @__PURE__ */ u("span", { children: t("Rows per page:") }),
      /* @__PURE__ */ u(
        "select",
        {
          value: state.pageSize,
          onChange: (e) => {
            table.setPageSize(Number(e.target.value));
          },
          className: "select select-sm select-bordered",
          children: [10, 20, 50, 100].map((pageSize) => /* @__PURE__ */ u("option", { value: pageSize, children: pageSize }, pageSize))
        }
      ),
      /* @__PURE__ */ u("span", { class: "flex-grow" }),
      /* @__PURE__ */ u("span", { children: t("A - B of N items", {
        from: state.pageSize * state.pageIndex + 1,
        to: Math.min(
          state.pageSize * (state.pageIndex + 1),
          table.getFilteredRowModel().rows.length
        ),
        total: table.getFilteredRowModel().rows.length
      }) }),
      /* @__PURE__ */ u(
        "input",
        {
          defaultValue: (state.pageIndex + 1).toString(),
          type: "number",
          onInput: (e) => {
            const value = e.target.value;
            table.setPageIndex(value ? Number(value) - 1 : 0);
          },
          className: "input input-bordered w-20 input-sm text-center"
        }
      ),
      /* @__PURE__ */ u("div", { class: "join", children: [
        /* @__PURE__ */ u(
          "button",
          {
            className: "join-item btn btn-sm",
            onClick: () => table.setPageIndex(0),
            disabled: !table.getCanPreviousPage(),
            children: /* @__PURE__ */ u(IconChevronLeftPipe, { size: 20 })
          }
        ),
        /* @__PURE__ */ u(
          "button",
          {
            className: "join-item btn btn-sm",
            onClick: () => table.previousPage(),
            disabled: !table.getCanPreviousPage(),
            children: /* @__PURE__ */ u(IconChevronLeft, { size: 20 })
          }
        ),
        /* @__PURE__ */ u(
          "button",
          {
            className: "join-item btn btn-sm",
            onClick: () => table.nextPage(),
            disabled: !table.getCanNextPage(),
            children: /* @__PURE__ */ u(IconChevronRight, { size: 20 })
          }
        ),
        /* @__PURE__ */ u(
          "button",
          {
            className: "join-item btn btn-sm",
            onClick: () => table.setPageIndex(table.getPageCount() - 1),
            disabled: !table.getCanNextPage(),
            children: /* @__PURE__ */ u(IconChevronRightPipe, { size: 20 })
          }
        )
      ] })
    ] });
  };
  function ExportDataModal({ title, table, show, onClose }) {
    const { t } = useTranslation("exporter");
    const [selectedFormat, setSelectedFormat] = useSignalState(EXPORT_FORMAT.JSON);
    const [loading, setLoading] = useSignalState(false);
    const [includeMetadata, toggleIncludeMetadata] = useToggle(false);
    const [currentProgress, setCurrentProgress] = useSignalState(0);
    const [totalProgress, setTotalProgress] = useSignalState(0);
    const selectedRows = table.getSelectedRowModel().rows;
    const onExport = async () => {
      setLoading(true);
      setTotalProgress(selectedRows.length);
      const allRecords = [];
      for (const row of selectedRows) {
        const allCells = row.getAllCells();
        const record = {};
        for (const cell of allCells) {
          const value = cell.getValue();
          const meta = cell.column.columnDef.meta;
          if ((meta == null ? void 0 : meta.exportable) === false) {
            continue;
          }
          let exportValue = (meta == null ? void 0 : meta.exportValue) ? meta.exportValue(row) : value;
          if (exportValue === void 0) {
            exportValue = null;
          }
          record[(meta == null ? void 0 : meta.exportKey) || cell.column.id] = exportValue;
        }
        if (includeMetadata) {
          record.metadata = row.original;
        }
        allRecords.push(record);
        setCurrentProgress(allRecords.length);
      }
      const headerTranslations = table.getAllColumns().reduce((acc, column) => {
        var _a, _b;
        const key = ((_a = column.columnDef.meta) == null ? void 0 : _a.exportKey) || column.id;
        const header = ((_b = column.columnDef.meta) == null ? void 0 : _b.exportHeader) || column.id;
        acc[key] = t(header);
        return acc;
      }, {});
      await exportData(
        allRecords,
        selectedFormat,
        `twitter-${title}-${Date.now()}.${selectedFormat.toLowerCase()}`,
        headerTranslations
      );
      setLoading(false);
    };
    return /* @__PURE__ */ u(
      Modal,
      {
        class: "max-w-sm md:max-w-screen-sm sm:max-w-screen-sm max-h-full",
        title: `${title} ${t("Data")}`,
        show,
        onClose,
        children: [
          /* @__PURE__ */ u("div", { class: "px-4 text-base", children: [
            /* @__PURE__ */ u("p", { class: "text-base-content text-opacity-60 mb-2 leading-5 text-sm", children: t(
              "Export captured data as JSON/HTML/CSV file. This may take a while depending on the amount of data. The exported file does not include media files such as images and videos but only the URLs."
            ) }),
            /* @__PURE__ */ u("div", { class: "flex items-center", children: [
              /* @__PURE__ */ u("p", { class: "mr-2 leading-8", children: t("Data length:") }),
              /* @__PURE__ */ u("span", { class: "font-mono leading-6 h-6 bg-base-200 px-2 rounded-md", children: selectedRows.length })
            ] }),
            /* @__PURE__ */ u("div", { class: "flex items-center", children: [
              /* @__PURE__ */ u("p", { class: "mr-2 leading-8", children: t("Include all metadata:") }),
              /* @__PURE__ */ u(
                "input",
                {
                  type: "checkbox",
                  class: "checkbox checkbox-sm",
                  checked: includeMetadata,
                  onChange: toggleIncludeMetadata
                }
              )
            ] }),
            /* @__PURE__ */ u("div", { class: "flex", children: [
              /* @__PURE__ */ u("p", { class: "mr-2 leading-8", children: t("Export as:") }),
              /* @__PURE__ */ u(
                "select",
                {
                  class: "select select-bordered select-sm w-32",
                  onChange: (e) => {
                    setSelectedFormat(e.target.value);
                  },
                  children: Object.values(EXPORT_FORMAT).map((type) => /* @__PURE__ */ u("option", { selected: type === selectedFormat, children: type }, type))
                }
              )
            ] }),
            selectedRows.length > 0 ? null : /* @__PURE__ */ u("div", { class: "flex items-center justify-center h-28 w-full", children: /* @__PURE__ */ u("p", { class: "text-base-content text-opacity-50", children: t("No data selected.") }) }),
            /* @__PURE__ */ u("div", { class: "flex flex-col mt-6", children: [
              /* @__PURE__ */ u(
                "progress",
                {
                  class: "progress progress-primary w-full",
                  value: currentProgress / (totalProgress || 1) * 100,
                  max: "100"
                }
              ),
              /* @__PURE__ */ u("span", { class: "text-sm leading-none mt-2 text-base-content text-opacity-60", children: `${currentProgress}/${selectedRows.length}` })
            ] })
          ] }),
          /* @__PURE__ */ u("div", { class: "flex space-x-2", children: [
            /* @__PURE__ */ u("span", { class: "flex-grow" }),
            /* @__PURE__ */ u("button", { class: "btn", onClick: onClose, children: t("Cancel") }),
            /* @__PURE__ */ u("button", { class: cx("btn btn-primary", loading && "btn-disabled"), onClick: onExport, children: [
              loading && /* @__PURE__ */ u("span", { class: "loading loading-spinner" }),
              t("Start Export")
            ] })
          ] })
        ]
      }
    );
  }
  function BaseTableView({
    title,
    records,
    columns: columns2,
    clear,
    renderActions,
    renderExtra
  }) {
    const { t } = useTranslation();
    const [mediaPreview, setMediaPreview] = useSignalState("");
    const [rawDataPreview, setRawDataPreview] = useSignalState(null);
    const table = useReactTable({
      data: records ?? [],
      columns: columns2,
      getCoreRowModel: tableCore.getCoreRowModel(),
      getFilteredRowModel: tableCore.getFilteredRowModel(),
      getSortedRowModel: tableCore.getSortedRowModel(),
      getPaginationRowModel: tableCore.getPaginationRowModel(),
      meta: {
        mediaPreview,
        setMediaPreview: (url) => setMediaPreview(url),
        rawDataPreview,
        setRawDataPreview: (data) => setRawDataPreview(data)
      }
    });
    const [showExportDataModal, toggleShowExportDataModal] = useToggle();
    hooks.useEffect(() => {
      setTimeout(() => {
        if (!table.getIsSomeRowsSelected()) {
          table.toggleAllRowsSelected(true);
        }
      }, 100);
    }, [table]);
    return /* @__PURE__ */ u(preact.Fragment, { children: [
      /* @__PURE__ */ u(SearchArea, { defaultValue: table.getState().globalFilter, onChange: table.setGlobalFilter }),
      /* @__PURE__ */ u("main", { class: "max-w-full grow overflow-scroll bg-base-200 overscroll-none", children: [
        /* @__PURE__ */ u("table", { class: "table table-pin-rows table-border-bc table-padding-sm", children: [
          /* @__PURE__ */ u("thead", { children: table.getHeaderGroups().map((headerGroup) => /* @__PURE__ */ u("tr", { children: headerGroup.headers.map((header) => /* @__PURE__ */ u(
            "th",
            {
              className: header.column.getCanSort() ? "cursor-pointer select-none" : "",
              onClick: header.column.getToggleSortingHandler(),
              children: [
                flexRender(header.column.columnDef.header, header.getContext()),
                header.column.getIsSorted() === "asc" && /* @__PURE__ */ u(IconSortAscending, { size: 15, class: "inline align-top ml-1" }),
                header.column.getIsSorted() === "desc" && /* @__PURE__ */ u(IconSortDescending, { size: 15, class: "inline align-top ml-1" })
              ]
            },
            header.id
          )) }, headerGroup.id)) }),
          /* @__PURE__ */ u("tbody", { children: table.getRowModel().rows.map((row) => /* @__PURE__ */ u("tr", { children: row.getVisibleCells().map((cell) => /* @__PURE__ */ u("td", { children: flexRender(cell.column.columnDef.cell, cell.getContext()) }, cell.id)) }, row.id)) })
        ] }),
        table.getRowModel().rows.length > 0 ? null : /* @__PURE__ */ u("div", { class: "flex items-center justify-center h-[320px] w-full", children: /* @__PURE__ */ u("p", { class: "text-base-content text-opacity-50", children: t("No data available.") }) })
      ] }),
      /* @__PURE__ */ u(Pagination, { table }),
      /* @__PURE__ */ u("div", { class: "flex mt-3 space-x-2", children: [
        /* @__PURE__ */ u("button", { class: "btn btn-neutral btn-ghost", onClick: clear, children: t("Clear") }),
        /* @__PURE__ */ u("span", { class: "flex-grow" }),
        renderActions == null ? void 0 : renderActions(table),
        /* @__PURE__ */ u("button", { class: "btn btn-primary", onClick: toggleShowExportDataModal, children: t("Export Data") })
      ] }),
      /* @__PURE__ */ u(
        Modal,
        {
          title: t("JSON View"),
          class: "max-w-xl",
          show: !!rawDataPreview,
          onClose: () => setRawDataPreview(null),
          children: /* @__PURE__ */ u("main", { class: "max-w-full max-h-[500px] overflow-scroll overscroll-none", children: typeof rawDataPreview === "string" ? /* @__PURE__ */ u("p", { class: "whitespace-pre-wrap", children: rawDataPreview }) : /* @__PURE__ */ u("pre", { class: "text-xs leading-none", children: JSON.stringify(rawDataPreview, null, 2) }) })
        }
      ),
      /* @__PURE__ */ u(
        Modal,
        {
          title: t("Media View"),
          class: "max-w-xl",
          show: !!mediaPreview,
          onClose: () => setMediaPreview(""),
          children: /* @__PURE__ */ u("main", { class: "max-w-full", children: mediaPreview.includes(".mp4") ? /* @__PURE__ */ u("video", { controls: true, class: "w-full max-h-[400px] object-contain", src: mediaPreview }) : /* @__PURE__ */ u("img", { class: "w-full max-h-[400px] object-contain", src: mediaPreview }) })
        }
      ),
      /* @__PURE__ */ u(
        ExportDataModal,
        {
          title,
          table,
          show: showExportDataModal,
          onClose: toggleShowExportDataModal
        }
      ),
      renderExtra == null ? void 0 : renderExtra(table)
    ] });
  }
  const columnHelper$2 = tableCore.createColumnHelper();
  const rtSourceAccessor = (row) => {
    const source = extractRetweetedTweet(row);
    return source ? extractTweetUserScreenName(source) : null;
  };
  const quoteSourceAccessor = (row) => {
    const source = extractQuotedTweet(row);
    return source ? extractTweetUserScreenName(source) : null;
  };
  const columns$2 = [
    columnHelper$2.display({
      id: "select",
      meta: { exportable: false },
      header: ({ table }) => /* @__PURE__ */ u(
        "input",
        {
          type: "checkbox",
          class: "checkbox checkbox-sm align-middle",
          checked: table.getIsAllRowsSelected(),
          indeterminate: table.getIsSomeRowsSelected(),
          onChange: table.getToggleAllRowsSelectedHandler()
        }
      ),
      cell: ({ row }) => /* @__PURE__ */ u(
        "input",
        {
          type: "checkbox",
          class: "checkbox checkbox-sm",
          checked: row.getIsSelected(),
          disabled: !row.getCanSelect(),
          indeterminate: row.getIsSomeSelected(),
          onChange: row.getToggleSelectedHandler()
        }
      )
    }),
    columnHelper$2.accessor("rest_id", {
      meta: { exportKey: "id", exportHeader: "ID" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "ID" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-20 break-all font-mono text-xs", children: info.getValue() })
    }),
    columnHelper$2.accessor((row) => +parseTwitterDateTime(row.legacy.created_at), {
      id: "created_at",
      meta: {
        exportKey: "created_at",
        exportHeader: "Date",
        exportValue: (row) => formatDateTime(
          parseTwitterDateTime(row.original.legacy.created_at),
          appOptionsManager.get("dateTimeFormat")
        )
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Date" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-24", children: /* @__PURE__ */ u("a", { class: "link", target: "_blank", href: getTweetURL(info.row.original), children: formatDateTime(info.getValue(), appOptionsManager.get("dateTimeFormat")) }) })
    }),
    columnHelper$2.accessor("legacy.full_text", {
      meta: {
        exportKey: "full_text",
        exportHeader: "Content",
        exportValue: (row) => extractTweetFullText(row.original)
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Content" }),
      cell: (info) => /* @__PURE__ */ u("div", { children: [
        /* @__PURE__ */ u(
          "p",
          {
            class: "w-60 whitespace-pre-wrap",
            dangerouslySetInnerHTML: {
              __html: strEntitiesToHTML(info.row.original.legacy.full_text, [
                ...info.row.original.legacy.entities.urls,
                ...info.row.original.legacy.entities.media ?? []
              ])
            }
          }
        ),
        info.row.original.note_tweet && /* @__PURE__ */ u(
          "button",
          {
            class: "link",
            onClick: () => {
              var _a;
              return (_a = info.table.options.meta) == null ? void 0 : _a.setRawDataPreview(
                extractTweetFullText(info.row.original)
              );
            },
            children: [
              ">>",
              " ",
              /* @__PURE__ */ u(Trans, { i18nKey: "Show Full Text" })
            ]
          }
        )
      ] })
    }),
    columnHelper$2.accessor((row) => extractTweetMedia(row).length, {
      id: "media",
      meta: {
        exportKey: "media",
        exportHeader: "Media",
        exportValue: (row) => extractTweetMedia(row.original).map((media) => ({
          type: media.type,
          url: media.url,
          thumbnail: formatTwitterImage(media.media_url_https, "thumb"),
          original: getMediaOriginalUrl(media),
          ext_alt_text: media.ext_alt_text
        }))
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Media" }),
      cell: (info) => /* @__PURE__ */ u(
        MediaDisplayColumn,
        {
          data: extractTweetMedia(info.row.original),
          onClick: (media) => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setMediaPreview(getMediaOriginalUrl(media));
          }
        }
      )
    }),
    columnHelper$2.accessor("core.user_results.result.core.screen_name", {
      meta: { exportKey: "screen_name", exportHeader: "Screen Name" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Screen Name" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "whitespace-pre", children: /* @__PURE__ */ u(
        "a",
        {
          class: "link",
          target: "_blank",
          href: getUserURL(info.row.original.core.user_results.result),
          children: [
            "@",
            info.getValue()
          ]
        }
      ) })
    }),
    columnHelper$2.accessor("core.user_results.result.core.name", {
      meta: { exportKey: "name", exportHeader: "Profile Name" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Profile Name" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-32", children: info.getValue() })
    }),
    columnHelper$2.accessor("core.user_results.result.avatar.image_url", {
      meta: { exportKey: "profile_image_url", exportHeader: "Profile Image" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Profile Image" }),
      cell: (info) => /* @__PURE__ */ u(
        "div",
        {
          class: "cursor-pointer",
          onClick: () => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setMediaPreview(getProfileImageOriginalUrl(info.getValue()));
          },
          children: /* @__PURE__ */ u("img", { class: "w-12 h-12 rounded", src: info.getValue() })
        }
      )
    }),
    columnHelper$2.accessor("core.user_results.result.rest_id", {
      meta: { exportKey: "user_id", exportHeader: "User ID" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "User ID" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-20 break-all font-mono text-xs", children: info.getValue() })
    }),
    columnHelper$2.accessor("legacy.in_reply_to_screen_name", {
      meta: {
        exportKey: "in_reply_to",
        exportHeader: "Replying To",
        exportValue: (row) => row.original.legacy.in_reply_to_status_id_str
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Replying To" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "whitespace-pre", children: info.row.original.legacy.in_reply_to_status_id_str ? /* @__PURE__ */ u("a", { class: "link", target: "_blank", href: getInReplyToTweetURL(info.row.original), children: [
        "@",
        info.getValue()
      ] }) : "N/A" })
    }),
    columnHelper$2.accessor(rtSourceAccessor, {
      id: "rt_source",
      meta: {
        exportKey: "retweeted_status",
        exportHeader: "RT Source",
        exportValue: (row) => {
          var _a;
          return (_a = extractRetweetedTweet(row.original)) == null ? void 0 : _a.rest_id;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "RT Source" }),
      cell: (info) => {
        const source = extractRetweetedTweet(info.row.original);
        return /* @__PURE__ */ u("p", { class: "whitespace-pre", children: source ? /* @__PURE__ */ u("a", { class: "link", target: "_blank", href: getTweetURL(source), children: [
          "@",
          info.getValue()
        ] }) : "N/A" });
      }
    }),
    columnHelper$2.accessor(quoteSourceAccessor, {
      id: "quote_source",
      meta: {
        exportKey: "quoted_status",
        exportHeader: "Quote Source",
        exportValue: (row) => {
          var _a;
          return (_a = extractQuotedTweet(row.original)) == null ? void 0 : _a.rest_id;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Quote Source" }),
      cell: (info) => {
        const source = extractQuotedTweet(info.row.original);
        return /* @__PURE__ */ u("p", { class: "whitespace-pre", children: source ? /* @__PURE__ */ u("a", { class: "link", target: "_blank", href: getTweetURL(source), children: [
          "@",
          info.getValue()
        ] }) : "N/A" });
      }
    }),
    columnHelper$2.display({
      id: "quoted_article_title",
      meta: {
        exportKey: "quoted_article_title",
        exportHeader: "Quoted Article Title",
        exportValue: (row) => {
          var _a, _b, _c, _d;
          return (_d = (_c = (_b = (_a = extractQuotedTweet(row.original)) == null ? void 0 : _a.article) == null ? void 0 : _b.article_results) == null ? void 0 : _c.result) == null ? void 0 : _d.title;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Quoted Article Title" }),
      cell: (info) => {
        var _a, _b, _c;
        const quoted = extractQuotedTweet(info.row.original);
        const title = (_c = (_b = (_a = quoted == null ? void 0 : quoted.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.title;
        return /* @__PURE__ */ u("p", { class: "w-40 whitespace-pre-wrap", children: title ?? "N/A" });
      }
    }),
    columnHelper$2.display({
      id: "quoted_article_preview",
      meta: {
        exportKey: "quoted_article_preview",
        exportHeader: "Quoted Article Preview",
        exportValue: (row) => {
          var _a, _b, _c, _d;
          return (_d = (_c = (_b = (_a = extractQuotedTweet(row.original)) == null ? void 0 : _a.article) == null ? void 0 : _b.article_results) == null ? void 0 : _c.result) == null ? void 0 : _d.preview_text;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Quoted Article Preview" }),
      cell: (info) => {
        var _a, _b, _c;
        const quoted = extractQuotedTweet(info.row.original);
        const preview = (_c = (_b = (_a = quoted == null ? void 0 : quoted.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.preview_text;
        return /* @__PURE__ */ u("p", { class: "w-60 whitespace-pre-wrap line-clamp-3", children: preview ?? "N/A" });
      }
    }),
    columnHelper$2.display({
      id: "quoted_article_url",
      meta: {
        exportKey: "quoted_article_url",
        exportHeader: "Quoted Article URL",
        exportValue: (row) => {
          var _a, _b, _c, _d;
          const articleId = (_d = (_c = (_b = (_a = extractQuotedTweet(row.original)) == null ? void 0 : _a.article) == null ? void 0 : _b.article_results) == null ? void 0 : _c.result) == null ? void 0 : _d.rest_id;
          return articleId ? `https://x.com/i/article/${articleId}` : null;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Quoted Article URL" }),
      cell: (info) => {
        var _a, _b, _c;
        const quoted = extractQuotedTweet(info.row.original);
        const articleId = (_c = (_b = (_a = quoted == null ? void 0 : quoted.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.rest_id;
        return articleId ? /* @__PURE__ */ u(
          "a",
          {
            href: `https://x.com/i/article/${articleId}`,
            target: "_blank",
            rel: "noopener noreferrer",
            class: "link text-xs",
            children: /* @__PURE__ */ u(IconLink, { size: 16 })
          }
        ) : /* @__PURE__ */ u("p", { children: "N/A" });
      }
    }),
    columnHelper$2.display({
      id: "quoted_article_cover",
      meta: {
        exportKey: "quoted_article_cover_url",
        exportHeader: "Quoted Article Cover",
        exportValue: (row) => {
          var _a, _b, _c, _d, _e, _f;
          return (_f = (_e = (_d = (_c = (_b = (_a = extractQuotedTweet(row.original)) == null ? void 0 : _a.article) == null ? void 0 : _b.article_results) == null ? void 0 : _c.result) == null ? void 0 : _d.cover_media) == null ? void 0 : _e.media_info) == null ? void 0 : _f.original_img_url;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Quoted Article Cover" }),
      cell: (info) => {
        var _a, _b, _c, _d, _e;
        const quoted = extractQuotedTweet(info.row.original);
        const coverUrl = (_e = (_d = (_c = (_b = (_a = quoted == null ? void 0 : quoted.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.cover_media) == null ? void 0 : _d.media_info) == null ? void 0 : _e.original_img_url;
        return coverUrl ? /* @__PURE__ */ u(
          "div",
          {
            class: "cursor-pointer",
            onClick: () => {
              var _a2;
              return (_a2 = info.table.options.meta) == null ? void 0 : _a2.setMediaPreview(coverUrl);
            },
            children: /* @__PURE__ */ u("img", { class: "w-16 h-10 object-cover rounded", src: coverUrl })
          }
        ) : /* @__PURE__ */ u("p", { children: "N/A" });
      }
    }),
    columnHelper$2.display({
      id: "media_tags",
      meta: {
        exportKey: "media_tags",
        exportHeader: "Media Tags",
        exportValue: (row) => extractTweetMediaTags(row.original)
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Media Tags" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: extractTweetMediaTags(info.row.original).length ? extractTweetMediaTags(info.row.original).map((tag) => /* @__PURE__ */ u("a", { class: "link inline-block", target: "_blank", href: getUserURL(tag.screen_name), children: [
        "@",
        tag.screen_name
      ] })) : "N/A" })
    }),
    columnHelper$2.accessor("legacy.favorite_count", {
      meta: { exportKey: "favorite_count", exportHeader: "Favorites" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Favorites" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() })
    }),
    columnHelper$2.accessor("legacy.retweet_count", {
      meta: { exportKey: "retweet_count", exportHeader: "Retweets" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Retweets" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() })
    }),
    columnHelper$2.accessor("legacy.bookmark_count", {
      meta: { exportKey: "bookmark_count", exportHeader: "Bookmarks" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Bookmarks" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() })
    }),
    columnHelper$2.accessor("legacy.quote_count", {
      meta: { exportKey: "quote_count", exportHeader: "Quotes" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Quotes" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() })
    }),
    columnHelper$2.accessor("legacy.reply_count", {
      meta: { exportKey: "reply_count", exportHeader: "Replies" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Replies" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() })
    }),
    columnHelper$2.accessor("views.count", {
      meta: {
        exportKey: "views_count",
        exportHeader: "Views",
        exportValue: (row) => {
          var _a;
          return typeof ((_a = row.original.views) == null ? void 0 : _a.count) === "undefined" ? null : +row.original.views.count;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Views" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$2.accessor("legacy.favorited", {
      meta: { exportKey: "favorited", exportHeader: "Favorited" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Favorited" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$2.accessor("legacy.retweeted", {
      meta: { exportKey: "retweeted", exportHeader: "Retweeted" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Retweeted" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$2.accessor("legacy.bookmarked", {
      meta: { exportKey: "bookmarked", exportHeader: "Bookmarked" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Bookmarked" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$2.accessor("article.article_results.result.title", {
      meta: {
        exportKey: "article_title",
        exportHeader: "Article Title"
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Article Title" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-40 whitespace-pre-wrap", children: info.getValue() ?? "N/A" })
    }),
    columnHelper$2.accessor("article.article_results.result.preview_text", {
      meta: {
        exportKey: "article_preview",
        exportHeader: "Article Preview"
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Article Preview" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-60 whitespace-pre-wrap line-clamp-3", children: info.getValue() ?? "N/A" })
    }),
    columnHelper$2.display({
      id: "article_url",
      meta: {
        exportKey: "article_url",
        exportHeader: "Article URL",
        exportValue: (row) => {
          var _a, _b, _c;
          const articleId = (_c = (_b = (_a = row.original.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.rest_id;
          return articleId ? `https://x.com/i/article/${articleId}` : null;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Article URL" }),
      cell: (info) => {
        var _a, _b, _c;
        const articleId = (_c = (_b = (_a = info.row.original.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.rest_id;
        return articleId ? /* @__PURE__ */ u(
          "a",
          {
            href: `https://x.com/i/article/${articleId}`,
            target: "_blank",
            rel: "noopener noreferrer",
            class: "link text-xs",
            children: /* @__PURE__ */ u(IconLink, { size: 16 })
          }
        ) : /* @__PURE__ */ u("p", { children: "N/A" });
      }
    }),
    columnHelper$2.display({
      id: "article_cover",
      meta: {
        exportKey: "article_cover_url",
        exportHeader: "Article Cover",
        exportValue: (row) => {
          var _a, _b, _c, _d, _e;
          return (_e = (_d = (_c = (_b = (_a = row.original.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.cover_media) == null ? void 0 : _d.media_info) == null ? void 0 : _e.original_img_url;
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Article Cover" }),
      cell: (info) => {
        var _a, _b, _c, _d, _e;
        const coverUrl = (_e = (_d = (_c = (_b = (_a = info.row.original.article) == null ? void 0 : _a.article_results) == null ? void 0 : _b.result) == null ? void 0 : _c.cover_media) == null ? void 0 : _d.media_info) == null ? void 0 : _e.original_img_url;
        return coverUrl ? /* @__PURE__ */ u(
          "div",
          {
            class: "cursor-pointer",
            onClick: () => {
              var _a2;
              return (_a2 = info.table.options.meta) == null ? void 0 : _a2.setMediaPreview(coverUrl);
            },
            children: /* @__PURE__ */ u("img", { class: "w-16 h-10 object-cover rounded", src: coverUrl })
          }
        ) : /* @__PURE__ */ u("p", { children: "N/A" });
      }
    }),
    columnHelper$2.display({
      id: "url",
      meta: {
        exportKey: "url",
        exportHeader: "URL",
        exportValue: (row) => getTweetURL(row.original)
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "URL" }),
      cell: (info) => /* @__PURE__ */ u("a", { href: getTweetURL(info.row.original), target: "_blank", children: /* @__PURE__ */ u(IconLink, {}) })
    }),
    columnHelper$2.display({
      id: "actions",
      meta: { exportable: false },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Actions" }),
      cell: (info) => /* @__PURE__ */ u("div", { class: "flex flex-row items-start space-x-1", children: /* @__PURE__ */ u(
        "button",
        {
          onClick: () => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setRawDataPreview(info.row.original);
          },
          class: "btn btn-xs btn-neutral whitespace-nowrap",
          children: /* @__PURE__ */ u(Trans, { i18nKey: "Details" })
        }
      ) })
    })
  ];
  const columnHelper$1 = tableCore.createColumnHelper();
  const columns$1 = [
    columnHelper$1.display({
      id: "select",
      meta: { exportable: false },
      header: ({ table }) => /* @__PURE__ */ u(
        "input",
        {
          type: "checkbox",
          class: "checkbox checkbox-sm align-middle",
          checked: table.getIsAllRowsSelected(),
          indeterminate: table.getIsSomeRowsSelected(),
          onChange: table.getToggleAllRowsSelectedHandler()
        }
      ),
      cell: ({ row }) => /* @__PURE__ */ u(
        "input",
        {
          type: "checkbox",
          class: "checkbox checkbox-sm",
          checked: row.getIsSelected(),
          disabled: !row.getCanSelect(),
          indeterminate: row.getIsSomeSelected(),
          onChange: row.getToggleSelectedHandler()
        }
      )
    }),
    columnHelper$1.accessor("rest_id", {
      meta: { exportKey: "id", exportHeader: "ID" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "ID" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-20 break-all font-mono text-xs", children: info.getValue() })
    }),
    columnHelper$1.accessor("core.screen_name", {
      meta: { exportKey: "screen_name", exportHeader: "Screen Name" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Screen Name" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "whitespace-pre", children: /* @__PURE__ */ u("a", { class: "link", target: "_blank", href: getUserURL(info.row.original), children: [
        "@",
        info.getValue()
      ] }) })
    }),
    columnHelper$1.accessor("core.name", {
      meta: { exportKey: "name", exportHeader: "Profile Name" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Profile Name" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-32", children: info.getValue() })
    }),
    columnHelper$1.accessor("legacy.description", {
      meta: { exportKey: "description", exportHeader: "Description" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Description" }),
      cell: (info) => {
        var _a;
        return /* @__PURE__ */ u(
          "p",
          {
            class: "w-52 break-words",
            dangerouslySetInnerHTML: {
              __html: strEntitiesToHTML(
                info.row.original.legacy.description || "N/A",
                (_a = info.row.original.legacy.entities) == null ? void 0 : _a.description.urls
              )
            }
          }
        );
      }
    }),
    columnHelper$1.accessor("avatar.image_url", {
      meta: { exportKey: "profile_image_url", exportHeader: "Profile Image" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Profile Image" }),
      cell: (info) => /* @__PURE__ */ u(
        "div",
        {
          class: "cursor-pointer",
          onClick: () => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setMediaPreview(getProfileImageOriginalUrl(info.getValue()));
          },
          children: /* @__PURE__ */ u("img", { class: "w-12 h-12 rounded", src: info.getValue() })
        }
      )
    }),
    columnHelper$1.accessor("legacy.profile_banner_url", {
      meta: { exportKey: "profile_banner_url", exportHeader: "Profile Banner" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Profile Banner" }),
      cell: (info) => /* @__PURE__ */ u(
        "div",
        {
          class: "cursor-pointer w-36 h-12",
          onClick: () => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setMediaPreview(info.getValue() ?? "");
          },
          children: info.getValue() ? /* @__PURE__ */ u("img", { class: "w-auto h-12 rounded", src: `${info.getValue()}/600x200` }) : /* @__PURE__ */ u("span", { class: "leading-[48px]", children: "N/A" })
        }
      )
    }),
    columnHelper$1.accessor("legacy.followers_count", {
      meta: { exportKey: "followers_count", exportHeader: "Followers" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Followers" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$1.accessor("legacy.friends_count", {
      meta: { exportKey: "friends_count", exportHeader: "FollowingCount" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "FollowingCount" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$1.accessor("legacy.statuses_count", {
      meta: { exportKey: "statuses_count", exportHeader: "Statuses" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Statuses" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$1.accessor("legacy.favourites_count", {
      meta: { exportKey: "favourites_count", exportHeader: "Favourites" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Favourites" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$1.accessor("legacy.listed_count", {
      meta: { exportKey: "listed_count", exportHeader: "Listed" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Listed" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$1.accessor("location.location", {
      meta: { exportKey: "location", exportHeader: "Location" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Location" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$1.accessor("legacy.url", {
      meta: { exportKey: "website", exportHeader: "Website" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Website" }),
      cell: (info) => {
        var _a, _b;
        return /* @__PURE__ */ u(
          "p",
          {
            dangerouslySetInnerHTML: {
              __html: strEntitiesToHTML(
                info.row.original.legacy.url || "N/A",
                (_b = (_a = info.row.original.legacy.entities) == null ? void 0 : _a.url) == null ? void 0 : _b.urls
              )
            }
          }
        );
      }
    }),
    columnHelper$1.accessor("legacy_extended_profile.birthdate", {
      meta: {
        exportKey: "birthdate",
        exportHeader: "Birthdate",
        exportValue: (row) => {
          var _a;
          return formatTwitterBirthdate((_a = row.original.legacy_extended_profile) == null ? void 0 : _a.birthdate);
        }
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Birthdate" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: formatTwitterBirthdate(info.getValue()) ?? "N/A" })
    }),
    columnHelper$1.accessor("verification.verified_type", {
      meta: { exportKey: "verified_type", exportHeader: "Verified Type" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Verified Type" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ?? "N/A" })
    }),
    columnHelper$1.accessor("is_blue_verified", {
      meta: { exportKey: "is_blue_verified", exportHeader: "Blue Verified" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Blue Verified" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$1.accessor("relationship_perspectives.following", {
      meta: { exportKey: "following", exportHeader: "Following" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Following" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$1.accessor("relationship_perspectives.followed_by", {
      meta: { exportKey: "followed_by", exportHeader: "Follows You" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Follows You" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$1.accessor("dm_permissions.can_dm", {
      meta: { exportKey: "can_dm", exportHeader: "Can DM" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Can DM" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$1.accessor("privacy.protected", {
      meta: { exportKey: "protected", exportHeader: "Protected" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Protected" }),
      cell: (info) => /* @__PURE__ */ u("p", { children: info.getValue() ? "YES" : "NO" })
    }),
    columnHelper$1.accessor((row) => +parseTwitterDateTime(row.core.created_at), {
      id: "created_at",
      meta: {
        exportKey: "created_at",
        exportHeader: "Created At",
        exportValue: (row) => formatDateTime(
          parseTwitterDateTime(row.original.core.created_at),
          appOptionsManager.get("dateTimeFormat")
        )
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Created At" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-24", children: info.getValue() ? formatDateTime(info.getValue(), appOptionsManager.get("dateTimeFormat")) : "N/A" })
    }),
    columnHelper$1.display({
      id: "url",
      meta: {
        exportKey: "url",
        exportHeader: "URL",
        exportValue: (row) => getUserURL(row.original)
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "URL" }),
      cell: (info) => /* @__PURE__ */ u("a", { href: getUserURL(info.row.original), target: "_blank", children: /* @__PURE__ */ u(IconLink, {}) })
    }),
    columnHelper$1.display({
      id: "actions",
      meta: { exportable: false },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Actions" }),
      cell: (info) => /* @__PURE__ */ u("div", { class: "flex flex-row items-start space-x-1", children: /* @__PURE__ */ u(
        "button",
        {
          onClick: () => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setRawDataPreview(info.row.original);
          },
          class: "btn btn-xs btn-neutral whitespace-nowrap",
          children: /* @__PURE__ */ u(Trans, { i18nKey: "Details" })
        }
      ) })
    })
  ];
  function TableView({ title, extension }) {
    const { t } = useTranslation();
    const { name: name2, type } = extension;
    const records = useCapturedRecords(name2, type);
    const clearCapturedData = useClearCaptures(name2);
    const [showExportMediaModal, toggleShowExportMediaModal] = useToggle();
    const columns2 = type === ExtensionType.TWEET ? columns$2 : columns$1;
    return /* @__PURE__ */ u(
      BaseTableView,
      {
        title,
        records: records ?? [],
        columns: columns2,
        clear: clearCapturedData,
        renderActions: () => /* @__PURE__ */ u("button", { class: "btn btn-secondary", onClick: toggleShowExportMediaModal, children: t("Export Media") }),
        renderExtra: (table) => /* @__PURE__ */ u(
          ExportMediaModal,
          {
            title,
            table,
            isTweet: type === ExtensionType.TWEET,
            show: showExportMediaModal,
            onClose: toggleShowExportMediaModal
          }
        )
      }
    );
  }
  function CommonModuleUI({ extension }) {
    const { t } = useTranslation();
    const [showModal, toggleShowModal] = useToggle();
    const title = t(extension.name.replace("Module", ""));
    const count = useCaptureCount(extension.name);
    if (extension.type !== "tweet" && extension.type !== "user") {
      throw new Error("Incorrect use of CommonModuleUI component.");
    }
    return /* @__PURE__ */ u(
      ExtensionPanel,
      {
        title,
        description: `${t("Captured:")} ${count}`,
        active: !!count && count > 0,
        onClick: toggleShowModal,
        indicatorColor: extension.type === ExtensionType.TWEET ? "bg-primary" : "bg-secondary",
        children: /* @__PURE__ */ u(
          Modal,
          {
            class: "max-w-4xl md:max-w-screen-md sm:max-w-screen-sm min-h-[512px]",
            title,
            show: showModal,
            onClose: toggleShowModal,
            children: /* @__PURE__ */ u(TableView, { title, extension })
          }
        )
      }
    );
  }
  const BookmarksInterceptor = async (req, res, ext) => {
    var _a, _b;
    if (/\/graphql\/.+\/CreateBookmark/.test(req.url)) {
      logger.info(`[BookmarksInterceptor] CreateBookmark API detected`);
      try {
        const json = JSON.parse(res.responseText);
        if (((_a = json.data) == null ? void 0 : _a.tweet_bookmark_put) === "Done" && req.body) {
          const requestBody = JSON.parse(req.body);
          const tweetId = (_b = requestBody.variables) == null ? void 0 : _b.tweet_id;
          if (tweetId) {
            const fullTweet = await databaseManager.getTweetById(tweetId);
            if (fullTweet) {
              logger.info(`[BookmarksInterceptor] Found full tweet data for ${tweetId}`);
              await webhookManager.triggerWebhooks("bookmark", fullTweet);
            } else {
              logger.info(`[BookmarksInterceptor] Tweet ${tweetId} not in database, using minimal data`);
              const minimalTweet = { rest_id: tweetId };
              await webhookManager.triggerWebhooks("bookmark", minimalTweet);
            }
            logger.info(`[BookmarksInterceptor] Webhook triggered for bookmark on tweet ${tweetId}`);
          }
        }
      } catch (err) {
        logger.error("[BookmarksInterceptor] Failed to process CreateBookmark", err);
      }
      return;
    }
    if (!/\/graphql\/.+\/Bookmarks/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.bookmark_timeline_v2.timeline.instructions,
        (entry) => extractTimelineTweet(entry.content.itemContent)
      );
      await databaseManager.extAddTweets(ext.name, newData);
      logger.info(`Bookmarks: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("Bookmarks: Failed to parse API response", err);
    }
  };
  class BookmarksModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "BookmarksModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return BookmarksInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const CommunityMembersInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/(members|moderators)SliceTimeline_Query/.test(req.url)) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      const result = json.data.communityResults.result;
      const newData = (result.members_slice ?? result.moderators_slice).items_results.map((item) => item.result).filter((user) => user.__typename === "User");
      databaseManager.extAddUsers(ext.name, newData);
      logger.info(`CommunityMembers: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("CommunityMembers: Failed to parse API response", err);
    }
  };
  class CommunityMembersModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "CommunityMembersModule");
      __publicField(this, "type", ExtensionType.USER);
    }
    intercept() {
      return CommunityMembersInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const CommunityTimelineInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/Community(Tweets|Media)Timeline/.test(req.url)) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      const result = json.data.communityResults.result;
      const timeline = result.ranked_community_timeline ?? result.community_media_timeline;
      const instructions = timeline.timeline.instructions;
      const newData = [];
      const timelineAddEntriesInstruction = instructions.find(
        (i) => i.type === "TimelineAddEntries"
      );
      const timelineAddEntriesInstructionEntries = (timelineAddEntriesInstruction == null ? void 0 : timelineAddEntriesInstruction.entries) ?? [];
      for (const entry of timelineAddEntriesInstructionEntries) {
        if (isTimelineEntryItem(entry)) {
          const tweet = extractTimelineTweet(entry.content.itemContent);
          if (tweet) {
            newData.push(tweet);
          }
        }
        if (isTimelineEntryCommunitiesGrid(entry)) {
          const tweetsInGrid = entry.content.items.map((i) => extractTimelineTweet(i.item.itemContent)).filter((t) => !!t);
          newData.push(...tweetsInGrid);
        }
      }
      const timelineAddToModuleInstruction = instructions.find(
        (i) => i.type === "TimelineAddToModule"
      );
      if (timelineAddToModuleInstruction == null ? void 0 : timelineAddToModuleInstruction.moduleItems) {
        const tweets = timelineAddToModuleInstruction.moduleItems.map((i) => extractTimelineTweet(i.item.itemContent)).filter((t) => !!t);
        newData.push(...tweets);
      }
      databaseManager.extAddTweets(ext.name, newData);
      logger.info(`CommunityTimeline: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("CommunityTimeline: Failed to parse API response", err);
    }
  };
  class CommunityTimelineModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "CommunityTimelineModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return CommunityTimelineInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const messagesSignal = signals.signal([]);
  const conversationsCollection = /* @__PURE__ */ new Map();
  const usersCollection = /* @__PURE__ */ new Map();
  const strategies = [
    {
      test: (url) => /\/dm\/inbox_initial_state\.json/.test(url),
      parse: (json) => ({
        entries: json.inbox_initial_state.entries,
        conversations: Object.values(json.inbox_initial_state.conversations),
        users: Object.values(json.inbox_initial_state.users)
      })
    },
    {
      test: (url) => /\/dm\/inbox_timeline\/trusted\.json/.test(url),
      parse: (json) => ({
        entries: json.inbox_timeline.entries,
        conversations: Object.values(json.inbox_timeline.conversations),
        users: Object.values(json.inbox_timeline.users)
      })
    },
    {
      test: (url) => /\/dm\/conversation\/\d+-?\d+\.json/.test(url),
      parse: (json) => ({
        entries: json.conversation_timeline.entries,
        conversations: Object.values(json.conversation_timeline.conversations),
        users: Object.values(json.conversation_timeline.users)
      })
    }
  ];
  const DirectMessagesInterceptor = (req, res) => {
    const strategy = strategies.find((s) => s.test(req.url));
    if (!strategy) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      const { entries, conversations, users } = strategy.parse(json);
      const messages = entries.map((entry) => entry.message).filter((message) => !!message);
      messagesSignal.value = [...messagesSignal.value, ...messages];
      conversations.filter(Boolean).forEach((c) => conversationsCollection.set(c.conversation_id, c));
      users.filter(Boolean).forEach((user) => usersCollection.set(user.id_str, user));
      logger.info(`DirectMessages: ${messages.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("DirectMessages: Failed to parse API response", err);
    }
  };
  function getUserScreeNameFromId(id) {
    const user = id ? usersCollection.get(id) : null;
    return user ? user.screen_name : "";
  }
  function getConversationTypeFromId(id) {
    const conversation = id ? conversationsCollection.get(id) : null;
    return conversation ? conversation.type : "";
  }
  function extractMessageMedia(message) {
    var _a, _b, _c;
    return [
      (_a = message.message_data.attachment) == null ? void 0 : _a.photo,
      (_b = message.message_data.attachment) == null ? void 0 : _b.video,
      (_c = message.message_data.attachment) == null ? void 0 : _c.animated_gif
    ].filter((m) => !!m);
  }
  const columnHelper = tableCore.createColumnHelper();
  const columns = [
    columnHelper.display({
      id: "select",
      meta: { exportable: false },
      header: ({ table }) => /* @__PURE__ */ u(
        "input",
        {
          type: "checkbox",
          class: "checkbox checkbox-sm align-middle",
          checked: table.getIsAllRowsSelected(),
          indeterminate: table.getIsSomeRowsSelected(),
          onChange: table.getToggleAllRowsSelectedHandler()
        }
      ),
      cell: ({ row }) => /* @__PURE__ */ u(
        "input",
        {
          type: "checkbox",
          class: "checkbox checkbox-sm",
          checked: row.getIsSelected(),
          disabled: !row.getCanSelect(),
          indeterminate: row.getIsSomeSelected(),
          onChange: row.getToggleSelectedHandler()
        }
      )
    }),
    columnHelper.accessor("id", {
      meta: { exportKey: "id", exportHeader: "ID" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "ID" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-20 break-all font-mono text-xs", children: info.getValue() })
    }),
    columnHelper.accessor("time", {
      meta: {
        exportKey: "time",
        exportHeader: "Date",
        exportValue: (row) => formatDateTime(dayjs(+row.original.time), appOptionsManager.get("dateTimeFormat"))
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Date" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-24", children: formatDateTime(+info.getValue(), appOptionsManager.get("dateTimeFormat")) })
    }),
    columnHelper.accessor("message_data.text", {
      meta: {
        exportKey: "text",
        exportHeader: "Content"
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Content" }),
      cell: (info) => {
        var _a;
        return /* @__PURE__ */ u("div", { children: /* @__PURE__ */ u(
          "p",
          {
            class: "w-60 whitespace-pre-wrap",
            dangerouslySetInnerHTML: {
              __html: strEntitiesToHTML(
                info.row.original.message_data.text,
                (_a = info.row.original.message_data.entities) == null ? void 0 : _a.urls
              )
            }
          }
        ) });
      }
    }),
    columnHelper.accessor((row) => extractMessageMedia(row), {
      id: "media",
      meta: {
        exportKey: "media",
        exportHeader: "Media",
        exportValue: (row) => extractMessageMedia(row.original).map((media) => ({
          type: media.type,
          url: media.url,
          thumbnail: formatTwitterImage(media.media_url_https, "thumb"),
          original: getMediaOriginalUrl(media),
          ext_alt_text: media.ext_alt_text
        }))
      },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Media" }),
      cell: (info) => /* @__PURE__ */ u(
        MediaDisplayColumn,
        {
          data: info.getValue().filter((m) => !!m),
          onClick: (media) => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setMediaPreview(getMediaOriginalUrl(media));
          }
        }
      )
    }),
    columnHelper.accessor((row) => getUserScreeNameFromId(row.message_data.sender_id), {
      id: "sender",
      meta: { exportKey: "sender", exportHeader: "Sender" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Sender" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "whitespace-pre", children: /* @__PURE__ */ u("a", { class: "link", target: "_blank", href: `https://twitter.com/${info.getValue()}`, children: [
        "@",
        info.getValue()
      ] }) })
    }),
    columnHelper.accessor((row) => getUserScreeNameFromId(row.message_data.recipient_id), {
      id: "recipient",
      meta: { exportKey: "recipient", exportHeader: "Recipient" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Recipient" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "whitespace-pre", children: info.getValue() ? /* @__PURE__ */ u("a", { class: "link", target: "_blank", href: `https://twitter.com/${info.getValue()}`, children: [
        "@",
        info.getValue()
      ] }) : "N/A" })
    }),
    columnHelper.accessor("conversation_id", {
      meta: { exportKey: "conversation_id", exportHeader: "Conversation ID" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Conversation ID" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "w-20 break-all font-mono text-xs", children: info.getValue() })
    }),
    columnHelper.accessor((row) => getConversationTypeFromId(row.conversation_id), {
      id: "conversation_type",
      meta: { exportKey: "conversation_type", exportHeader: "Conversation Type" },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Conversation Type" }),
      cell: (info) => /* @__PURE__ */ u("p", { class: "whitespace-pre", children: info.getValue() })
    }),
    columnHelper.display({
      id: "actions",
      meta: { exportable: false },
      header: () => /* @__PURE__ */ u(Trans, { i18nKey: "Actions" }),
      cell: (info) => /* @__PURE__ */ u("div", { class: "flex flex-row items-start space-x-1", children: /* @__PURE__ */ u(
        "button",
        {
          onClick: () => {
            var _a;
            return (_a = info.table.options.meta) == null ? void 0 : _a.setRawDataPreview(info.row.original);
          },
          class: "btn btn-xs btn-neutral whitespace-nowrap",
          children: /* @__PURE__ */ u(Trans, { i18nKey: "Details" })
        }
      ) })
    })
  ];
  function DirectMessagesUI() {
    const { t } = useTranslation();
    const [showModal, toggleShowModal] = useToggle();
    const title = t("DirectMessages");
    const count = messagesSignal.value.length;
    return /* @__PURE__ */ u(
      ExtensionPanel,
      {
        title,
        description: `${t("Captured:")} ${count}`,
        active: !!count && count > 0,
        onClick: toggleShowModal,
        indicatorColor: "bg-accent",
        children: /* @__PURE__ */ u(
          Modal,
          {
            class: "max-w-4xl md:max-w-screen-md sm:max-w-screen-sm min-h-[512px]",
            title,
            show: showModal,
            onClose: toggleShowModal,
            children: /* @__PURE__ */ u(
              BaseTableView,
              {
                title,
                records: messagesSignal.value,
                columns,
                clear: () => messagesSignal.value = []
              }
            )
          }
        )
      }
    );
  }
  class DirectMessagesModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "DirectMessagesModule");
      __publicField(this, "type", ExtensionType.CUSTOM);
    }
    intercept() {
      return DirectMessagesInterceptor;
    }
    render() {
      return DirectMessagesUI;
    }
  }
  const FollowersInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/(BlueVerified)*Followers/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.user.result.timeline.timeline.instructions,
        (entry) => extractTimelineUser(entry.content.itemContent)
      );
      databaseManager.extAddUsers(ext.name, newData);
      logger.info(`Followers: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("Followers: Failed to parse API response", err);
    }
  };
  class FollowersModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "FollowersModule");
      __publicField(this, "type", ExtensionType.USER);
    }
    intercept() {
      return FollowersInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const FollowingInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/Following/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.user.result.timeline.timeline.instructions,
        (entry) => extractTimelineUser(entry.content.itemContent)
      );
      databaseManager.extAddUsers(ext.name, newData);
      logger.info(`Following: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("Following: Failed to parse API response", err);
    }
  };
  class FollowingModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "FollowingModule");
      __publicField(this, "type", ExtensionType.USER);
    }
    intercept() {
      return FollowingInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const HomeTimelineInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/Home(Latest)?Timeline/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.home.home_timeline_urt.instructions,
        (entry) => extractTimelineTweet(entry.content.itemContent)
      );
      databaseManager.extAddTweets(ext.name, newData);
      logger.info(`HomeTimeline: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("HomeTimeline: Failed to parse API response", err);
    }
  };
  class HomeTimelineModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "HomeTimelineModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return HomeTimelineInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const LikesInterceptor = async (req, res, ext) => {
    var _a, _b;
    if (/\/graphql\/.+\/FavoriteTweet/.test(req.url)) {
      logger.info(`[LikesInterceptor] FavoriteTweet API detected`);
      try {
        const json = JSON.parse(res.responseText);
        if (((_a = json.data) == null ? void 0 : _a.favorite_tweet) === "Done" && req.body) {
          const requestBody = JSON.parse(req.body);
          const tweetId = (_b = requestBody.variables) == null ? void 0 : _b.tweet_id;
          if (tweetId) {
            const fullTweet = await databaseManager.getTweetById(tweetId);
            if (fullTweet) {
              logger.info(`[LikesInterceptor] Found full tweet data for ${tweetId}`);
              await webhookManager.triggerWebhooks("like", fullTweet);
            } else {
              logger.info(`[LikesInterceptor] Tweet ${tweetId} not in database, using minimal data`);
              const minimalTweet = { rest_id: tweetId };
              await webhookManager.triggerWebhooks("like", minimalTweet);
            }
            logger.info(`[LikesInterceptor] Webhook triggered for like on tweet ${tweetId}`);
          }
        }
      } catch (err) {
        logger.error("[LikesInterceptor] Failed to process FavoriteTweet", err);
      }
      return;
    }
    if (!/\/graphql\/.+\/Likes/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.user.result.timeline.timeline.instructions,
        (entry) => extractTimelineTweet(entry.content.itemContent)
      );
      await databaseManager.extAddTweets(ext.name, newData);
      logger.info(`Likes: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("Likes: Failed to parse API response", err);
    }
  };
  class LikesModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "LikesModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return LikesInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const ListMembersInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/ListMembers/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.list.members_timeline.timeline.instructions,
        (entry) => extractTimelineUser(entry.content.itemContent)
      );
      databaseManager.extAddUsers(ext.name, newData);
      logger.info(`ListMembers: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("ListMembers: Failed to parse API response", err);
    }
  };
  class ListMembersModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "ListMembersModule");
      __publicField(this, "type", ExtensionType.USER);
    }
    intercept() {
      return ListMembersInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const ListSubscribersInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/ListSubscribers/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.list.subscribers_timeline.timeline.instructions,
        (entry) => extractTimelineUser(entry.content.itemContent)
      );
      databaseManager.extAddUsers(ext.name, newData);
      logger.info(`ListSubscribers: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("ListSubscribers: Failed to parse API response", err);
    }
  };
  class ListSubscribersModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "ListSubscribersModule");
      __publicField(this, "type", ExtensionType.USER);
    }
    intercept() {
      return ListSubscribersInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const ListTimelineInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/ListLatestTweetsTimeline/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.list.tweets_timeline.timeline.instructions,
        (entry) => extractTimelineTweet(entry.content.itemContent)
      );
      databaseManager.extAddTweets(ext.name, newData);
      logger.info(`ListTimeline: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("ListTimeline: Failed to parse API response", err);
    }
  };
  class ListTimelineModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "ListTimelineModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return ListTimelineInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const RetweetersInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/Retweeters/.test(req.url)) {
      return;
    }
    try {
      const newData = extractDataFromResponse(
        res,
        (json) => json.data.retweeters_timeline.timeline.instructions,
        (entry) => extractTimelineUser(entry.content.itemContent)
      );
      databaseManager.extAddUsers(ext.name, newData);
      logger.info(`Retweeters: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("Retweeters: Failed to parse API response", err);
    }
  };
  class RetweetersModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "RetweetersModule");
      __publicField(this, "type", ExtensionType.USER);
    }
    intercept() {
      return RetweetersInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const colors = {
    info: "text-base-content",
    warn: "text-warning",
    error: "text-error"
  };
  function Logs({ lines }) {
    const reversed = lines.value.slice().reverse();
    return /* @__PURE__ */ u("pre", { class: "leading-none text-xs max-h-48 bg-base-200 overflow-y-scroll m-0 px-1 py-2.5 no-scrollbar rounded-box-half", children: reversed.map((line) => /* @__PURE__ */ u("span", { class: colors[line.type], children: [
      "#",
      line.index,
      " ",
      line.line,
      "\n"
    ] }, line.index)) });
  }
  function RuntimeLogsPanel() {
    return /* @__PURE__ */ u(preact.Fragment, { children: [
      /* @__PURE__ */ u("div", { class: "divider mt-0 mb-1" }),
      /* @__PURE__ */ u(Logs, { lines: logLinesSignal })
    ] });
  }
  class RuntimeLogsModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "RuntimeLogsModule");
    }
    render() {
      return RuntimeLogsPanel;
    }
  }
  const SearchTimelineInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/SearchTimeline/.test(req.url)) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      const instructions = json.data.search_by_raw_query.search_timeline.timeline.instructions;
      const newTweets = [];
      const newUsers = [];
      const newLists = [];
      const timelineAddEntriesInstruction = instructions.find(
        (i) => i.type === "TimelineAddEntries"
      );
      const timelineAddToModuleInstruction = instructions.find(
        (i) => i.type === "TimelineAddToModule"
      );
      const timelineAddEntriesInstructionEntries = (timelineAddEntriesInstruction == null ? void 0 : timelineAddEntriesInstruction.entries) ?? [];
      for (const entry of timelineAddEntriesInstructionEntries) {
        if (isTimelineEntryTweet(entry)) {
          const tweet = extractTimelineTweet(entry.content.itemContent);
          if (tweet) {
            newTweets.push(tweet);
          }
        }
        if (isTimelineEntrySearchGrid(entry)) {
          const tweetsInSearchGrid = entry.content.items.map((i) => extractTimelineTweet(i.item.itemContent)).filter((t) => !!t);
          newTweets.push(...tweetsInSearchGrid);
        }
        if (isTimelineEntryUser(entry)) {
          const user = extractTimelineUser(entry.content.itemContent);
          if (user) {
            newUsers.push(user);
          }
        }
        if (isTimelineEntryListSearch(entry)) {
          const lists = entry.content.items.map((i) => i.item.itemContent.list);
          newLists.push(...lists);
        }
      }
      if (timelineAddToModuleInstruction) {
        const items = timelineAddToModuleInstruction.moduleItems.map((i) => i.item.itemContent);
        const tweets = items.filter((i) => i.__typename === "TimelineTweet").map((t) => extractTimelineTweet(t)).filter((t) => !!t);
        newTweets.push(...tweets);
        const lists = items.filter((i) => i.__typename === "TimelineTwitterList").map((i) => i.list);
        newLists.push(...lists);
      }
      databaseManager.extAddTweets(ext.name, newTweets);
      logger.info(`SearchTimeline: ${newTweets.length} items received`);
      if (newLists.length > 0) {
        logger.warn(
          `SearchList: ${newLists.length} lists received but ignored (Reason: not implemented)`,
          newLists
        );
      }
      if (newUsers.length > 0) {
        logger.warn(
          `SearchUser: ${newUsers.length} users received but ignored (Reason: not implemented)`,
          newUsers
        );
      }
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("SearchTimeline: Failed to parse API response", err);
    }
  };
  class SearchTimelineModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "SearchTimelineModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return SearchTimelineInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const TweetDetailInterceptor = (req, res, ext) => {
    const isTweetDetail = /\/graphql\/.+\/TweetDetail/.test(req.url);
    const isModeratedTimeline = /\/graphql\/.+\/ModeratedTimeline/.test(req.url);
    if (!isTweetDetail && !isModeratedTimeline) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      let instructions = [];
      if (isTweetDetail) {
        instructions = json.data.threaded_conversation_with_injections_v2.instructions;
      } else if (isModeratedTimeline) {
        instructions = json.data.tweet.result.timeline_response.timeline.instructions;
      }
      const newData = [];
      const timelineAddEntriesInstruction = instructions.find(
        (i) => i.type === "TimelineAddEntries"
      );
      const timelineAddEntriesInstructionEntries = (timelineAddEntriesInstruction == null ? void 0 : timelineAddEntriesInstruction.entries) ?? [];
      for (const entry of timelineAddEntriesInstructionEntries) {
        if (isTimelineEntryTweet(entry)) {
          const tweet = extractTimelineTweet(entry.content.itemContent);
          if (tweet) {
            newData.push(tweet);
          }
        }
        if (isTweetDetail && isTimelineEntryConversationThread(entry)) {
          const tweetsInConversation = entry.content.items.map((i) => {
            if (i.entryId.includes("-tweet-")) {
              return extractTimelineTweet(i.item.itemContent);
            }
          });
          newData.push(...tweetsInConversation.filter((t) => !!t));
        }
      }
      const timelineAddToModuleInstruction = instructions.find(
        (i) => i.type === "TimelineAddToModule"
      );
      if (timelineAddToModuleInstruction) {
        const tweetsInConversation = timelineAddToModuleInstruction.moduleItems.map((i) => extractTimelineTweet(i.item.itemContent)).filter((t) => !!t);
        newData.push(...tweetsInConversation);
      }
      databaseManager.extAddTweets(ext.name, newData);
      logger.info(`TweetDetail: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("TweetDetail: Failed to parse API response", err);
    }
  };
  class TweetDetailModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "TweetDetailModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return TweetDetailInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const UserDetailInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/UserByScreenName/.test(req.url)) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      const newData = [json.data.user.result];
      databaseManager.extAddUsers(ext.name, newData);
      logger.info(`UserDetail: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("UserDetail: Failed to parse API response", err);
    }
  };
  class UserDetailModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "UserDetailModule");
      __publicField(this, "type", ExtensionType.USER);
    }
    intercept() {
      return UserDetailInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const UserMediaInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/UserMedia/.test(req.url)) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      const instructions = json.data.user.result.timeline.timeline.instructions;
      const newData = [];
      const timelineAddEntriesInstruction = instructions.find(
        (i) => i.type === "TimelineAddEntries"
      );
      const timelineAddEntriesInstructionEntries = (timelineAddEntriesInstruction == null ? void 0 : timelineAddEntriesInstruction.entries) ?? [];
      for (const entry of timelineAddEntriesInstructionEntries) {
        if (isTimelineEntryProfileGrid(entry)) {
          const tweetsInSearchGrid = entry.content.items.map((i) => extractTimelineTweet(i.item.itemContent)).filter((t) => !!t);
          newData.push(...tweetsInSearchGrid);
        }
      }
      const timelineAddToModuleInstruction = instructions.find(
        (i) => i.type === "TimelineAddToModule"
      );
      if (timelineAddToModuleInstruction) {
        const tweetsInProfileGrid = timelineAddToModuleInstruction.moduleItems.map((i) => extractTimelineTweet(i.item.itemContent)).filter((t) => !!t);
        newData.push(...tweetsInProfileGrid);
      }
      databaseManager.extAddTweets(ext.name, newData);
      logger.info(`UserMedia: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("UserMedia: Failed to parse API response", err);
    }
  };
  class UserMediaModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "UserMediaModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return UserMediaInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const UserTweetsInterceptor = (req, res, ext) => {
    if (!/\/graphql\/.+\/UserTweets/.test(req.url)) {
      return;
    }
    try {
      const json = JSON.parse(res.responseText);
      const instructions = json.data.user.result.timeline.timeline.instructions;
      const newData = [];
      const timelinePinEntryInstruction = instructions.find(
        (i) => i.type === "TimelinePinEntry"
      );
      if (timelinePinEntryInstruction) {
        const tweet = extractTimelineTweet(timelinePinEntryInstruction.entry.content.itemContent);
        if (tweet) {
          newData.push(tweet);
        }
      }
      const timelineAddEntriesInstruction = instructions.find(
        (i) => i.type === "TimelineAddEntries"
      );
      const timelineAddEntriesInstructionEntries = (timelineAddEntriesInstruction == null ? void 0 : timelineAddEntriesInstruction.entries) ?? [];
      for (const entry of timelineAddEntriesInstructionEntries) {
        if (isTimelineEntryTweet(entry)) {
          const tweet = extractTimelineTweet(entry.content.itemContent);
          if (tweet) {
            newData.push(tweet);
          }
        }
        if (isTimelineEntryProfileConversation(entry)) {
          const tweetsInConversation = entry.content.items.map((i) => extractTimelineTweet(i.item.itemContent)).filter((t) => !!t);
          newData.push(...tweetsInConversation);
        }
      }
      databaseManager.extAddTweets(ext.name, newData);
      logger.info(`UserTweets: ${newData.length} items received`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status, res.responseText);
      logger.errorWithBanner("UserTweets: Failed to parse API response", err);
    }
  };
  class UserTweetsModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "UserTweetsModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return UserTweetsInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  const TWEET_DISPLAY_ENDPOINTS = [
    /\/graphql\/.+\/HomeLatestTimeline/,
    /\/graphql\/.+\/HomeTimeline/,
    /\/graphql\/.+\/TweetDetail/,
    /\/graphql\/.+\/UserTweets/,
    /\/graphql\/.+\/UserMedia/,
    /\/graphql\/.+\/SearchTimeline/,
    /\/graphql\/.+\/Bookmarks/,
    /\/graphql\/.+\/Likes/,
    /\/graphql\/.+\/ListLatestTweetsTimeline/
  ];
  function detectSourcePage(url) {
    if (/HomeLatestTimeline|HomeTimeline/.test(url)) return "home";
    if (/UserTweets|UserMedia/.test(url)) return "profile";
    if (/TweetDetail/.test(url)) return "detail";
    if (/SearchTimeline/.test(url)) return "search";
    if (/Bookmarks/.test(url)) return "bookmarks";
    if (/Likes/.test(url)) return "likes";
    if (/ListLatestTweetsTimeline/.test(url)) return "list";
    return "unknown";
  }
  function extractTweetsFromResponse(res, url) {
    var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o, _p, _q, _r, _s, _t, _u, _v, _w, _x, _y, _z;
    try {
      const json = JSON.parse(res.responseText);
      const sourcePage = detectSourcePage(url);
      if (sourcePage === "detail") {
        const response = json;
        const tweet = (_b = (_a = response == null ? void 0 : response.data) == null ? void 0 : _a.tweetResult) == null ? void 0 : _b.result;
        if (tweet && tweet.rest_id) {
          return [tweet];
        }
        return [];
      }
      let instructions;
      if (sourcePage === "home") {
        instructions = (_e = (_d = (_c = json == null ? void 0 : json.data) == null ? void 0 : _c.home) == null ? void 0 : _d.home_timeline_urt) == null ? void 0 : _e.instructions;
      } else if (sourcePage === "profile") {
        instructions = (_j = (_i = (_h = (_g = (_f = json == null ? void 0 : json.data) == null ? void 0 : _f.user) == null ? void 0 : _g.result) == null ? void 0 : _h.timeline) == null ? void 0 : _i.timeline) == null ? void 0 : _j.instructions;
      } else if (sourcePage === "search") {
        instructions = (_n = (_m = (_l = (_k = json == null ? void 0 : json.data) == null ? void 0 : _k.search_by_raw_query) == null ? void 0 : _l.search_timeline) == null ? void 0 : _m.timeline) == null ? void 0 : _n.instructions;
      } else if (sourcePage === "bookmarks") {
        instructions = (_q = (_p = (_o = json == null ? void 0 : json.data) == null ? void 0 : _o.bookmark_timeline_v2) == null ? void 0 : _p.timeline) == null ? void 0 : _q.instructions;
      } else if (sourcePage === "likes") {
        instructions = (_v = (_u = (_t = (_s = (_r = json == null ? void 0 : json.data) == null ? void 0 : _r.user) == null ? void 0 : _s.result) == null ? void 0 : _t.timeline) == null ? void 0 : _u.timeline) == null ? void 0 : _v.instructions;
      } else if (sourcePage === "list") {
        instructions = (_z = (_y = (_x = (_w = json == null ? void 0 : json.data) == null ? void 0 : _w.list) == null ? void 0 : _x.tweets_timeline) == null ? void 0 : _y.timeline) == null ? void 0 : _z.instructions;
      }
      if (!instructions) {
        return [];
      }
      return extractDataFromResponse(
        res,
        () => instructions,
        (entry) => extractTimelineTweet(entry.content.itemContent)
      );
    } catch {
      return [];
    }
  }
  const BrowsingHistoryInterceptor = async (req, res, ext) => {
    if (!TWEET_DISPLAY_ENDPOINTS.some((pattern) => pattern.test(req.url))) {
      return;
    }
    try {
      const tweets = extractTweetsFromResponse(res, req.url);
      if (tweets.length === 0) {
        return;
      }
      const sourcePage = detectSourcePage(req.url);
      const now = Date.now();
      const currentUrl = window.location.href;
      const recentHistories = await databaseManager.getRecentBrowsingHistories(DEFAULT_DEDUP_WINDOW_MS);
      const recentTweetIds = new Set((recentHistories || []).map((h2) => h2.tweet_id));
      const newTweets = tweets.filter((tweet) => !recentTweetIds.has(tweet.rest_id));
      if (newTweets.length === 0) {
        return;
      }
      const histories = newTweets.map((tweet) => ({
        id: generateBrowsingHistoryId(tweet.rest_id, now),
        tweet_id: tweet.rest_id,
        viewed_at: now,
        source_page: sourcePage,
        url: currentUrl
      }));
      await databaseManager.addBrowsingHistories(histories);
      await databaseManager.extAddTweets(ext.name, newTweets);
      await webhookManager.triggerWebhooksBatch("view", newTweets);
      logger.info(`BrowsingHistory: ${newTweets.length} tweets recorded from ${sourcePage}`);
    } catch (err) {
      logger.debug(req.method, req.url, res.status);
      logger.errorWithBanner("BrowsingHistory: Failed to process response", err);
    }
  };
  class BrowsingHistoryModule extends Extension {
    constructor() {
      super(...arguments);
      __publicField(this, "name", "BrowsingHistoryModule");
      __publicField(this, "type", ExtensionType.TWEET);
    }
    intercept() {
      return BrowsingHistoryInterceptor;
    }
    render() {
      return CommonModuleUI;
    }
  }
  extensionManager.add(FollowersModule);
  extensionManager.add(FollowingModule);
  extensionManager.add(UserDetailModule);
  extensionManager.add(ListMembersModule);
  extensionManager.add(ListSubscribersModule);
  extensionManager.add(CommunityMembersModule);
  extensionManager.add(RetweetersModule);
  extensionManager.add(HomeTimelineModule);
  extensionManager.add(ListTimelineModule);
  extensionManager.add(CommunityTimelineModule);
  extensionManager.add(BookmarksModule);
  extensionManager.add(LikesModule);
  extensionManager.add(UserTweetsModule);
  extensionManager.add(UserMediaModule);
  extensionManager.add(TweetDetailModule);
  extensionManager.add(SearchTimelineModule);
  extensionManager.add(BrowsingHistoryModule);
  extensionManager.add(DirectMessagesModule);
  extensionManager.add(RuntimeLogsModule);
  extensionManager.start();
  webhookManager.startQueue();
  function mountApp() {
    const root = document.createElement("div");
    root.id = "memoizex-root";
    document.body.append(root);
    preact.render(/* @__PURE__ */ u(App, {}), root);
  }
  if (document.readyState === "loading") {
    document.addEventListener("DOMContentLoaded", mountApp);
  } else {
    mountApp();
  }

})(preact, preactHooks, preactSignals, i18next, dayjs, Dexie, DexieExportImport, FileSaver, TableCore);