From 1c135a79ca4185891c78904d6c73c0c4f28425d1 Mon Sep 17 00:00:00 2001 From: Youwes09 Date: Mon, 13 Apr 2026 21:28:57 -0500 Subject: [PATCH] Feat: Enforce & Block for Scanlators --- src/components/series/SeriesDetail.svelte | 97 ++++++++++++++++++----- 1 file changed, 75 insertions(+), 22 deletions(-) diff --git a/src/components/series/SeriesDetail.svelte b/src/components/series/SeriesDetail.svelte index 93bf6d1..a87944b 100644 --- a/src/components/series/SeriesDetail.svelte +++ b/src/components/series/SeriesDetail.svelte @@ -98,11 +98,19 @@ .sort((a, b) => a.localeCompare(b)) ); - const scanlatorFilter = $derived((getPref("scanlatorFilter") ?? []) as string[]); + const scanlatorFilter = $derived((getPref("scanlatorFilter") ?? []) as string[]); + const scanlatorBlacklist = $derived((getPref("scanlatorBlacklist") ?? []) as string[]); + const scanlatorForce = $derived((getPref("scanlatorForce") ?? false) as boolean); + + let scanTab: "prefer" | "block" = $state("prefer"); const sortedChapters = $derived.by(() => { let base = [...chapters]; + if (scanlatorBlacklist.length > 0) { + base = base.filter(c => !scanlatorBlacklist.includes(c.scanlator ?? "")); + } + if (sortMode === "chapterNumber") base.sort((a, b) => a.chapterNumber - b.chapterNumber); else if (sortMode === "uploadDate") base.sort((a, b) => Number(a.uploadDate ?? 0) - Number(b.uploadDate ?? 0)); else base.sort((a, b) => a.sourceOrder - b.sourceOrder); @@ -119,7 +127,9 @@ for (const ch of base) { const existing = seen.get(ch.chapterNumber); if (!existing) { - seen.set(ch.chapterNumber, ch); + if (!scanlatorForce || scanlatorFilter.includes(ch.scanlator ?? "")) { + seen.set(ch.chapterNumber, ch); + } } else { const np = scanlatorFilter.indexOf(ch.scanlator ?? ""); const op = scanlatorFilter.indexOf(existing.scanlator ?? ""); @@ -782,33 +792,64 @@ {#if availableScanlators.length > 1}
- {#if scanFilterOpen} {/if}
@@ -1284,6 +1325,18 @@ .scan-filter-item:hover { background: var(--bg-overlay); color: var(--text-primary); } .scan-filter-item-active { color: var(--accent-fg); background: var(--accent-muted); } .scan-filter-item-active:hover { background: var(--accent-dim); } + .scan-filter-tabs { display: flex; gap: 2px; background: var(--bg-overlay); border: 1px solid var(--border-base); border-radius: var(--radius-sm); padding: 2px; } + .scan-filter-tab { font-family: var(--font-ui); font-size: var(--text-2xs); letter-spacing: var(--tracking-wide); padding: 2px 8px; border-radius: 2px; border: none; background: none; color: var(--text-faint); cursor: pointer; transition: color var(--t-fast), background var(--t-fast); } + .scan-filter-tab:hover { color: var(--text-muted); } + .scan-filter-tab.scan-filter-tab-active { background: var(--bg-surface); color: var(--text-primary); box-shadow: 0 1px 3px rgba(0,0,0,0.3); } + .scan-filter-force-row { display: flex; align-items: center; justify-content: space-between; padding: 5px 10px; } + .scan-filter-force-label { font-family: var(--font-ui); font-size: var(--text-xs); color: var(--text-muted); letter-spacing: var(--tracking-wide); cursor: default; text-decoration: underline; text-decoration-style: dotted; text-decoration-color: var(--border-strong); text-underline-offset: 3px; } + .scan-force-toggle { font-family: var(--font-ui); font-size: var(--text-2xs); letter-spacing: var(--tracking-wide); padding: 2px 8px; border-radius: var(--radius-sm); border: 1px solid var(--border-dim); background: none; color: var(--text-faint); cursor: pointer; transition: color var(--t-base), border-color var(--t-base), background var(--t-base); } + .scan-force-toggle:hover { color: var(--text-muted); border-color: var(--border-strong); } + .scan-force-toggle.scan-force-on { color: var(--accent-fg); border-color: var(--accent-dim); background: var(--accent-muted); } .scan-filter-check { width: 13px; height: 13px; border-radius: 2px; border: 1px solid var(--border-strong); background: transparent; flex-shrink: 0; display: flex; align-items: center; justify-content: center; color: var(--bg-base); transition: background var(--t-base), border-color var(--t-base); } .scan-filter-check-on { background: var(--accent); border-color: var(--accent); } + .scan-filter-check-block { background: var(--color-error); border-color: var(--color-error); } + .scan-filter-item-block { color: var(--color-error) !important; background: color-mix(in srgb, var(--color-error) 8%, transparent) !important; } + .scan-filter-item-block:hover { background: color-mix(in srgb, var(--color-error) 14%, transparent) !important; }