Files
pinkudex/components/select/SelectionBar.tsx
T
2026-05-26 22:46:00 +02:00

107 lines
4.5 KiB
TypeScript

"use client";
import { useTransition } from "react";
import { useSelection } from "./SelectionProvider";
import { useRouter } from "next/navigation";
import { Trash2, X, ListChecks } from "lucide-react";
import { deleteImages, bulkSetWatched, bulkSetMark } from "@/app/actions/bulk";
import { useUndoDeleteToast } from "@/components/select/UndoDeleteToast";
import { useSettings } from "@/components/settings/SettingsProvider";
import { MarkAsMenu } from "./MarkAsMenu";
import { useWatchQueue } from "@/components/queue/WatchQueueProvider";
import { dispatchQueueRemove } from "@/components/queue/watchQueueEvents";
import { ListVideo } from "lucide-react";
export function SelectionBar() {
const { ids, clear, visibleIds, selectMany } = useSelection();
const { settings } = useSettings();
const { show: showUndo } = useUndoDeleteToast();
const [pending, start] = useTransition();
const router = useRouter();
const queue = useWatchQueue();
if (ids.size === 0) return null;
const count = ids.size;
const allVisibleSelected = visibleIds.length > 0 && visibleIds.every((id) => ids.has(id));
const onDelete = (e: React.MouseEvent) => {
const permanent = e.shiftKey || !settings.useRecycleBin;
if (permanent) {
if (!confirm(`Permanently delete ${count} cover${count === 1 ? "" : "s"}? Cannot be undone.`)) return;
}
const targetIds = Array.from(ids);
start(async () => {
await deleteImages(targetIds, permanent ? { permanent: true } : undefined);
clear();
router.refresh();
if (!permanent) showUndo(targetIds);
});
};
const onSelectAllToggle = () => {
if (allVisibleSelected) clear();
else selectMany(visibleIds);
};
const onMarkAs = (action: "watched" | "unwatched" | "vip" | "favorite" | "unmark") => {
const targetIds = Array.from(ids);
start(async () => {
try {
if (action === "watched") { await bulkSetWatched(targetIds, true); dispatchQueueRemove(targetIds); }
else if (action === "unwatched") await bulkSetWatched(targetIds, false);
else if (action === "vip") await bulkSetMark(targetIds, "vip");
else if (action === "favorite") await bulkSetMark(targetIds, "favorite");
else if (action === "unmark") await bulkSetMark(targetIds, "unmarked");
router.refresh();
} catch (err) {
console.error(`[bulk ${action}] failed:`, err);
}
});
};
return (
<div className="fixed bottom-[80px] left-1/2 -translate-x-1/2 z-50">
<div
className="rounded-2xl shadow-2xl px-4 py-2.5 flex items-center gap-3 border border-[var(--color-glass-border-strong)] backdrop-blur-2xl"
style={{ background: "color-mix(in oklch, var(--color-bg-0) 85%, transparent)" }}
>
<span className="text-sm font-mono tabular-nums">
<span className="text-[var(--color-cyan)] font-semibold">{count}</span>
<span className="text-[var(--color-fg-dim)]"> selected</span>
</span>
<div className="w-px h-5 bg-[var(--color-glass-border)]" />
{visibleIds.length > 0 && (
<button
onClick={onSelectAllToggle}
className="flex items-center gap-1.5 text-xs px-3 py-1.5 rounded-lg glass hover:text-[var(--color-fg)]"
>
<ListChecks className="w-3.5 h-3.5" />
{allVisibleSelected ? "Deselect All" : `All (${visibleIds.length})`}
</button>
)}
<button
onClick={() => { queue.addMany(Array.from(ids)); }}
title="Add to watch queue"
className="flex items-center gap-1.5 text-xs px-3 py-1.5 rounded-lg glass hover:text-[var(--color-fg)]"
>
<ListVideo className="w-3.5 h-3.5" /> Queue
</button>
<MarkAsMenu onAction={onMarkAs} disabled={pending} />
<button
onClick={onDelete}
disabled={pending}
title={settings.useRecycleBin ? "Send to trash · Shift-click for permanent delete" : "Delete permanently"}
className="flex items-center gap-1.5 text-xs px-3 py-1.5 rounded-lg bg-[var(--color-coral)]/15 text-[var(--color-coral)] border border-[var(--color-coral)]/40 hover:bg-[var(--color-coral)]/25 disabled:opacity-40"
>
<Trash2 className="w-3.5 h-3.5" /> {pending ? "Deleting…" : "Delete"}
</button>
<button
onClick={clear}
className="flex items-center gap-1.5 text-xs px-2 py-1.5 rounded-lg text-[var(--color-fg-dim)] hover:text-[var(--color-fg)]"
>
<X className="w-3.5 h-3.5" /> Clear
</button>
</div>
</div>
);
}