Chore: Temporary Home-Page (Until Fixed with Rewrite)

This commit is contained in:
Youwes09
2026-03-19 22:53:51 -05:00
parent 161b1f9f52
commit 43630ef72d
12 changed files with 522 additions and 380 deletions
+403
View File
@@ -0,0 +1,403 @@
<script lang="ts">
import {
ClockCounterClockwise, Trash, MagnifyingGlass, Play, Books,
X as XIcon,
} from "phosphor-svelte";
import { thumbUrl, gql } from "../../lib/client";
import { GET_CHAPTERS } from "../../lib/queries";
import {
history, readingStats, openReader,
} from "../../store";
import type { HistoryEntry } from "../../store";
let search = "";
let confirmClearAll = false;
function timeAgo(ts: number): string {
const diff = Date.now() - ts, m = Math.floor(diff / 60000);
if (m < 1) return "Just now";
if (m < 60) return `${m}m ago`;
const h = Math.floor(m / 60);
if (h < 24) return `${h}h ago`;
const d = Math.floor(h / 24);
if (d < 7) return `${d}d ago`;
return new Date(ts).toLocaleDateString("en-US", { month: "short", day: "numeric" });
}
function dayLabel(ts: number): string {
const d = new Date(ts), now = new Date();
if (d.toDateString() === now.toDateString()) return "Today";
const yest = new Date(now); yest.setDate(now.getDate() - 1);
if (d.toDateString() === yest.toDateString()) return "Yesterday";
const weekAgo = new Date(now); weekAgo.setDate(now.getDate() - 7);
if (d > weekAgo) return d.toLocaleDateString("en-US", { weekday: "long" });
return d.toLocaleDateString("en-US", {
month: "long", day: "numeric",
year: d.getFullYear() !== now.getFullYear() ? "numeric" : undefined,
});
}
function formatReadTime(mins: number): string {
if (mins < 1) return `${Math.round(mins * 60)}s`;
if (mins < 60) return `${Math.round(mins)}m`;
const h = Math.floor(mins / 60), r = mins % 60;
if (h < 24) return r === 0 ? `${h}h` : `${h}h ${r}m`;
const d = Math.floor(h / 24), rh = h % 24;
return rh === 0 ? `${d}d` : `${d}d ${rh}h`;
}
const SESSION_GAP_MS = 30 * 60 * 1000;
interface Session {
mangaId: number;
mangaTitle: string;
thumbnailUrl: string;
latestChapterId: number;
latestChapterName: string;
latestPageNumber: number;
firstChapterName: string;
chapterCount: number;
readAt: number;
}
function buildSessions(entries: HistoryEntry[]): Session[] {
if (!entries.length) return [];
const sessions: Session[] = [];
let i = 0;
while (i < entries.length) {
const anchor = entries[i];
const group: HistoryEntry[] = [anchor];
let j = i + 1;
while (j < entries.length) {
const next = entries[j];
if (next.mangaId === anchor.mangaId && anchor.readAt - next.readAt <= SESSION_GAP_MS) {
group.push(next); j++;
} else break;
}
const latest = group[0], oldest = group[group.length - 1];
sessions.push({
mangaId: latest.mangaId,
mangaTitle: latest.mangaTitle,
thumbnailUrl: latest.thumbnailUrl,
latestChapterId: latest.chapterId,
latestChapterName: latest.chapterName,
latestPageNumber: latest.pageNumber,
firstChapterName: oldest.chapterName,
chapterCount: group.length,
readAt: latest.readAt,
});
i = j;
}
return sessions;
}
$: filtered = search.trim()
? $history.filter((e) =>
e.mangaTitle.toLowerCase().includes(search.toLowerCase()) ||
e.chapterName.toLowerCase().includes(search.toLowerCase())
)
: $history;
$: sessions = buildSessions(filtered);
$: groups = (() => {
const map = new Map<string, Session[]>();
for (const s of sessions) {
const l = dayLabel(s.readAt);
if (!map.has(l)) map.set(l, []);
map.get(l)!.push(s);
}
return Array.from(map.entries()).map(([label, items]) => ({ label, items }));
})();
$: stats = {
uniqueChapters: new Set($history.map((e) => e.chapterId)).size,
uniqueManga: new Set($history.map((e) => e.mangaId)).size,
estimatedMinutes: Math.round(new Set($history.map((e) => e.chapterId)).size * 4.5),
};
function clearAll() {
history.set([]);
confirmClearAll = false;
}
function clearManga(mangaId: number, e: MouseEvent) {
e.stopPropagation();
history.update((h) => h.filter((x) => x.mangaId !== mangaId));
}
async function resume(session: Session) {
try {
const d = await gql<{ chapters: { nodes: any[] } }>(GET_CHAPTERS, { mangaId: session.mangaId });
const chapters = [...d.chapters.nodes].sort((a, b) => a.sourceOrder - b.sourceOrder);
const ch = chapters.find((c) => c.id === session.latestChapterId) ?? chapters[0];
if (ch) openReader(ch, chapters);
} catch {}
}
</script>
<div class="root">
<div class="header">
<h1 class="heading">History</h1>
<div class="header-right">
<div class="search-wrap">
<MagnifyingGlass size={12} class="search-icon" weight="light" />
<input class="search" placeholder="Search history…" bind:value={search} />
{#if search}
<button class="search-clear" on:click={() => (search = "")}>
<XIcon size={10} weight="bold" />
</button>
{/if}
</div>
{#if $history.length > 0}
{#if confirmClearAll}
<div class="confirm-row">
<span class="confirm-label">Clear all activity?</span>
<button class="confirm-yes" on:click={clearAll}>Clear</button>
<button class="confirm-no" on:click={() => (confirmClearAll = false)}>Cancel</button>
</div>
{:else}
<button class="clear-btn" on:click={() => (confirmClearAll = true)} title="Clear all activity">
<Trash size={13} weight="light" />
</button>
{/if}
{/if}
</div>
</div>
<div class="stats-bar">
<span class="stat-item">
<span class="stat-val">{stats.uniqueChapters}</span>
<span class="stat-label">chapters</span>
</span>
<span class="stat-sep"></span>
<span class="stat-item">
<span class="stat-val">{stats.uniqueManga}</span>
<span class="stat-label">series</span>
</span>
<span class="stat-sep"></span>
<span class="stat-item">
<span class="stat-val">{formatReadTime(stats.estimatedMinutes)}</span>
<span class="stat-label">est. time</span>
</span>
{#if $readingStats.currentStreakDays > 0}
<span class="stat-sep"></span>
<span class="stat-item">
<span class="stat-val">{$readingStats.currentStreakDays}d</span>
<span class="stat-label">streak</span>
</span>
{/if}
</div>
{#if $history.length === 0}
<div class="empty">
<ClockCounterClockwise size={32} weight="light" class="empty-icon" />
<p class="empty-text">No reading history yet</p>
<p class="empty-hint">Chapters you read will appear here</p>
</div>
{:else if sessions.length === 0}
<div class="empty">
<Books size={28} weight="light" class="empty-icon" />
<p class="empty-text">No results for "{search}"</p>
</div>
{:else}
<div class="list">
{#each groups as { label, items } (label)}
<div class="group">
<p class="group-label">
<span>{label}</span>
<span class="group-count">{items.length}</span>
</p>
{#each items as session (session.latestChapterId + ":" + session.readAt)}
<div class="row-wrap">
<button class="row" on:click={() => resume(session)}>
<div class="thumb-wrap">
<img
src={thumbUrl(session.thumbnailUrl)}
alt={session.mangaTitle}
class="thumb"
loading="lazy"
decoding="async"
/>
{#if session.chapterCount > 1}
<span class="session-badge">{session.chapterCount}</span>
{/if}
</div>
<div class="info">
<span class="manga-title">{session.mangaTitle}</span>
<span class="chapter-name">
{#if session.chapterCount > 1}
<span class="chapter-range">
{session.firstChapterName}
<span class="range-sep"></span>
{session.latestChapterName}
</span>
{:else}
{session.latestChapterName}
{#if session.latestPageNumber > 1}
<span class="page-badge">p.{session.latestPageNumber}</span>
{/if}
{/if}
</span>
</div>
<span class="time">{timeAgo(session.readAt)}</span>
<Play size={11} weight="fill" class="play-icon" />
</button>
<button
class="row-delete"
on:click={(e) => clearManga(session.mangaId, e)}
title="Remove {session.mangaTitle} from history"
aria-label="Remove from history"
>
<XIcon size={9} weight="bold" />
</button>
</div>
{/each}
</div>
{/each}
</div>
{/if}
</div>
<style>
.root { display: flex; flex-direction: column; height: 100%; overflow: hidden; animation: fadeIn 0.14s ease both; }
.header {
display: flex; align-items: center; justify-content: space-between;
padding: var(--sp-5) var(--sp-6) var(--sp-3); flex-shrink: 0;
}
.heading {
font-family: var(--font-ui); font-size: var(--text-xs); font-weight: var(--weight-normal);
color: var(--text-faint); letter-spacing: var(--tracking-wider); text-transform: uppercase;
}
.header-right { display: flex; align-items: center; gap: var(--sp-2); }
.search-wrap { position: relative; display: flex; align-items: center; }
.search-wrap :global(.search-icon) { position: absolute; left: 9px; color: var(--text-faint); pointer-events: none; }
.search {
background: var(--bg-raised); border: 1px solid var(--border-dim);
border-radius: var(--radius-md); padding: 5px 28px 5px 26px;
color: var(--text-primary); font-size: var(--text-sm); width: 180px; outline: none;
transition: border-color var(--t-base);
}
.search::placeholder { color: var(--text-faint); }
.search:focus { border-color: var(--border-strong); }
.search-clear {
position: absolute; right: 7px; display: flex; align-items: center; justify-content: center;
color: var(--text-faint); background: none; border: none; cursor: pointer;
padding: 2px; transition: color var(--t-base);
}
.search-clear:hover { color: var(--text-muted); }
.confirm-row { display: flex; align-items: center; gap: var(--sp-2); }
.confirm-label { font-family: var(--font-ui); font-size: var(--text-xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
.confirm-yes {
font-family: var(--font-ui); font-size: var(--text-xs); padding: 4px 10px;
border-radius: var(--radius-sm); border: 1px solid var(--color-error);
background: var(--color-error-bg); color: var(--color-error); cursor: pointer;
transition: filter var(--t-base);
}
.confirm-yes:hover { filter: brightness(1.15); }
.confirm-no {
font-family: var(--font-ui); font-size: var(--text-xs); padding: 4px 10px;
border-radius: var(--radius-sm); border: 1px solid var(--border-dim);
background: none; color: var(--text-faint); cursor: pointer;
transition: background var(--t-base);
}
.confirm-no:hover { background: var(--bg-raised); color: var(--text-muted); }
.clear-btn {
display: flex; align-items: center; justify-content: center;
width: 28px; height: 28px; border-radius: var(--radius-md);
color: var(--text-faint); background: none; border: none; cursor: pointer;
transition: color var(--t-base), background var(--t-base);
}
.clear-btn:hover { color: var(--color-error); background: var(--color-error-bg); }
.stats-bar {
display: flex; align-items: center; gap: var(--sp-3);
padding: 0 var(--sp-6) var(--sp-3); flex-shrink: 0;
}
.stat-item { display: flex; align-items: baseline; gap: 4px; }
.stat-val { font-family: var(--font-ui); font-size: var(--text-sm); font-weight: var(--weight-medium); color: var(--text-secondary); }
.stat-label { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); }
.stat-sep { width: 1px; height: 10px; background: var(--border-dim); flex-shrink: 0; }
.list { flex: 1; overflow-y: auto; padding: 0 var(--sp-4) var(--sp-6); scrollbar-width: none; }
.list::-webkit-scrollbar { display: none; }
.group { margin-bottom: var(--sp-4); }
.group-label {
display: flex; align-items: center; gap: var(--sp-2);
font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint);
letter-spacing: var(--tracking-wider); text-transform: uppercase;
padding: var(--sp-1) var(--sp-2) var(--sp-2);
}
.group-count {
font-family: var(--font-ui); font-size: 9px; color: var(--text-faint);
background: var(--bg-raised); border: 1px solid var(--border-dim);
padding: 1px 5px; border-radius: var(--radius-full); letter-spacing: 0;
text-transform: none;
}
.row-wrap {
display: flex; align-items: center;
border-radius: var(--radius-md);
transition: background var(--t-fast);
}
.row-wrap:hover { background: var(--bg-raised); }
.row-wrap:hover .row-delete { opacity: 1; }
.row {
flex: 1; display: flex; align-items: center; gap: var(--sp-3);
padding: 8px var(--sp-2); border-radius: var(--radius-md);
border: none; background: none; text-align: left; cursor: pointer; min-width: 0;
}
.row:hover :global(.play-icon) { opacity: 1; }
.row-delete {
display: flex; align-items: center; justify-content: center;
flex-shrink: 0; width: 22px; height: 22px; border-radius: var(--radius-sm);
border: none; background: none; color: var(--text-faint); cursor: pointer;
opacity: 0; transition: opacity var(--t-base), color var(--t-base), background var(--t-base);
margin-right: var(--sp-1);
}
.row-delete:hover { color: var(--color-error); background: var(--color-error-bg); }
.thumb-wrap { position: relative; flex-shrink: 0; }
.thumb {
width: 36px; height: 52px; border-radius: var(--radius-sm); object-fit: cover;
display: block; background: var(--bg-raised); border: 1px solid var(--border-dim);
}
.session-badge {
position: absolute; bottom: -4px; right: -6px;
background: var(--accent-muted); border: 1px solid var(--accent-dim); color: var(--accent-fg);
font-family: var(--font-ui); font-size: 9px; font-weight: 600;
padding: 1px 4px; border-radius: 6px; line-height: 1.4; pointer-events: none;
}
.info { flex: 1; display: flex; flex-direction: column; gap: 3px; overflow: hidden; min-width: 0; }
.manga-title {
font-size: var(--text-base); font-weight: var(--weight-medium);
color: var(--text-secondary); white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.chapter-name {
font-size: var(--text-sm); color: var(--text-muted);
display: flex; align-items: center; gap: var(--sp-1); min-width: 0;
}
.chapter-range {
display: flex; align-items: center; gap: 5px;
white-space: nowrap; overflow: hidden; text-overflow: ellipsis; color: var(--text-muted);
}
.range-sep { color: var(--text-faint); font-size: 10px; flex-shrink: 0; }
.page-badge { font-family: var(--font-ui); font-size: var(--text-2xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); flex-shrink: 0; }
.time { font-family: var(--font-ui); font-size: var(--text-xs); color: var(--text-faint); letter-spacing: var(--tracking-wide); flex-shrink: 0; white-space: nowrap; }
:global(.play-icon) { color: var(--text-faint); flex-shrink: 0; opacity: 0; transition: opacity var(--t-base); }
.empty { flex: 1; display: flex; flex-direction: column; align-items: center; justify-content: center; gap: var(--sp-2); }
:global(.empty-icon) { color: var(--text-faint); }
.empty-text { font-size: var(--text-base); color: var(--text-muted); }
.empty-hint { font-size: var(--text-sm); color: var(--text-faint); }
@keyframes fadeIn { from { opacity: 0 } to { opacity: 1 } }
</style>