- Resume scroll position only when arriving via 继续阅读 (?resume=1).
Plain chapter-list / drawer clicks now actively scroll to top on mount.
- Progress format extended to {chapter, page, ratio} for within-page
precision; legacy bare-number and {chapter, page} still read correctly.
- Tappable skeleton logo (sunflower outline, spins) while a page loads;
tap force-fetches a fresh signed URL.
- Viewport-priority image loading: second IntersectionObserver at margin 0
marks truly-visible pages, drives <img fetchpriority="high"> and fires
immediate single-page fetches that cut the batch queue.
- Bounded image cache: unmount previous chapter's <img> elements when
currentPage > 5 into the new chapter; placeholders stay for layout.
One AbortController per live chapter; unmount aborts in-flight batches.
- Hashed chapter IDs on the wire via hashids; DB PKs unchanged.
- Origin/Referer allowlist + rate limiting on all /api/* routes via a
withGuards(opts, handler) wrapper (eliminates 6-line boilerplate x5).
- robots.txt allows Googlebot/Bingbot/Slurp/DuckDuckBot/Baiduspider/
YandexBot only; disallows /api/ for all UAs.
- Extract pure helpers for future TDD: lib/scroll-ratio.ts (calcScrollRatio,
scrollOffsetFromRatio), lib/progress.ts (parseProgress + injectable
StorageLike), lib/rate-limit.ts (optional { now, store, ipOf } deps),
lib/api-guards.ts.
- New env keys: HASHIDS_SALT, ALLOWED_ORIGINS (wired into docker-compose).
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
58 lines
1.5 KiB
TypeScript
58 lines
1.5 KiB
TypeScript
"use client";
|
|
|
|
import { useEffect, useState } from "react";
|
|
import Link from "next/link";
|
|
import { readProgress, type ReadingProgress } from "@/lib/progress";
|
|
|
|
type ChapterLite = {
|
|
number: number;
|
|
title: string;
|
|
};
|
|
|
|
type Props = {
|
|
mangaSlug: string;
|
|
chapters: ChapterLite[];
|
|
};
|
|
|
|
export { readProgress, writeProgress } from "@/lib/progress";
|
|
export type { ReadingProgress } from "@/lib/progress";
|
|
|
|
export function ReadingProgressButton({ mangaSlug, chapters }: Props) {
|
|
const [progress, setProgress] = useState<ReadingProgress | null>(null);
|
|
|
|
useEffect(() => {
|
|
setProgress(readProgress(mangaSlug));
|
|
}, [mangaSlug]);
|
|
|
|
if (chapters.length === 0) return null;
|
|
const first = chapters[0];
|
|
const resumeChapter =
|
|
progress !== null
|
|
? chapters.find((c) => c.number === progress.chapter)
|
|
: null;
|
|
const target = resumeChapter ?? first;
|
|
const href = resumeChapter
|
|
? `/manga/${mangaSlug}/${target.number}?resume=1`
|
|
: `/manga/${mangaSlug}/${target.number}`;
|
|
|
|
return (
|
|
<Link
|
|
href={href}
|
|
scroll={false}
|
|
className="flex items-center justify-center gap-3 w-full py-3 mb-6 px-4 text-sm font-semibold bg-accent hover:bg-accent-hover text-white rounded-xl transition-colors active:scale-[0.98]"
|
|
>
|
|
{resumeChapter ? (
|
|
<>
|
|
<span>继续阅读</span>
|
|
<span className="opacity-50">·</span>
|
|
<span className="truncate">
|
|
#{resumeChapter.number} {resumeChapter.title}
|
|
</span>
|
|
</>
|
|
) : (
|
|
"开始阅读"
|
|
)}
|
|
</Link>
|
|
);
|
|
}
|