mirror of
https://github.com/xCyanGrizzly/DragonsStash.git
synced 2026-05-10 22:01:16 +00:00
- Auto-extract preview images from ZIP/RAR/7z archives during ingestion - Upload custom preview images via package drawer - Select preview from archive contents with on-demand extraction UI - Manually add Telegram channels by t.me link, username, or invite link - Invite code UX: bulk create, copy link, usage tracking, delete confirm - Incomplete upload recovery: verify dest messages on worker startup - Rebuild package DB by scanning destination channel with live progress Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
173 lines
5.7 KiB
TypeScript
173 lines
5.7 KiB
TypeScript
"use client";
|
|
|
|
import { useState, useTransition } from "react";
|
|
import { toast } from "sonner";
|
|
import { Download, Plus } from "lucide-react";
|
|
import { getChannelColumns } from "./channel-columns";
|
|
import { DestinationCard } from "./destination-card";
|
|
import { ChannelPickerDialog } from "./channel-picker-dialog";
|
|
import { JoinChannelDialog } from "./join-channel-dialog";
|
|
import {
|
|
deleteChannel,
|
|
toggleChannelActive,
|
|
setChannelType,
|
|
setChannelCategory,
|
|
rescanChannel,
|
|
} from "../actions";
|
|
import { DataTable } from "@/components/shared/data-table";
|
|
import { DeleteDialog } from "@/components/shared/delete-dialog";
|
|
import { Button } from "@/components/ui/button";
|
|
import type { AccountRow, ChannelRow, GlobalDestination } from "@/lib/telegram/admin-queries";
|
|
import { useDataTable } from "@/hooks/use-data-table";
|
|
|
|
interface ChannelsTabProps {
|
|
channels: ChannelRow[];
|
|
globalDestination: GlobalDestination;
|
|
accounts: AccountRow[];
|
|
}
|
|
|
|
export function ChannelsTab({ channels, globalDestination, accounts }: ChannelsTabProps) {
|
|
const [isPending, startTransition] = useTransition();
|
|
const [deleteId, setDeleteId] = useState<string | null>(null);
|
|
const [rescanId, setRescanId] = useState<string | null>(null);
|
|
const [fetchChannelsAccountId, setFetchChannelsAccountId] = useState<string | null>(null);
|
|
const [joinDialogOpen, setJoinDialogOpen] = useState(false);
|
|
|
|
// Find the first authenticated account for "Fetch Channels"
|
|
const authenticatedAccounts = accounts.filter((a) => a.authState === "AUTHENTICATED" && a.isActive);
|
|
|
|
const columns = getChannelColumns({
|
|
onToggleActive: (id) => {
|
|
startTransition(async () => {
|
|
const result = await toggleChannelActive(id);
|
|
if (result.success) toast.success("Channel toggled");
|
|
else toast.error(result.error);
|
|
});
|
|
},
|
|
onDelete: (id) => setDeleteId(id),
|
|
onSetType: (id, type) => {
|
|
startTransition(async () => {
|
|
const result = await setChannelType(id, type);
|
|
if (result.success) toast.success(`Channel set as ${type.toLowerCase()}`);
|
|
else toast.error(result.error);
|
|
});
|
|
},
|
|
onRescan: (id) => setRescanId(id),
|
|
onSetCategory: (id, category) => {
|
|
startTransition(async () => {
|
|
const result = await setChannelCategory(id, category);
|
|
if (result.success) toast.success(category ? `Category set to "${category}"` : "Category removed");
|
|
else toast.error(result.error);
|
|
});
|
|
},
|
|
});
|
|
|
|
const { table } = useDataTable({
|
|
data: channels,
|
|
columns,
|
|
pageCount: 1,
|
|
});
|
|
|
|
const handleDelete = () => {
|
|
if (!deleteId) return;
|
|
startTransition(async () => {
|
|
const result = await deleteChannel(deleteId);
|
|
if (result.success) {
|
|
toast.success("Channel deleted");
|
|
setDeleteId(null);
|
|
} else {
|
|
toast.error(result.error);
|
|
}
|
|
});
|
|
};
|
|
|
|
const handleRescan = () => {
|
|
if (!rescanId) return;
|
|
startTransition(async () => {
|
|
const result = await rescanChannel(rescanId);
|
|
if (result.success) {
|
|
toast.success("Channel scan progress reset — it will be fully rescanned on the next sync");
|
|
setRescanId(null);
|
|
} else {
|
|
toast.error(result.error);
|
|
}
|
|
});
|
|
};
|
|
|
|
const handleFetchChannels = () => {
|
|
if (authenticatedAccounts.length > 0) {
|
|
setFetchChannelsAccountId(authenticatedAccounts[0].id);
|
|
} else {
|
|
toast.error("No authenticated accounts available. Add and authenticate an account first.");
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className="space-y-4">
|
|
<DestinationCard destination={globalDestination} channels={channels} />
|
|
|
|
<div className="flex items-center gap-2">
|
|
<Button
|
|
variant="outline"
|
|
onClick={handleFetchChannels}
|
|
disabled={authenticatedAccounts.length === 0}
|
|
>
|
|
<Download className="mr-2 h-4 w-4" />
|
|
Fetch Channels
|
|
</Button>
|
|
<Button
|
|
variant="outline"
|
|
onClick={() => setJoinDialogOpen(true)}
|
|
disabled={authenticatedAccounts.length === 0}
|
|
>
|
|
<Plus className="mr-2 h-4 w-4" />
|
|
Add Channel
|
|
</Button>
|
|
</div>
|
|
|
|
{channels.length > 0 && (
|
|
<p className="text-xs text-muted-foreground">
|
|
Channels discovered via "Fetch Channels" are automatically activated as sources.
|
|
</p>
|
|
)}
|
|
|
|
<DataTable
|
|
table={table}
|
|
emptyMessage="No channels yet. Click "Fetch Channels" above to discover and add source channels."
|
|
/>
|
|
|
|
<DeleteDialog
|
|
open={!!deleteId}
|
|
onOpenChange={(open) => !open && setDeleteId(null)}
|
|
title="Delete Channel"
|
|
description="This will permanently delete this channel and unlink it from all accounts. Existing packages will NOT be deleted."
|
|
onConfirm={handleDelete}
|
|
isLoading={isPending}
|
|
/>
|
|
|
|
<DeleteDialog
|
|
open={!!rescanId}
|
|
onOpenChange={(open) => !open && setRescanId(null)}
|
|
title="Rescan Channel"
|
|
description="This will reset all scan progress for this channel. On the next sync the worker will re-process every message from the beginning. Packages that are already in the library will be skipped (deduplication by hash), but any missing files will be re-downloaded and re-uploaded. This may take a long time for large channels."
|
|
confirmLabel="Rescan"
|
|
onConfirm={handleRescan}
|
|
isLoading={isPending}
|
|
/>
|
|
|
|
<ChannelPickerDialog
|
|
accountId={fetchChannelsAccountId}
|
|
open={!!fetchChannelsAccountId}
|
|
onOpenChange={(open) => {
|
|
if (!open) setFetchChannelsAccountId(null);
|
|
}}
|
|
/>
|
|
|
|
<JoinChannelDialog
|
|
open={joinDialogOpen}
|
|
onOpenChange={setJoinDialogOpen}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|