mirror of
https://github.com/moku-project/Moku.git
synced 2026-06-13 09:19:56 -05:00
84 lines
3.0 KiB
TypeScript
84 lines
3.0 KiB
TypeScript
import { gql, getServerUrl } from "@api/client";
|
|
import { getBlobUrl, preloadBlobUrls } from "@core/cache/imageCache";
|
|
import { dedupeRequest } from "@core/async/batchRequests";
|
|
import { FETCH_CHAPTER_PAGES } from "@api/mutations/chapters";
|
|
|
|
const pageCache = new Map<number, string[]>();
|
|
const inflight = new Map<number, Promise<string[]>>();
|
|
const resolvedUrlCache = new Map<string, Promise<string>>();
|
|
const aspectCache = new Map<string, number>();
|
|
|
|
export function resolveUrl(url: string, useBlob: boolean, priority = 0): Promise<string> {
|
|
if (!useBlob) return Promise.resolve(url);
|
|
const cached = resolvedUrlCache.get(url);
|
|
if (cached) return cached;
|
|
const p = getBlobUrl(url, priority).catch(err => {
|
|
resolvedUrlCache.delete(url);
|
|
return Promise.reject(err);
|
|
});
|
|
resolvedUrlCache.set(url, p);
|
|
return p;
|
|
}
|
|
|
|
export function fetchPages(
|
|
chapterId: number,
|
|
useBlob: boolean,
|
|
signal?: AbortSignal,
|
|
priorityPage = 0,
|
|
): Promise<string[]> {
|
|
const cached = pageCache.get(chapterId);
|
|
if (cached) return Promise.resolve(cached);
|
|
if (signal?.aborted) return Promise.reject(new DOMException("Aborted", "AbortError"));
|
|
|
|
if (!inflight.has(chapterId)) {
|
|
const p = dedupeRequest(`chapter-pages:${chapterId}`, () =>
|
|
gql<{ fetchChapterPages: { pages: string[] } }>(FETCH_CHAPTER_PAGES, { chapterId })
|
|
.then(d => {
|
|
const urls = d.fetchChapterPages.pages.map(p => p.startsWith("http") ? p : `${getServerUrl()}${p}`);
|
|
if (useBlob && urls[priorityPage]) getBlobUrl(urls[priorityPage], 999);
|
|
pageCache.set(chapterId, urls);
|
|
return urls;
|
|
})
|
|
).finally(() => inflight.delete(chapterId));
|
|
inflight.set(chapterId, p);
|
|
}
|
|
|
|
const base = inflight.get(chapterId)!;
|
|
if (!signal) return base;
|
|
return new Promise((resolve, reject) => {
|
|
signal.addEventListener("abort", () => reject(new DOMException("Aborted", "AbortError")), { once: true });
|
|
base.then(resolve, reject);
|
|
});
|
|
}
|
|
|
|
export function measureAspect(url: string, useBlob: boolean): Promise<number> {
|
|
if (aspectCache.has(url)) return Promise.resolve(aspectCache.get(url)!);
|
|
return resolveUrl(url, useBlob).then(src => new Promise(res => {
|
|
const img = new Image();
|
|
img.onload = () => { const r = img.naturalHeight > 0 ? img.naturalWidth / img.naturalHeight : 0.67; aspectCache.set(url, r); res(r); };
|
|
img.onerror = () => res(0.67);
|
|
img.src = src;
|
|
}));
|
|
}
|
|
|
|
export function preloadImage(url: string, useBlob: boolean): void {
|
|
if (useBlob) { preloadBlobUrls([url], 0); return; }
|
|
resolveUrl(url, useBlob).then(src => { new Image().src = src; }).catch(() => {});
|
|
}
|
|
|
|
export function clearResolvedUrlCache(): void {
|
|
resolvedUrlCache.clear();
|
|
aspectCache.clear();
|
|
}
|
|
|
|
export function clearPageCache(chapterId?: number): void {
|
|
if (chapterId !== undefined) {
|
|
pageCache.delete(chapterId);
|
|
inflight.delete(chapterId);
|
|
} else {
|
|
pageCache.clear();
|
|
inflight.clear();
|
|
resolvedUrlCache.clear();
|
|
aspectCache.clear();
|
|
}
|
|
} |