mirror of
https://github.com/moku-project/Moku.git
synced 2026-06-13 01:09:56 -05:00
306 lines
10 KiB
Svelte
306 lines
10 KiB
Svelte
<script lang="ts">
|
|
import { onMount } from "svelte";
|
|
import { invoke } from "@tauri-apps/api/core";
|
|
import { listen } from "@tauri-apps/api/event";
|
|
import { getVersion } from "@tauri-apps/api/app";
|
|
import { getCurrentWindow } from "@tauri-apps/api/window";
|
|
import { gql } from "./lib/client";
|
|
import { GET_DOWNLOAD_STATUS } from "./lib/queries";
|
|
import { store, addToast, setActiveDownloads, setSettingsOpen } from "./store/state.svelte";
|
|
import type { DownloadStatus, DownloadQueueItem } from "./lib/types";
|
|
import Layout from "./components/layout/Layout.svelte";
|
|
import Reader from "./components/reader/Reader.svelte";
|
|
import Settings from "./components/settings/Settings.svelte";
|
|
import ThemeEditor from "./components/settings/ThemeEditor.svelte";
|
|
import TitleBar from "./components/layout/TitleBar.svelte";
|
|
import Toaster from "./components/layout/Toaster.svelte";
|
|
import SplashScreen from "./components/layout/SplashScreen.svelte";
|
|
import MangaPreview from "./components/shared/MangaPreview.svelte";
|
|
|
|
let themeStyleEl: HTMLStyleElement | null = null;
|
|
|
|
$effect(() => {
|
|
const themeId = store.settings.theme ?? "dark";
|
|
const isCustom = themeId.startsWith("custom:");
|
|
|
|
if (!isCustom) {
|
|
themeStyleEl?.remove();
|
|
themeStyleEl = null;
|
|
document.documentElement.setAttribute("data-theme", themeId);
|
|
return;
|
|
}
|
|
|
|
const custom = store.settings.customThemes?.find(t => t.id === themeId);
|
|
if (!custom) {
|
|
themeStyleEl?.remove();
|
|
themeStyleEl = null;
|
|
document.documentElement.setAttribute("data-theme", "dark");
|
|
return;
|
|
}
|
|
|
|
const vars = Object.entries(custom.tokens)
|
|
.map(([k, v]) => ` --${k}: ${v};`)
|
|
.join("\n");
|
|
const css = `[data-theme="custom"] {\n${vars}\n}`;
|
|
|
|
if (!themeStyleEl) {
|
|
themeStyleEl = document.createElement("style");
|
|
themeStyleEl.id = "moku-custom-theme";
|
|
document.head.appendChild(themeStyleEl);
|
|
}
|
|
themeStyleEl.textContent = css;
|
|
document.documentElement.setAttribute("data-theme", "custom");
|
|
});
|
|
|
|
let themeEditorOpen = $state(false);
|
|
let themeEditorEditId = $state<string | null>(null);
|
|
|
|
function openThemeEditor(id?: string | null) {
|
|
themeEditorEditId = id ?? null;
|
|
themeEditorOpen = true;
|
|
}
|
|
|
|
function closeThemeEditor() {
|
|
themeEditorOpen = false;
|
|
themeEditorEditId = null;
|
|
}
|
|
|
|
const MAX_ATTEMPTS = 10;
|
|
const win = getCurrentWindow();
|
|
|
|
let serverProbeOk = $state(false);
|
|
let appReady = $state(false);
|
|
let failed = $state(false);
|
|
let notConfigured = $state(false);
|
|
let idle = $state(false);
|
|
let devSplash = $state(false);
|
|
let platformScale = $state(1);
|
|
|
|
function applyZoom() {
|
|
const normalized = store.settings.uiScale * platformScale;
|
|
document.documentElement.style.zoom = `${normalized}%`;
|
|
document.documentElement.style.setProperty("--ui-scale", String(normalized));
|
|
document.documentElement.style.setProperty("--visual-vh", `${window.innerHeight / (normalized / 100)}px`);
|
|
}
|
|
|
|
let prevQueue: DownloadQueueItem[] = [];
|
|
let idleTimer: ReturnType<typeof setTimeout> | null = null;
|
|
let pollInterval: ReturnType<typeof setInterval>;
|
|
let unlistenDownload: (() => void) | undefined;
|
|
|
|
function detectCompletions(prev: DownloadQueueItem[], next: DownloadQueueItem[]) {
|
|
for (const item of prev) {
|
|
if (item.state !== "DOWNLOADING") continue;
|
|
if (!next.some(q => q.chapter.id === item.chapter.id)) {
|
|
const manga = item.chapter.manga;
|
|
addToast({ kind: "success", title: "Chapter downloaded",
|
|
body: manga ? `${manga.title} — ${item.chapter.name}` : item.chapter.name,
|
|
duration: 4000 });
|
|
}
|
|
}
|
|
}
|
|
|
|
function applyQueue(next: DownloadQueueItem[]) {
|
|
detectCompletions(prevQueue, next);
|
|
prevQueue = next;
|
|
setActiveDownloads(next.map(item => ({
|
|
chapterId: item.chapter.id, mangaId: item.chapter.mangaId, progress: item.progress,
|
|
})));
|
|
}
|
|
|
|
function resetIdle() {
|
|
if (idleTimer) clearTimeout(idleTimer);
|
|
if (idle) return;
|
|
const ms = (store.settings.idleTimeoutMin ?? 5) * 60 * 1000;
|
|
if (ms === 0) return;
|
|
idleTimer = setTimeout(() => idle = true, ms);
|
|
}
|
|
|
|
const idleEvents = ["mousemove", "mousedown", "keydown", "touchstart", "wheel"] as const;
|
|
|
|
$effect(() => {
|
|
if (!appReady) return;
|
|
idleEvents.forEach(e => window.addEventListener(e, resetIdle, { passive: true }));
|
|
resetIdle();
|
|
return () => idleEvents.forEach(e => window.removeEventListener(e, resetIdle));
|
|
});
|
|
|
|
$effect(() => {
|
|
store.settings.uiScale; platformScale;
|
|
applyZoom();
|
|
});
|
|
|
|
$effect(() => {
|
|
if (!appReady) return;
|
|
const poll = () => gql<{ downloadStatus: DownloadStatus }>(GET_DOWNLOAD_STATUS)
|
|
.then(d => applyQueue(d.downloadStatus.queue)).catch(console.error);
|
|
poll();
|
|
pollInterval = setInterval(poll, 2000);
|
|
return () => clearInterval(pollInterval);
|
|
});
|
|
|
|
async function checkForUpdateSilently() {
|
|
try {
|
|
const [currentVersion, releases] = await Promise.all([
|
|
getVersion(),
|
|
invoke<Array<{ tag_name: string; html_url: string }>>("list_releases"),
|
|
]);
|
|
|
|
const valid = releases.filter(r => typeof r.tag_name === "string" && r.tag_name.trim());
|
|
if (!valid.length) return;
|
|
|
|
const parse = (tag: string): number[] =>
|
|
tag.replace(/^v/, "").split(".").map(Number);
|
|
|
|
const compare = (a: number[], b: number[]): number => {
|
|
for (let i = 0; i < 3; i++) {
|
|
if ((a[i] ?? 0) !== (b[i] ?? 0)) return (b[i] ?? 0) - (a[i] ?? 0);
|
|
}
|
|
return 0;
|
|
};
|
|
|
|
const latestTag = valid
|
|
.map(r => r.tag_name)
|
|
.sort((a, b) => compare(parse(a), parse(b)))[0]
|
|
.replace(/^v/, "");
|
|
|
|
const isNewer = compare(parse(latestTag), parse(currentVersion)) < 0;
|
|
if (isNewer) {
|
|
addToast({
|
|
kind: "info",
|
|
title: `Update available — v${latestTag}`,
|
|
body: "Open Settings → About to install.",
|
|
duration: 8000,
|
|
});
|
|
}
|
|
} catch {}
|
|
}
|
|
|
|
let cancelProbe = false;
|
|
|
|
function startProbe() {
|
|
cancelProbe = false;
|
|
failed = false;
|
|
let tries = 0;
|
|
|
|
async function probe() {
|
|
if (cancelProbe) return;
|
|
tries++;
|
|
try {
|
|
const rawUrl = store.settings.serverUrl;
|
|
const base = typeof rawUrl === "string" && rawUrl.trim()
|
|
? rawUrl.replace(/\/$/, "")
|
|
: "http://127.0.0.1:4567";
|
|
const s = store.settings;
|
|
const auth: Record<string, string> = s.serverAuthEnabled && s.serverAuthUser && s.serverAuthPass
|
|
? { Authorization: `Basic ${btoa(`${s.serverAuthUser.trim()}:${s.serverAuthPass.trim()}`)}` }
|
|
: {};
|
|
const res = await fetch(`${base}/api/graphql`, {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json", ...auth },
|
|
body: JSON.stringify({ query: "{ __typename }" }),
|
|
signal: AbortSignal.timeout(2000),
|
|
});
|
|
if (res.ok && !cancelProbe) { serverProbeOk = true; return; }
|
|
} catch {}
|
|
if (tries >= MAX_ATTEMPTS && !cancelProbe) { failed = true; return; }
|
|
if (!cancelProbe) setTimeout(probe, 750);
|
|
}
|
|
|
|
setTimeout(probe, 800);
|
|
}
|
|
|
|
onMount(async () => {
|
|
document.addEventListener("contextmenu", e => e.preventDefault());
|
|
(window as any).__mokuShowSplash = () => devSplash = true;
|
|
|
|
platformScale = await invoke<number>("get_platform_ui_scale").catch(() => 1);
|
|
applyZoom();
|
|
|
|
store.isFullscreen = await win.isFullscreen();
|
|
const unlistenResize = await win.onResized(async () => {
|
|
store.isFullscreen = await win.isFullscreen();
|
|
});
|
|
|
|
if (store.settings.autoStartServer) {
|
|
invoke<void>("spawn_server", { binary: store.settings.serverBinary }).catch((err: any) => {
|
|
if (err?.kind === "NotConfigured") {
|
|
notConfigured = true;
|
|
} else {
|
|
console.warn("Could not start server:", err);
|
|
}
|
|
});
|
|
}
|
|
|
|
startProbe();
|
|
|
|
type P = { chapterId: number; mangaId: number; progress: number }[];
|
|
unlistenDownload = await listen<P>("download-progress", e => { setActiveDownloads(e.payload); });
|
|
|
|
return () => {
|
|
cancelProbe = true;
|
|
unlistenResize();
|
|
if (store.settings.autoStartServer) invoke("kill_server").catch(() => {});
|
|
if (idleTimer) clearTimeout(idleTimer);
|
|
if (pollInterval) clearInterval(pollInterval);
|
|
unlistenDownload?.();
|
|
delete (window as any).__mokuShowSplash;
|
|
};
|
|
});
|
|
|
|
$effect(() => {
|
|
if (!appReady) return;
|
|
const timer = setTimeout(checkForUpdateSilently, 5_000);
|
|
return () => clearTimeout(timer);
|
|
});
|
|
|
|
function handleRetry() {
|
|
failed = false;
|
|
notConfigured = false;
|
|
serverProbeOk = false;
|
|
startProbe();
|
|
}
|
|
|
|
function handleBypass() {
|
|
cancelProbe = true;
|
|
serverProbeOk = true;
|
|
appReady = true;
|
|
}
|
|
</script>
|
|
|
|
{#if devSplash}
|
|
<SplashScreen mode="idle" showFps showCards={store.settings.splashCards ?? true}
|
|
onDismiss={() => setTimeout(() => devSplash = false, 340)} />
|
|
{:else if !appReady}
|
|
<SplashScreen mode="loading" ringFull={serverProbeOk} {failed} {notConfigured}
|
|
showCards={store.settings.splashCards ?? true}
|
|
onReady={() => appReady = true}
|
|
onRetry={handleRetry}
|
|
onBypass={handleBypass} />
|
|
{:else}
|
|
<div class="root">
|
|
{#if idle && !store.activeChapter}
|
|
<SplashScreen mode="idle" showCards={store.settings.splashCards ?? true}
|
|
onDismiss={() => { idle = false; resetIdle(); }} />
|
|
{/if}
|
|
{#if !store.activeChapter && !store.isFullscreen}<TitleBar />{/if}
|
|
<div class="content">
|
|
{#if store.activeChapter}<Reader />{:else}<Layout />{/if}
|
|
</div>
|
|
{#if store.settingsOpen}<Settings onOpenThemeEditor={openThemeEditor} />{/if}
|
|
{#if themeEditorOpen}
|
|
<ThemeEditor
|
|
bind:editingId={themeEditorEditId}
|
|
onClose={closeThemeEditor}
|
|
/>
|
|
{/if}
|
|
<MangaPreview />
|
|
<Toaster />
|
|
</div>
|
|
{/if}
|
|
|
|
<style>
|
|
.root { display: flex; flex-direction: column; height: 100%; overflow: hidden; }
|
|
.content { flex: 1; overflow: hidden; }
|
|
</style>
|