mirror of
https://github.com/moku-project/Moku.git
synced 2026-06-13 09:19:56 -05:00
Feat: Notifications on Source Install/Uninstall (#31)
This commit is contained in:
@@ -38,8 +38,6 @@ In-Progress:
|
|||||||
|
|
||||||
- Add Small QOL Animations where Appropriate
|
- Add Small QOL Animations where Appropriate
|
||||||
|
|
||||||
- Add Descriptions to Reader Auto Features in Settings (Auto-Advance, etc)
|
|
||||||
|
|
||||||
- Add Flathub Support (Pending Video)
|
- Add Flathub Support (Pending Video)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
import { gql } from "../../lib/client";
|
import { gql } from "../../lib/client";
|
||||||
import Thumbnail from "../shared/Thumbnail.svelte";
|
import Thumbnail from "../shared/Thumbnail.svelte";
|
||||||
import { GET_EXTENSIONS, FETCH_EXTENSIONS, UPDATE_EXTENSION, INSTALL_EXTERNAL_EXTENSION, GET_SETTINGS, SET_EXTENSION_REPOS } from "../../lib/queries";
|
import { GET_EXTENSIONS, FETCH_EXTENSIONS, UPDATE_EXTENSION, INSTALL_EXTERNAL_EXTENSION, GET_SETTINGS, SET_EXTENSION_REPOS } from "../../lib/queries";
|
||||||
import { store } from "../../store/state.svelte";
|
import { store, addToast } from "../../store/state.svelte";
|
||||||
import type { Extension } from "../../lib/types";
|
import type { Extension } from "../../lib/types";
|
||||||
|
|
||||||
type Filter = "installed" | "available" | "updates" | "all";
|
type Filter = "installed" | "available" | "updates" | "all";
|
||||||
@@ -66,11 +66,24 @@
|
|||||||
|
|
||||||
function removeRepo(url: string) { saveRepos(repos.filter((r) => r !== url)); }
|
function removeRepo(url: string) { saveRepos(repos.filter((r) => r !== url)); }
|
||||||
|
|
||||||
async function mutate(fn: () => Promise<unknown>, pkgName: string) {
|
async function mutate(fn: () => Promise<unknown>, pkgName: string, op: "install" | "update" | "uninstall") {
|
||||||
working = new Set(working).add(pkgName);
|
working = new Set(working).add(pkgName);
|
||||||
await fn().catch(console.error);
|
const label = extensions.find((e) => e.pkgName === pkgName)?.name ?? pkgName;
|
||||||
await load();
|
try {
|
||||||
working.delete(pkgName); working = new Set(working);
|
await fn();
|
||||||
|
await load();
|
||||||
|
const toastMap = {
|
||||||
|
install: { kind: "download" as const, title: "Extension installed", body: label },
|
||||||
|
update: { kind: "success" as const, title: "Extension updated", body: label },
|
||||||
|
uninstall: { kind: "info" as const, title: "Extension removed", body: label },
|
||||||
|
};
|
||||||
|
addToast(toastMap[op]);
|
||||||
|
} catch (e: any) {
|
||||||
|
await load();
|
||||||
|
addToast({ kind: "error", title: "Extension error", body: e instanceof Error ? e.message : String(e) });
|
||||||
|
} finally {
|
||||||
|
working.delete(pkgName); working = new Set(working);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function installExternal() {
|
async function installExternal() {
|
||||||
@@ -83,8 +96,12 @@
|
|||||||
await gql(INSTALL_EXTERNAL_EXTENSION, { url });
|
await gql(INSTALL_EXTERNAL_EXTENSION, { url });
|
||||||
installSuccess = true; externalUrl = "";
|
installSuccess = true; externalUrl = "";
|
||||||
await load();
|
await load();
|
||||||
|
addToast({ kind: "download", title: "Extension installed", body: url.split("/").pop() ?? url });
|
||||||
setTimeout(() => { panel = null; installSuccess = false; }, 1500);
|
setTimeout(() => { panel = null; installSuccess = false; }, 1500);
|
||||||
} catch (e: any) { installError = e instanceof Error ? e.message : "Install failed"; }
|
} catch (e: any) {
|
||||||
|
installError = e instanceof Error ? e.message : "Install failed";
|
||||||
|
addToast({ kind: "error", title: "Install failed", body: installError });
|
||||||
|
}
|
||||||
finally { installing = false; }
|
finally { installing = false; }
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -255,13 +272,13 @@
|
|||||||
<CircleNotch size={14} weight="light" class="anim-spin" style="color:var(--text-faint)" />
|
<CircleNotch size={14} weight="light" class="anim-spin" style="color:var(--text-faint)" />
|
||||||
{:else if primary.hasUpdate}
|
{:else if primary.hasUpdate}
|
||||||
<div class="row-actions">
|
<div class="row-actions">
|
||||||
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, update: true }), primary.pkgName)}>Update</button>
|
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, update: true }), primary.pkgName, "update")}>Update</button>
|
||||||
<button class="action-btn-dim" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, uninstall: true }), primary.pkgName)}>Remove</button>
|
<button class="action-btn-dim" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, uninstall: true }), primary.pkgName, "uninstall")}>Remove</button>
|
||||||
</div>
|
</div>
|
||||||
{:else if primary.isInstalled}
|
{:else if primary.isInstalled}
|
||||||
<button class="action-btn-dim" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, uninstall: true }), primary.pkgName)}>Remove</button>
|
<button class="action-btn-dim" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, uninstall: true }), primary.pkgName, "uninstall")}>Remove</button>
|
||||||
{:else}
|
{:else}
|
||||||
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, install: true }), primary.pkgName)}>Install</button>
|
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: primary.pkgName, install: true }), primary.pkgName, "install")}>Install</button>
|
||||||
{/if}
|
{/if}
|
||||||
{#if hasVariants}
|
{#if hasVariants}
|
||||||
<button class="expand-btn" onclick={() => toggleExpand(base)} title="{variants.length + 1} languages">
|
<button class="expand-btn" onclick={() => toggleExpand(base)} title="{variants.length + 1} languages">
|
||||||
@@ -282,11 +299,11 @@
|
|||||||
{#if working.has(v.pkgName)}
|
{#if working.has(v.pkgName)}
|
||||||
<CircleNotch size={14} weight="light" class="anim-spin" style="color:var(--text-faint)" />
|
<CircleNotch size={14} weight="light" class="anim-spin" style="color:var(--text-faint)" />
|
||||||
{:else if v.hasUpdate}
|
{:else if v.hasUpdate}
|
||||||
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: v.pkgName, update: true }), v.pkgName)}>Update</button>
|
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: v.pkgName, update: true }), v.pkgName, "update")}>Update</button>
|
||||||
{:else if v.isInstalled}
|
{:else if v.isInstalled}
|
||||||
<button class="action-btn-dim" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: v.pkgName, uninstall: true }), v.pkgName)}>Remove</button>
|
<button class="action-btn-dim" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: v.pkgName, uninstall: true }), v.pkgName, "uninstall")}>Remove</button>
|
||||||
{:else}
|
{:else}
|
||||||
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: v.pkgName, install: true }), v.pkgName)}>Install</button>
|
<button class="action-btn" onclick={() => mutate(() => gql(UPDATE_EXTENSION, { id: v.pkgName, install: true }), v.pkgName, "install")}>Install</button>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
import { gql } from "../../lib/client";
|
import { gql } from "../../lib/client";
|
||||||
import Thumbnail from "../shared/Thumbnail.svelte";
|
import Thumbnail from "../shared/Thumbnail.svelte";
|
||||||
import { FETCH_SOURCE_MANGA, UPDATE_MANGA, GET_CATEGORIES, CREATE_CATEGORY, UPDATE_MANGA_CATEGORIES } from "../../lib/queries";
|
import { FETCH_SOURCE_MANGA, UPDATE_MANGA, GET_CATEGORIES, CREATE_CATEGORY, UPDATE_MANGA_CATEGORIES } from "../../lib/queries";
|
||||||
import { store, setActiveSource, setActiveManga, setNavPage } from "../../store/state.svelte";
|
import { store, setActiveSource, setActiveManga, setNavPage, addToast } from "../../store/state.svelte";
|
||||||
import type { Manga, Category } from "../../lib/types";
|
import type { Manga, Category } from "../../lib/types";
|
||||||
import ContextMenu, { type MenuEntry } from "../shared/ContextMenu.svelte";
|
import ContextMenu, { type MenuEntry } from "../shared/ContextMenu.svelte";
|
||||||
|
|
||||||
@@ -58,8 +58,14 @@
|
|||||||
return [
|
return [
|
||||||
{ label: m.inLibrary ? "In Library" : "Add to library", icon: BookmarkSimple, disabled: m.inLibrary,
|
{ label: m.inLibrary ? "In Library" : "Add to library", icon: BookmarkSimple, disabled: m.inLibrary,
|
||||||
onClick: () => gql(UPDATE_MANGA, { id: m.id, inLibrary: true })
|
onClick: () => gql(UPDATE_MANGA, { id: m.id, inLibrary: true })
|
||||||
.then(() => mangas = mangas.map((x) => x.id === m.id ? { ...x, inLibrary: true } : x))
|
.then(() => {
|
||||||
.catch(console.error) },
|
mangas = mangas.map((x) => x.id === m.id ? { ...x, inLibrary: true } : x);
|
||||||
|
addToast({ kind: "success", title: "Added to library", body: m.title });
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
addToast({ kind: "error", title: "Failed to add to library", body: m.title });
|
||||||
|
console.error(e);
|
||||||
|
}) },
|
||||||
...(categories.length > 0 ? [
|
...(categories.length > 0 ? [
|
||||||
{ separator: true } as MenuEntry,
|
{ separator: true } as MenuEntry,
|
||||||
...categories.map((cat): MenuEntry => ({
|
...categories.map((cat): MenuEntry => ({
|
||||||
|
|||||||
Reference in New Issue
Block a user