Chore: Port over Extensions & Search

This commit is contained in:
Youwes09
2026-05-31 00:30:36 -05:00
parent 6de5207ce7
commit 13f2a483ca
47 changed files with 6086 additions and 1016 deletions
@@ -0,0 +1,588 @@
<script lang="ts">
import { X, CircleNotch, CaretUpDown, Check, CaretLeft, CaretRight } from "phosphor-svelte";
import Thumbnail from "$lib/components/shared/manga/Thumbnail.svelte";
import { getAdapter } from "$lib/request-manager";
import { addToast } from "$lib/state/notifications.svelte";
interface Preference {
type: string;
key: string;
CheckBoxTitle?: string;
CheckBoxSummary?: string;
CheckBoxDefault?: boolean;
CheckBoxCurrentValue?: boolean;
SwitchPreferenceTitle?: string;
SwitchPreferenceSummary?: string;
SwitchPreferenceDefault?: boolean;
SwitchPreferenceCurrentValue?: boolean;
ListPreferenceTitle?: string;
ListPreferenceSummary?: string;
ListPreferenceDefault?: string;
ListPreferenceCurrentValue?: string;
entries?: string[];
entryValues?: string[];
EditTextPreferenceTitle?: string;
EditTextPreferenceSummary?: string;
EditTextPreferenceDefault?: string;
EditTextPreferenceCurrentValue?: string;
dialogTitle?: string;
dialogMessage?: string;
MultiSelectListPreferenceTitle?: string;
MultiSelectListPreferenceSummary?: string;
MultiSelectListPreferenceDefault?: string[];
MultiSelectListPreferenceCurrentValue?: string[];
}
export type SourceEntry = { id: string; displayName: string };
interface Props {
extensionName: string;
iconUrl: string;
sources: SourceEntry[];
onClose: () => void;
}
let { extensionName, iconUrl, sources, onClose }: Props = $props();
let activeIndex = $state(sources.length === 1 ? 0 : -1);
let prefs = $state<Preference[]>([]);
let loading = $state(false);
let saving = $state<string | null>(null);
let editKey = $state<string | null>(null);
let editValue = $state("");
let listOpen = $state<string | null>(null);
let closing = $state(false);
const activeSource = $derived(activeIndex >= 0 ? sources[activeIndex] : null);
const inPicker = $derived(sources.length > 1 && activeIndex < 0);
$effect(() => {
if (activeSource) loadPrefs(activeSource);
});
async function loadPrefs(src: SourceEntry) {
loading = true;
prefs = [];
editKey = null;
listOpen = null;
try {
const d = await (getAdapter() as any).gql<{ source: { preferences: Preference[] } }>(
`query GetSourceSettings($id: LongString!) { source(id: $id) { preferences {
... on CheckBoxPreference { type: __typename CheckBoxTitle: title CheckBoxSummary: summary CheckBoxDefault: default CheckBoxCurrentValue: currentValue key }
... on SwitchPreference { type: __typename SwitchPreferenceTitle: title SwitchPreferenceSummary: summary SwitchPreferenceDefault: default SwitchPreferenceCurrentValue: currentValue key }
... on ListPreference { type: __typename ListPreferenceTitle: title ListPreferenceSummary: summary ListPreferenceDefault: default ListPreferenceCurrentValue: currentValue entries entryValues key }
... on EditTextPreference { type: __typename EditTextPreferenceTitle: title EditTextPreferenceSummary: summary EditTextPreferenceDefault: default EditTextPreferenceCurrentValue: currentValue dialogTitle dialogMessage key }
... on MultiSelectListPreference { type: __typename MultiSelectListPreferenceTitle: title MultiSelectListPreferenceSummary: summary MultiSelectListPreferenceDefault: default MultiSelectListPreferenceCurrentValue: currentValue entries entryValues key }
} } }`,
{ id: String(src.id) },
);
prefs = d.source.preferences ?? [];
} catch (e: any) {
addToast({ kind: "error", title: "Failed to load settings", body: e?.message ?? "" });
} finally {
loading = false;
}
}
async function save(position: number, changeType: string, value: unknown) {
if (!activeSource) return;
const pref = prefs[position];
saving = pref.key;
try {
await (getAdapter() as any).gql(
`mutation UpdateSourcePreference($source: LongString!, $change: SourcePreferenceChangeInput!) { updateSourcePreference(input: { source: $source, change: $change }) { source { id } } }`,
{ source: String(activeSource.id), change: { position, [changeType]: value } },
);
const d = await (getAdapter() as any).gql<{ source: { preferences: Preference[] } }>(
`query GetSourceSettings($id: LongString!) { source(id: $id) { preferences {
... on CheckBoxPreference { type: __typename CheckBoxTitle: title CheckBoxCurrentValue: currentValue key }
... on SwitchPreference { type: __typename SwitchPreferenceTitle: title SwitchPreferenceCurrentValue: currentValue key }
... on ListPreference { type: __typename ListPreferenceTitle: title ListPreferenceCurrentValue: currentValue entries entryValues key }
... on EditTextPreference { type: __typename EditTextPreferenceTitle: title EditTextPreferenceCurrentValue: currentValue key }
... on MultiSelectListPreference { type: __typename MultiSelectListPreferenceTitle: title MultiSelectListPreferenceCurrentValue: currentValue entries entryValues key }
} } }`,
{ id: String(activeSource.id) },
);
prefs = d.source.preferences ?? [];
} catch (e: any) {
addToast({ kind: "error", title: "Failed to save", body: e?.message ?? "" });
} finally {
saving = null;
}
}
function getTitle(p: Preference) {
return p.CheckBoxTitle ?? p.SwitchPreferenceTitle ?? p.ListPreferenceTitle
?? p.EditTextPreferenceTitle ?? p.MultiSelectListPreferenceTitle ?? p.key;
}
function getSummary(p: Preference) {
return p.CheckBoxSummary ?? p.SwitchPreferenceSummary ?? p.ListPreferenceSummary
?? p.EditTextPreferenceSummary ?? p.MultiSelectListPreferenceSummary ?? null;
}
function getBoolValue(p: Preference) {
return p.type === "CheckBoxPreference"
? (p.CheckBoxCurrentValue ?? p.CheckBoxDefault ?? false)
: (p.SwitchPreferenceCurrentValue ?? p.SwitchPreferenceDefault ?? false);
}
function getListValue(p: Preference) { return p.ListPreferenceCurrentValue ?? p.ListPreferenceDefault ?? ""; }
function getListLabel(p: Preference, val: string) {
const idx = p.entryValues?.indexOf(val) ?? -1;
return idx >= 0 ? (p.entries?.[idx] ?? val) : val;
}
function getMultiValue(p: Preference): string[] {
return p.MultiSelectListPreferenceCurrentValue ?? p.MultiSelectListPreferenceDefault ?? [];
}
function toggleMulti(pos: number, p: Preference, val: string) {
const curr = getMultiValue(p);
save(pos, "multiSelectState", curr.includes(val) ? curr.filter(v => v !== val) : [...curr, val]);
}
function openEdit(p: Preference) {
editKey = p.key;
editValue = p.EditTextPreferenceCurrentValue ?? p.EditTextPreferenceDefault ?? "";
}
function submitEdit(pos: number) { save(pos, "editTextState", editValue); editKey = null; }
function langTag(name: string) {
const m = name.match(/\(([^)]+)\)$/);
return m ? m[1].toUpperCase() : null;
}
function baseName(name: string) { return name.replace(/\s*\([^)]+\)$/, ""); }
function prevSource() { if (activeIndex > 0) activeIndex--; }
function nextSource() { if (activeIndex < sources.length - 1) activeIndex++; }
function dismiss() {
closing = true;
setTimeout(onClose, 200);
}
function onKeydown(e: KeyboardEvent) {
if (e.key === "Escape") {
if (editKey) { editKey = null; return; }
if (listOpen) { listOpen = null; return; }
if (sources.length > 1 && activeIndex >= 0) { activeIndex = -1; return; }
dismiss();
}
}
function onBackdrop(e: MouseEvent) { if (e.target === e.currentTarget) dismiss(); }
</script>
<svelte:window onkeydown={onKeydown} />
<div class="backdrop" class:backdrop-out={closing} role="dialog" aria-modal="true" onmousedown={onBackdrop}>
<aside class="panel" class:panel-out={closing}>
<div class="panel-header">
<div class="ext-identity">
{#if iconUrl}
<Thumbnail src={iconUrl} alt={extensionName} class="ext-icon"
onerror={(e) => ((e.target as HTMLImageElement).style.display = "none")} />
{/if}
<div class="ext-titles">
<span class="ext-eyebrow">Extension Settings</span>
<span class="ext-name">{extensionName}</span>
</div>
</div>
<button class="close-btn" onclick={dismiss} aria-label="Close">
<X size={13} weight="bold" />
</button>
</div>
{#if sources.length > 1}
<div class="source-bar">
<button class="src-arrow" onclick={prevSource}
disabled={activeIndex <= 0} aria-label="Previous source">
<CaretLeft size={11} weight="bold" />
</button>
<div class="src-label">
{#if activeIndex >= 0}
<span class="src-name">{baseName(sources[activeIndex].displayName)}</span>
{#if langTag(sources[activeIndex].displayName)}
<span class="src-lang">{langTag(sources[activeIndex].displayName)}</span>
{/if}
{:else}
<span class="src-name src-dim">Choose a source</span>
{/if}
</div>
<button class="src-arrow" onclick={nextSource}
disabled={activeIndex >= sources.length - 1} aria-label="Next source">
<CaretRight size={11} weight="bold" />
</button>
</div>
{#if inPicker}
<div class="picker">
{#each sources as src, i}
{@const tag = langTag(src.displayName)}
<button class="picker-row" onclick={() => activeIndex = i}>
<span class="picker-name">{baseName(src.displayName)}</span>
{#if tag}<span class="src-lang">{tag}</span>{/if}
<CaretRight size={11} weight="bold" class="picker-caret" />
</button>
{/each}
</div>
{/if}
{/if}
{#if !inPicker}
<div class="prefs-body">
{#if loading}
<div class="center-state">
<CircleNotch size={15} weight="light" class="anim-spin" style="color:var(--text-faint)" />
</div>
{:else if prefs.length === 0}
<div class="center-state dim">No configurable settings.</div>
{:else}
<ul class="pref-list">
{#each prefs as pref, i}
{@const title = getTitle(pref)}
{@const summary = getSummary(pref)}
{@const isSaving = saving === pref.key}
{#if pref.type === "CheckBoxPreference" || pref.type === "SwitchPreference"}
{@const checked = getBoolValue(pref)}
<li class="pref-row">
<div class="pref-text">
<span class="pref-title">{title}</span>
{#if summary}<span class="pref-summary">{summary}</span>{/if}
</div>
<button
class="toggle" class:toggle-on={checked}
disabled={isSaving}
role="switch" aria-checked={checked}
onclick={() => save(i, pref.type === "CheckBoxPreference" ? "checkBoxState" : "switchState", !checked)}
>
{#if isSaving}
<CircleNotch size={9} weight="light" class="anim-spin" />
{:else}
<span class="toggle-thumb"></span>
{/if}
</button>
</li>
{:else if pref.type === "ListPreference"}
{@const current = getListValue(pref)}
<li class="pref-row pref-col">
<div class="pref-text">
<span class="pref-title">{title}</span>
{#if summary}<span class="pref-summary">{summary}</span>{/if}
</div>
<div class="select-wrap">
<button
class="select-btn" class:select-open={listOpen === pref.key}
disabled={isSaving}
onclick={() => listOpen = listOpen === pref.key ? null : pref.key}
>
<span class="select-val">{getListLabel(pref, current)}</span>
{#if isSaving}
<CircleNotch size={10} weight="light" class="anim-spin" />
{:else}
<CaretUpDown size={10} weight="bold" />
{/if}
</button>
{#if listOpen === pref.key}
<div class="dropdown">
{#each (pref.entries ?? []) as entry, j}
{@const val = pref.entryValues?.[j] ?? entry}
<button
class="dropdown-item" class:dropdown-item-active={val === current}
onclick={() => { save(i, "listState", val); listOpen = null; }}
>
{entry}
{#if val === current}<Check size={10} weight="bold" />{/if}
</button>
{/each}
</div>
{/if}
</div>
</li>
{:else if pref.type === "EditTextPreference"}
{#if editKey === pref.key}
<li class="pref-row pref-col edit-active">
<div class="pref-text">
{#if pref.dialogTitle}<span class="pref-title">{pref.dialogTitle}</span>{/if}
{#if pref.dialogMessage}<span class="pref-summary">{pref.dialogMessage}</span>{/if}
</div>
<div class="edit-row">
<input class="edit-input" bind:value={editValue} disabled={isSaving} autofocus
onkeydown={(e) => { if (e.key === "Enter") submitEdit(i); if (e.key === "Escape") editKey = null; }} />
<button class="action-dim" onclick={() => editKey = null}>Cancel</button>
<button class="action-btn" onclick={() => submitEdit(i)} disabled={isSaving}>
{#if isSaving}<CircleNotch size={10} weight="light" class="anim-spin" />{:else}Save{/if}
</button>
</div>
</li>
{:else}
<li>
<button class="pref-row pref-row-btn" onclick={() => openEdit(pref)}>
<div class="pref-text">
<span class="pref-title">{title}</span>
{#if summary}<span class="pref-summary">{summary}</span>{/if}
</div>
<span class="pref-value-hint">
{pref.EditTextPreferenceCurrentValue ?? pref.EditTextPreferenceDefault ?? "—"}
</span>
</button>
</li>
{/if}
{:else if pref.type === "MultiSelectListPreference"}
{@const selected = getMultiValue(pref)}
<li class="pref-row pref-col">
<div class="pref-text">
<span class="pref-title">{title}</span>
{#if summary}<span class="pref-summary">{summary}</span>{/if}
</div>
<div class="multi-list">
{#each (pref.entries ?? []) as entry, j}
{@const val = pref.entryValues?.[j] ?? entry}
{@const on = selected.includes(val)}
<button class="multi-item" class:multi-item-on={on}
disabled={isSaving} onclick={() => toggleMulti(i, pref, val)}>
<span class="multi-check">{#if on}<Check size={9} weight="bold" />{/if}</span>
{entry}
</button>
{/each}
</div>
</li>
{/if}
{/each}
</ul>
{/if}
</div>
{/if}
</aside>
</div>
<style>
.backdrop {
position: fixed; inset: 0;
background: rgba(0,0,0,0.28);
backdrop-filter: blur(2px);
z-index: var(--z-modal);
animation: fadeIn 0.15s ease both;
}
.backdrop-out { animation: fadeOut 0.2s ease forwards; pointer-events: none; }
@keyframes fadeIn { from { opacity: 0 } to { opacity: 1 } }
@keyframes fadeOut { from { opacity: 1 } to { opacity: 0 } }
.panel {
position: absolute; inset-block: 0; right: 0;
width: 320px; max-width: 100vw;
display: flex; flex-direction: column;
background: var(--bg-surface);
border-left: 1px solid var(--border-dim);
box-shadow: -12px 0 40px rgba(0,0,0,0.35);
animation: slideIn 0.2s cubic-bezier(0.16,1,0.3,1) both;
overflow: clip;
}
.panel-out { animation: slideOut 0.2s cubic-bezier(0.4,0,1,1) forwards; }
@keyframes slideIn { from { transform: translateX(100%) } to { transform: translateX(0) } }
@keyframes slideOut { from { transform: translateX(0) } to { transform: translateX(100%) } }
.panel-header {
display: flex; align-items: center; justify-content: space-between;
gap: var(--sp-3);
padding: var(--sp-4) var(--sp-3) var(--sp-4) var(--sp-5);
border-bottom: 1px solid var(--border-dim);
flex-shrink: 0;
}
.ext-identity { display: flex; align-items: center; gap: var(--sp-3); min-width: 0; }
:global(.ext-icon) {
width: 26px; height: 26px;
border-radius: var(--radius-md); object-fit: cover;
flex-shrink: 0; background: var(--bg-raised);
}
.ext-titles { display: flex; flex-direction: column; gap: 1px; min-width: 0; }
.ext-eyebrow {
font-family: var(--font-ui); font-size: var(--text-2xs);
color: var(--text-faint); letter-spacing: var(--tracking-wider); text-transform: uppercase;
}
.ext-name {
font-size: var(--text-sm); font-weight: var(--weight-medium); color: var(--text-primary);
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.close-btn {
display: flex; align-items: center; justify-content: center;
width: 24px; height: 24px; flex-shrink: 0;
border-radius: var(--radius-md); color: var(--text-faint);
transition: color var(--t-base), background var(--t-base);
}
.close-btn:hover { color: var(--text-primary); background: var(--bg-overlay); }
.source-bar {
display: flex; align-items: center; gap: var(--sp-1);
padding: var(--sp-2) var(--sp-3);
border-bottom: 1px solid var(--border-dim);
background: var(--bg-raised);
flex-shrink: 0;
}
.src-arrow {
display: flex; align-items: center; justify-content: center;
width: 22px; height: 22px; flex-shrink: 0;
border-radius: var(--radius-sm); color: var(--text-faint);
transition: color var(--t-base), background var(--t-base);
}
.src-arrow:hover:not(:disabled) { color: var(--text-primary); background: var(--bg-overlay); }
.src-arrow:disabled { opacity: 0.3; cursor: default; }
.src-label {
flex: 1; display: flex; align-items: center; justify-content: center;
gap: var(--sp-2); min-width: 0;
}
.src-name {
font-family: var(--font-ui); font-size: var(--text-xs);
color: var(--text-secondary); font-weight: var(--weight-medium);
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.src-dim { color: var(--text-faint); font-weight: normal; }
.src-lang {
font-family: var(--font-ui); font-size: 10px; letter-spacing: var(--tracking-wide);
color: var(--text-faint); background: var(--bg-overlay);
border: 1px solid var(--border-dim); border-radius: var(--radius-sm);
padding: 1px 5px; flex-shrink: 0;
}
.picker { display: flex; flex-direction: column; padding: var(--sp-1) 0; flex: 1; overflow-y: auto; min-height: 0; }
.picker-row {
display: flex; align-items: center; gap: var(--sp-3);
padding: 9px var(--sp-5); text-align: left;
transition: background var(--t-fast);
}
.picker-row:hover { background: var(--bg-raised); }
.picker-name {
flex: 1; font-size: var(--text-sm); color: var(--text-secondary);
font-weight: var(--weight-medium);
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
:global(.picker-caret) { color: var(--text-faint); flex-shrink: 0; }
.prefs-body { flex: 1; overflow-y: auto; }
.center-state {
display: flex; align-items: center; justify-content: center;
padding: var(--sp-10);
}
.dim { font-family: var(--font-ui); font-size: var(--text-xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
.pref-list { list-style: none; padding: var(--sp-1) 0; margin: 0; display: flex; flex-direction: column; }
.pref-row {
display: flex; align-items: center; gap: var(--sp-4);
padding: 11px var(--sp-5);
border-bottom: 1px solid var(--border-dim);
}
.pref-row:last-child { border-bottom: none; }
.pref-col { flex-direction: column; align-items: stretch; gap: var(--sp-2); }
.pref-row-btn {
width: 100%; text-align: left;
display: flex; align-items: center; gap: var(--sp-4);
padding: 11px var(--sp-5);
border-bottom: 1px solid var(--border-dim);
transition: background var(--t-fast);
}
.pref-row-btn:hover { background: var(--bg-raised); }
.edit-active { background: var(--bg-raised); }
.pref-text { display: flex; flex-direction: column; gap: 2px; flex: 1; min-width: 0; }
.pref-title { font-size: var(--text-sm); color: var(--text-secondary); font-weight: var(--weight-medium); }
.pref-summary {
font-family: var(--font-ui); font-size: var(--text-2xs);
color: var(--text-faint); letter-spacing: var(--tracking-wide); line-height: 1.5;
}
.pref-value-hint {
font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-muted);
letter-spacing: var(--tracking-wide); flex-shrink: 0;
max-width: 100px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
}
.toggle {
position: relative; width: 30px; height: 17px; border-radius: 9px;
background: var(--bg-overlay); border: 1px solid var(--border-strong);
flex-shrink: 0;
display: flex; align-items: center; justify-content: center;
transition: background var(--t-base), border-color var(--t-base);
}
.toggle-on { background: var(--accent-muted); border-color: var(--accent-dim); }
.toggle-thumb {
position: absolute; left: 2px; width: 11px; height: 11px;
border-radius: 50%; background: var(--text-faint);
transition: left var(--t-base), background var(--t-base); pointer-events: none;
}
.toggle-on .toggle-thumb { left: 15px; background: var(--accent-fg); }
.toggle:disabled { opacity: 0.4; cursor: default; }
.select-wrap { position: relative; }
.select-btn {
display: flex; align-items: center; justify-content: space-between; gap: var(--sp-2);
width: 100%; padding: 6px var(--sp-3);
background: var(--bg-base); border: 1px solid var(--border-strong);
border-radius: var(--radius-md); color: var(--text-secondary); font-size: var(--text-sm);
transition: border-color var(--t-base);
}
.select-btn:hover:not(:disabled) { border-color: var(--border-focus); }
.select-btn:disabled { opacity: 0.4; cursor: default; }
.select-open { border-color: var(--border-focus); }
.select-val { flex: 1; text-align: left; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.dropdown {
position: absolute; top: calc(100% + 4px); left: 0; right: 0;
background: var(--bg-surface); border: 1px solid var(--border-strong);
border-radius: var(--radius-md); overflow: hidden;
box-shadow: var(--shadow-lg); z-index: 10;
animation: dropIn 0.1s cubic-bezier(0.16,1,0.3,1) both;
}
@keyframes dropIn { from { opacity: 0; transform: translateY(-4px); } to { opacity: 1; transform: none; } }
.dropdown-item {
display: flex; align-items: center; justify-content: space-between;
width: 100%; padding: 7px var(--sp-3);
font-size: var(--text-sm); color: var(--text-secondary);
transition: background var(--t-fast);
}
.dropdown-item:hover { background: var(--bg-raised); }
.dropdown-item-active { color: var(--accent-fg); }
.edit-row { display: flex; gap: var(--sp-2); }
.edit-input {
flex: 1; background: var(--bg-base); border: 1px solid var(--border-strong);
border-radius: var(--radius-md); padding: 6px var(--sp-3);
color: var(--text-primary); font-size: var(--text-sm);
outline: none; transition: border-color var(--t-base);
}
.edit-input:focus { border-color: var(--border-focus); }
.action-btn {
font-family: var(--font-ui); font-size: var(--text-xs); letter-spacing: var(--tracking-wide);
padding: 5px 11px; border-radius: var(--radius-md);
background: var(--accent-muted); color: var(--accent-fg); border: 1px solid var(--accent-dim);
flex-shrink: 0; display: flex; align-items: center; gap: var(--sp-1);
transition: filter var(--t-base);
}
.action-btn:hover:not(:disabled) { filter: brightness(1.1); }
.action-btn:disabled { opacity: 0.4; cursor: default; }
.action-dim {
font-family: var(--font-ui); font-size: var(--text-xs); letter-spacing: var(--tracking-wide);
padding: 5px 11px; border-radius: var(--radius-md);
background: none; color: var(--text-faint); border: 1px solid var(--border-dim);
flex-shrink: 0; transition: color var(--t-base), border-color var(--t-base);
}
.action-dim:hover { color: var(--text-secondary); border-color: var(--border-strong); }
.multi-list { display: flex; flex-direction: column; gap: 1px; }
.multi-item {
display: flex; align-items: center; gap: var(--sp-2);
padding: 6px var(--sp-2); border-radius: var(--radius-md);
font-size: var(--text-sm); color: var(--text-muted); border: 1px solid transparent;
transition: background var(--t-fast), color var(--t-fast), border-color var(--t-fast);
}
.multi-item:hover:not(:disabled) { background: var(--bg-raised); color: var(--text-secondary); }
.multi-item-on { color: var(--accent-fg); background: var(--accent-muted); border-color: var(--accent-dim); }
.multi-check {
width: 13px; height: 13px; border-radius: var(--radius-sm);
border: 1px solid var(--border-strong); background: var(--bg-base);
display: flex; align-items: center; justify-content: center;
flex-shrink: 0; color: var(--accent-fg);
transition: background var(--t-fast), border-color var(--t-fast);
}
.multi-item-on .multi-check { background: var(--accent-muted); border-color: var(--accent-dim); }
</style>
@@ -0,0 +1,445 @@
<script lang="ts">
import { X, MagnifyingGlass, CircleNotch, ArrowRight, Check, Warning, Sparkle, Swap } from "phosphor-svelte";
import { getAdapter } from "$lib/request-manager";
import Thumbnail from "$lib/components/shared/manga/Thumbnail.svelte";
import { resolvedCover } from "$lib/core/cover/coverResolver";
import { addToast } from "$lib/state/notifications.svelte";
import { settingsState } from "$lib/state/settings.svelte";
import type { Manga, Chapter, Source } from "$lib/types";
import type { LibraryManga } from "$lib/components/extensions/lib/extensionLibrary";
interface Props {
sourceId: string;
sourceName: string;
sourceIconUrl: string;
manga: LibraryManga[];
onClose: () => void;
onDone: () => void;
}
let { sourceId, sourceName, sourceIconUrl, manga, onClose, onDone }: Props = $props();
type Phase = "pick-target" | "review" | "migrating" | "done";
interface EntryResult {
manga: LibraryManga;
match: Manga | null;
chapters: Chapter[];
similarity: number;
status: "pending" | "searching" | "found" | "no-match" | "migrated" | "failed";
error?: string;
}
function titleSimilarity(a: string, b: string): number {
const norm = (s: string) =>
s.toLowerCase().replace(/[^a-z0-9\s]/g, "").split(/\s+/).filter(Boolean);
const wordsA = new Set(norm(a));
const wordsB = new Set(norm(b));
if (wordsA.size === 0 || wordsB.size === 0) return 0;
const intersection = [...wordsA].filter(w => wordsB.has(w)).length;
return intersection / new Set([...wordsA, ...wordsB]).size;
}
function onKey(e: KeyboardEvent) { if (e.key === "Escape" && phase !== "migrating") onClose(); }
let phase: Phase = $state("pick-target");
let allSources: Source[] = $state([]);
let loadingSources = $state(true);
let targetSource: Source | null = $state(null);
let selectedLang = $state("all");
let langStripEl: HTMLDivElement | undefined = $state();
let entries: EntryResult[] = $state([]);
let searchProgress = $state({ done: 0, total: 0 });
let migrateProgress = $state({ done: 0, total: 0, failed: 0 });
const availableLangs = $derived.by(() => {
const langs = Array.from(new Set<string>(allSources.map(s => s.lang))).sort();
const en = langs.indexOf("en");
if (en > 0) { langs.splice(en, 1); langs.unshift("en"); }
return langs;
});
const hasMultipleLangs = $derived(availableLangs.length > 1);
const visibleSources = $derived.by(() => {
if (selectedLang !== "all") return allSources.filter(s => s.lang === selectedLang);
const map = new Map<string, Source>();
for (const s of allSources) {
const existing = map.get(s.name);
if (!existing || s.lang < existing.lang) map.set(s.name, s);
}
return Array.from(map.values());
});
const foundCount = $derived(entries.filter(e => e.status === "found").length);
const noMatchCount = $derived(entries.filter(e => e.status === "no-match").length);
const migratedCount = $derived(entries.filter(e => e.status === "migrated").length);
const failedCount = $derived(entries.filter(e => e.status === "failed").length);
$effect(() => {
getAdapter().getSources().then(nodes => ({ sources: { nodes } }))
.then(d => {
allSources = d.sources.nodes.filter(s => s.id !== "0" && s.id !== sourceId);
const prefLang = settingsState.settings.preferredExtensionLang ?? "";
const langs = new Set(allSources.map(s => s.lang));
if (prefLang && langs.has(prefLang) && langs.size > 1) selectedLang = prefLang;
})
.catch(console.error)
.finally(() => { loadingSources = false; });
window.addEventListener("keydown", onKey);
return () => window.removeEventListener("keydown", onKey);
});
function scrollLangStrip(dir: -1 | 1) {
if (!langStripEl) return;
const chips = Array.from(langStripEl.children) as HTMLElement[];
const viewEnd = langStripEl.scrollLeft + langStripEl.clientWidth;
if (dir === 1) {
const next = chips.find(c => c.offsetLeft + c.offsetWidth > viewEnd + 2);
if (next) langStripEl.scrollTo({ left: next.offsetLeft, behavior: "smooth" });
} else {
const prev = [...chips].reverse().find(c => c.offsetLeft < langStripEl!.scrollLeft - 2);
if (prev) langStripEl.scrollTo({ left: prev.offsetLeft + prev.offsetWidth - langStripEl.clientWidth, behavior: "smooth" });
}
}
async function startSearch(target: Source) {
targetSource = target;
phase = "review";
entries = manga.map(m => ({ manga: m, match: null, chapters: [], similarity: 0, status: "pending" }));
searchProgress = { done: 0, total: manga.length };
for (let i = 0; i < entries.length; i++) {
entries[i] = { ...entries[i], status: "searching" };
try {
const d = await gql<{ fetchSourceManga: { mangas: Manga[] } }>(FETCH_SOURCE_MANGA, {
source: target.id, type: "SEARCH", page: 1, query: entries[i].manga.title,
});
const results = d.fetchSourceManga.mangas
.map(m => ({ manga: m, similarity: titleSimilarity(entries[i].manga.title, m.title) }))
.sort((a, b) => b.similarity - a.similarity);
if (results.length > 0 && results[0].similarity > 0.3) {
entries[i] = { ...entries[i], match: results[0].manga, similarity: results[0].similarity, status: "found" };
} else {
entries[i] = { ...entries[i], status: "no-match" };
}
} catch (e: any) {
entries[i] = { ...entries[i], status: "no-match", error: e.message };
}
searchProgress = { done: i + 1, total: manga.length };
}
}
function setEntryMatch(idx: number, match: Manga, similarity: number) {
entries[idx] = { ...entries[idx], match, similarity, status: "found" };
}
function excludeEntry(idx: number) {
entries[idx] = { ...entries[idx], status: "no-match", match: null };
}
async function startMigration() {
const toMigrate = entries.filter(e => e.status === "found" && e.match);
migrateProgress = { done: 0, total: toMigrate.length, failed: 0 };
phase = "migrating";
for (const entry of toMigrate) {
const idx = entries.indexOf(entry);
try {
const d = await gql<{ fetchChapters: { chapters: Chapter[] } }>(FETCH_CHAPTERS, { mangaId: entry.match!.id });
const newChaps = d.fetchChapters.chapters;
const toMarkRead: number[] = [];
const toMarkBookmarked: number[] = [];
for (const nc of newChaps) {
const oldIdx = entries[idx].manga;
if (oldIdx) {
toMarkRead.push(nc.id);
}
}
if (toMarkRead.length)
await getAdapter().markChaptersRead(toMarkRead.map(String), true);
await getAdapter().addToLibrary(String(entry.match!.id));
await getAdapter().removeFromLibrary(String(entry.manga.id));
entries[idx] = { ...entries[idx], status: "migrated" };
migrateProgress = { ...migrateProgress, done: migrateProgress.done + 1 };
} catch (e: any) {
entries[idx] = { ...entries[idx], status: "failed", error: e.message };
migrateProgress = { ...migrateProgress, done: migrateProgress.done + 1, failed: migrateProgress.failed + 1 };
}
}
phase = "done";
addToast({
kind: "success",
title: "Migration complete",
body: `${migrateProgress.done - migrateProgress.failed} migrated, ${migrateProgress.failed} failed`,
});
}
</script>
<div class="overlay" onclick={(e) => { if (e.target === e.currentTarget && phase !== "migrating") onClose(); }}>
<div class="modal">
<div class="modal-header">
<div class="source-context">
<div class="source-icon-wrap">
<Thumbnail src={sourceIconUrl} alt={sourceName} class="src-icon" onerror={(e) => (e.target as HTMLImageElement).style.display = "none"} />
</div>
<div class="source-context-info">
<span class="modal-eyebrow">Source migration</span>
<span class="modal-title">{sourceName}</span>
<span class="modal-sub">{manga.length} {manga.length === 1 ? "title" : "titles"} in library</span>
</div>
</div>
{#if phase !== "migrating"}
<button class="close-btn" onclick={onClose}><X size={14} weight="light" /></button>
{/if}
</div>
<div class="body">
{#if phase === "pick-target"}
<div class="phase-label-row">
<span class="phase-label">Select destination source</span>
</div>
{#if loadingSources}
<div class="centered"><CircleNotch size={16} weight="light" class="anim-spin" style="color:var(--text-faint)" /></div>
{:else if allSources.length === 0}
<div class="centered"><span class="hint">No other sources installed.</span></div>
{:else}
{#if hasMultipleLangs}
<div class="src-lang-bar">
<button class="src-lang-nav" onclick={() => scrollLangStrip(-1)}></button>
<div class="src-lang-chips" bind:this={langStripEl}>
<button class="src-lang-chip" class:src-lang-chip-active={selectedLang === "all"} onclick={() => selectedLang = "all"}>All</button>
{#each availableLangs as lang}
<button class="src-lang-chip" class:src-lang-chip-active={selectedLang === lang} onclick={() => selectedLang = lang}>
{lang.toUpperCase()}
</button>
{/each}
</div>
<button class="src-lang-nav" onclick={() => scrollLangStrip(1)}></button>
</div>
{/if}
<div class="source-list">
{#each visibleSources as src}
<button class="source-row" onclick={() => startSearch(src)}>
<div class="source-icon-wrap">
<Thumbnail src={src.iconUrl} alt={src.name} class="source-icon" onerror={(e) => (e.target as HTMLImageElement).style.display = "none"} />
</div>
<div class="source-info">
<span class="source-name">{src.displayName}</span>
<span class="source-meta">{src.lang.toUpperCase()}{src.isNsfw ? " · NSFW" : ""}</span>
</div>
<ArrowRight size={13} weight="light" class="source-arrow" />
</button>
{/each}
</div>
{/if}
{:else if phase === "review" || phase === "migrating" || phase === "done"}
<div class="review-header">
<div class="review-route">
<div class="review-source">
<div class="source-icon-wrap small">
<Thumbnail src={sourceIconUrl} alt={sourceName} class="source-icon" onerror={(e) => (e.target as HTMLImageElement).style.display = "none"} />
</div>
<span class="review-source-name">{sourceName}</span>
</div>
<ArrowRight size={14} weight="light" style="color:var(--text-faint);flex-shrink:0" />
{#if targetSource}
<div class="review-source">
<div class="source-icon-wrap small">
<Thumbnail src={targetSource.iconUrl} alt={targetSource.name} class="source-icon" onerror={(e) => (e.target as HTMLImageElement).style.display = "none"} />
</div>
<span class="review-source-name">{targetSource.displayName}</span>
</div>
{/if}
</div>
{#if phase === "review"}
<div class="review-progress-row">
<div class="review-progress-bar">
<div class="review-progress-fill" style="width:{searchProgress.total ? (searchProgress.done / searchProgress.total) * 100 : 0}%"></div>
</div>
<span class="review-progress-label">
{#if searchProgress.done < searchProgress.total}
Searching {searchProgress.done + 1} / {searchProgress.total}
{:else}
{foundCount} found · {noMatchCount} no match
{/if}
</span>
</div>
{:else if phase === "migrating"}
<div class="review-progress-row">
<div class="review-progress-bar">
<div class="review-progress-fill" style="width:{migrateProgress.total ? (migrateProgress.done / migrateProgress.total) * 100 : 0}%"></div>
</div>
<span class="review-progress-label">Migrating {migrateProgress.done} / {migrateProgress.total}</span>
</div>
{:else}
<div class="done-summary">
<Check size={13} weight="bold" style="color:var(--color-success)" />
<span class="done-label">{migratedCount} migrated{failedCount > 0 ? ` · ${failedCount} failed` : ""}</span>
</div>
{/if}
</div>
<div class="entry-list">
{#each entries as entry, idx}
<div class="entry-row" class:entry-migrated={entry.status === "migrated"} class:entry-failed={entry.status === "failed"}>
<div class="entry-cover-wrap">
<Thumbnail src={resolvedCover(entry.manga.id, entry.manga.thumbnailUrl)} alt={entry.manga.title} class="entry-cover" />
</div>
<div class="entry-info">
<span class="entry-title">{entry.manga.title}</span>
{#if entry.status === "found" && entry.match}
<span class="entry-match">
<Sparkle size={9} weight="fill" style="color:var(--accent-fg);flex-shrink:0" />
{entry.match.title}
<span class="entry-sim">{Math.round(entry.similarity * 100)}%</span>
</span>
{:else if entry.status === "no-match"}
<span class="entry-no-match">No match found</span>
{:else if entry.status === "searching"}
<span class="entry-searching">Searching…</span>
{:else if entry.status === "migrated"}
<span class="entry-done">Migrated</span>
{:else if entry.status === "failed"}
<span class="entry-fail">{entry.error ?? "Failed"}</span>
{/if}
</div>
<div class="entry-status">
{#if entry.status === "searching"}
<CircleNotch size={13} weight="light" class="anim-spin" style="color:var(--text-faint)" />
{:else if entry.status === "found"}
<div class="entry-cover-match">
<Thumbnail src={resolvedCover(entry.match!.id, entry.match!.thumbnailUrl)} alt={entry.match!.title} class="entry-match-cover" />
</div>
{#if phase === "review"}
<button class="entry-exclude-btn" onclick={() => excludeEntry(idx)} title="Exclude from migration">
<X size={10} weight="bold" />
</button>
{/if}
{:else if entry.status === "migrated"}
<Check size={13} weight="bold" style="color:var(--color-success)" />
{:else if entry.status === "failed"}
<Warning size={13} weight="light" style="color:var(--color-error)" />
{/if}
</div>
</div>
{/each}
</div>
{#if phase === "review" && searchProgress.done === searchProgress.total}
<div class="review-actions">
<button class="back-btn" onclick={() => { phase = "pick-target"; entries = []; }}>Change source</button>
<button class="migrate-btn" onclick={startMigration} disabled={foundCount === 0}>
<Swap size={13} weight="bold" />
Migrate {foundCount} {foundCount === 1 ? "title" : "titles"}
</button>
</div>
{/if}
{#if phase === "done"}
<div class="review-actions">
<button class="migrate-btn" onclick={onDone}><Check size={13} weight="bold" /> Done</button>
</div>
{/if}
{/if}
</div>
</div>
</div>
<style>
.overlay { position: fixed; inset: 0; background: rgba(0,0,0,0.6); display: flex; align-items: center; justify-content: center; z-index: 200; animation: fadeIn 0.1s ease both; }
.modal { background: var(--bg-base); border: 1px solid var(--border-base); border-radius: var(--radius-xl); width: 560px; max-height: 84vh; display: flex; flex-direction: column; overflow: hidden; box-shadow: 0 8px 40px rgba(0,0,0,0.5); }
.modal-header { display: flex; align-items: flex-start; justify-content: space-between; gap: var(--sp-3); padding: var(--sp-4) var(--sp-5); border-bottom: 1px solid var(--border-dim); flex-shrink: 0; }
.source-context { display: flex; align-items: center; gap: var(--sp-3); min-width: 0; }
.source-icon-wrap { width: 36px; height: 36px; border-radius: var(--radius-md); overflow: hidden; flex-shrink: 0; background: var(--bg-raised); border: 1px solid var(--border-dim); }
.source-icon-wrap.small { width: 20px; height: 20px; border-radius: var(--radius-sm); }
:global(.src-icon) { width: 100%; height: 100%; object-fit: cover; }
:global(.source-icon) { width: 100%; height: 100%; object-fit: cover; }
.source-context-info { display: flex; flex-direction: column; gap: 2px; min-width: 0; }
.modal-eyebrow { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wider); text-transform: uppercase; }
.modal-title { font-size: var(--text-base); font-weight: var(--weight-medium); color: var(--text-primary); letter-spacing: var(--tracking-tight); }
.modal-sub { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
.close-btn { display: flex; align-items: center; justify-content: center; width: 24px; height: 24px; border-radius: var(--radius-md); color: var(--text-faint); background: none; border: none; cursor: pointer; transition: color var(--t-base), background var(--t-base); flex-shrink: 0; margin-top: 2px; }
.close-btn:hover { color: var(--text-muted); background: var(--bg-raised); }
.body { flex: 1; overflow: hidden; display: flex; flex-direction: column; min-height: 0; }
.phase-label-row { padding: var(--sp-3) var(--sp-4) var(--sp-2); flex-shrink: 0; }
.phase-label { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-widest); text-transform: uppercase; }
.centered { flex: 1; display: flex; align-items: center; justify-content: center; padding: var(--sp-8); }
.hint { font-family: var(--font-ui); font-size: var(--text-xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
.src-lang-bar { display: flex; align-items: center; gap: var(--sp-1); padding: var(--sp-2); border-bottom: 1px solid var(--border-dim); flex-shrink: 0; }
.src-lang-nav { display: flex; align-items: center; justify-content: center; width: 22px; height: 22px; flex-shrink: 0; border-radius: var(--radius-sm); border: 1px solid var(--border-dim); background: none; color: var(--text-faint); font-size: 15px; line-height: 1; cursor: pointer; transition: color var(--t-base), background var(--t-base); }
.src-lang-nav:hover { color: var(--text-muted); background: var(--bg-raised); }
.src-lang-chips { display: flex; align-items: center; gap: var(--sp-1); flex: 1; min-width: 0; overflow-x: auto; scrollbar-width: none; }
.src-lang-chips::-webkit-scrollbar { display: none; }
.src-lang-chip { font-family: var(--font-ui); font-size: var(--text-2xs); letter-spacing: var(--tracking-wide); padding: 3px 8px; border-radius: var(--radius-sm); border: 1px solid var(--border-dim); background: none; color: var(--text-faint); cursor: pointer; white-space: nowrap; flex-shrink: 0; transition: color var(--t-base), border-color var(--t-base), background var(--t-base); }
.src-lang-chip:hover { color: var(--text-muted); background: var(--bg-raised); }
.src-lang-chip-active { color: var(--accent-fg); border-color: var(--accent-dim); background: var(--accent-muted); }
.source-list { flex: 1; overflow-y: auto; padding: var(--sp-2); display: flex; flex-direction: column; gap: 1px; }
.source-row { display: flex; align-items: center; gap: var(--sp-3); padding: 8px var(--sp-3); border-radius: var(--radius-md); border: 1px solid transparent; background: none; text-align: left; width: 100%; cursor: pointer; transition: background var(--t-fast), border-color var(--t-fast); }
.source-row:hover { background: var(--bg-raised); border-color: var(--border-dim); }
.source-info { flex: 1; display: flex; flex-direction: column; gap: 2px; overflow: hidden; }
.source-name { font-size: var(--text-sm); font-weight: var(--weight-medium); color: var(--text-secondary); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.source-meta { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
:global(.source-arrow) { color: var(--text-faint); opacity: 0; transition: opacity var(--t-base); flex-shrink: 0; }
.source-row:hover :global(.source-arrow) { opacity: 1; }
.review-header { padding: var(--sp-3) var(--sp-4); border-bottom: 1px solid var(--border-dim); flex-shrink: 0; display: flex; flex-direction: column; gap: var(--sp-2); }
.review-route { display: flex; align-items: center; gap: var(--sp-2); }
.review-source { display: flex; align-items: center; gap: var(--sp-2); min-width: 0; }
.review-source-name { font-size: var(--text-xs); color: var(--text-secondary); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; font-weight: var(--weight-medium); }
.review-progress-row { display: flex; align-items: center; gap: var(--sp-3); }
.review-progress-bar { flex: 1; height: 3px; background: var(--bg-overlay); border-radius: var(--radius-full); overflow: hidden; }
.review-progress-fill { height: 100%; background: var(--accent); border-radius: var(--radius-full); transition: width 0.3s ease; }
.review-progress-label { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); white-space: nowrap; flex-shrink: 0; }
.done-summary { display: flex; align-items: center; gap: var(--sp-2); }
.done-label { font-family: var(--font-ui); font-size: var(--text-xs); color: var(--text-secondary); letter-spacing: var(--tracking-wide); }
.entry-list { flex: 1; overflow-y: auto; padding: var(--sp-2); display: flex; flex-direction: column; gap: 1px; }
.entry-row { display: flex; align-items: center; gap: var(--sp-3); padding: 7px var(--sp-3); border-radius: var(--radius-md); border: 1px solid transparent; transition: background var(--t-fast); }
.entry-row:hover { background: var(--bg-raised); }
.entry-migrated { opacity: 0.5; }
.entry-failed { border-color: rgba(180,60,60,0.15); background: rgba(180,60,60,0.04); }
.entry-cover-wrap { width: 28px; height: 42px; border-radius: var(--radius-sm); overflow: hidden; background: var(--bg-raised); border: 1px solid var(--border-dim); flex-shrink: 0; }
:global(.entry-cover) { width: 100%; height: 100%; object-fit: cover; }
.entry-info { flex: 1; display: flex; flex-direction: column; gap: 3px; min-width: 0; overflow: hidden; }
.entry-title { font-size: var(--text-sm); color: var(--text-secondary); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.entry-match { display: flex; align-items: center; gap: 4px; font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.entry-sim { background: var(--bg-overlay); border: 1px solid var(--border-dim); border-radius: var(--radius-sm); padding: 0 4px; font-size: 9px; flex-shrink: 0; }
.entry-no-match { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
.entry-searching { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
.entry-done { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--color-success); letter-spacing: var(--tracking-wide); }
.entry-fail { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--color-error); letter-spacing: var(--tracking-wide); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.entry-status { display: flex; align-items: center; gap: var(--sp-1); flex-shrink: 0; }
.entry-cover-match { width: 24px; height: 36px; border-radius: var(--radius-sm); overflow: hidden; background: var(--bg-raised); border: 1px solid var(--border-dim); flex-shrink: 0; }
:global(.entry-match-cover) { width: 100%; height: 100%; object-fit: cover; }
.entry-exclude-btn { display: flex; align-items: center; justify-content: center; width: 18px; height: 18px; border-radius: var(--radius-sm); color: var(--text-faint); background: none; border: none; cursor: pointer; transition: color var(--t-base), background var(--t-base); flex-shrink: 0; }
.entry-exclude-btn:hover { color: var(--color-error); background: var(--bg-raised); }
.review-actions { display: flex; justify-content: flex-end; gap: var(--sp-2); padding: var(--sp-3) var(--sp-4); border-top: 1px solid var(--border-dim); flex-shrink: 0; }
.back-btn { font-family: var(--font-ui); font-size: var(--text-xs); letter-spacing: var(--tracking-wide); padding: 6px 10px; border-radius: var(--radius-md); background: none; color: var(--text-muted); border: 1px solid var(--border-dim); cursor: pointer; transition: color var(--t-base), border-color var(--t-base), background var(--t-base); }
.back-btn:hover { color: var(--text-secondary); border-color: var(--border-strong); background: var(--bg-raised); }
.migrate-btn { display: flex; align-items: center; gap: var(--sp-2); font-family: var(--font-ui); font-size: var(--text-xs); letter-spacing: var(--tracking-wide); padding: 7px 16px; border-radius: var(--radius-md); background: var(--accent-dim); border: 1px solid var(--accent); color: var(--accent-fg); cursor: pointer; transition: background var(--t-base), border-color var(--t-base); }
.migrate-btn:hover:not(:disabled) { background: var(--accent-muted); border-color: var(--accent-bright); }
.migrate-btn:disabled { opacity: 0.4; cursor: default; }
@keyframes fadeIn { from { opacity: 0 } to { opacity: 1 } }
</style>