189 lines
6.0 KiB
TypeScript
189 lines
6.0 KiB
TypeScript
"use client";
|
|
|
|
import {
|
|
ArrowUpRight,
|
|
FileText,
|
|
Link,
|
|
LoaderCircle,
|
|
MoreHorizontal,
|
|
RefreshCw,
|
|
StarOff,
|
|
Trash2,
|
|
} from "lucide-react";
|
|
|
|
import {
|
|
DropdownMenu,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuSeparator,
|
|
DropdownMenuTrigger,
|
|
} from "@/components/ui/dropdown-menu";
|
|
import {
|
|
SidebarGroup,
|
|
SidebarGroupLabel,
|
|
SidebarMenu,
|
|
SidebarMenuAction,
|
|
SidebarMenuButton,
|
|
SidebarMenuItem,
|
|
useSidebar,
|
|
} from "@/components/ui/sidebar";
|
|
import { createClient } from "@/utils/supabase/client";
|
|
import { toast } from "sonner";
|
|
import { SSE } from "sse.js";
|
|
import { useEffect, useState } from "react";
|
|
|
|
export function NavDocuments({
|
|
documents: ogDocuments,
|
|
}: {
|
|
documents: {
|
|
id: string;
|
|
disabled?: boolean;
|
|
name: string;
|
|
url: string;
|
|
emoji?: string;
|
|
}[];
|
|
}) {
|
|
const { isMobile } = useSidebar();
|
|
const supabase = createClient();
|
|
const [documents, setDocuments] = useState(ogDocuments);
|
|
|
|
useEffect(() => {
|
|
// watch for changes in the documents table, update the state when it changes
|
|
|
|
const handleRecordInserted = (payload: any) => {
|
|
const newDocument = payload.new;
|
|
setDocuments((prev) => [...prev, newDocument]);
|
|
};
|
|
const handleRecordUpdated = (payload: any) => {
|
|
const updatedDocument = payload.new;
|
|
setDocuments((prev) =>
|
|
prev.map((doc) =>
|
|
doc.id === updatedDocument.id ? updatedDocument : doc
|
|
)
|
|
);
|
|
};
|
|
const handleRecordDeleted = (payload: any) => {
|
|
const deletedDocument = payload.old;
|
|
setDocuments((prev) =>
|
|
prev.filter((doc) => doc.id !== deletedDocument.id)
|
|
);
|
|
};
|
|
|
|
const subscription = supabase
|
|
.channel("documents")
|
|
.on(
|
|
"postgres_changes",
|
|
{ event: "INSERT", schema: "public", table: "documents" },
|
|
handleRecordInserted
|
|
)
|
|
.on(
|
|
"postgres_changes",
|
|
{ event: "UPDATE", schema: "public", table: "documents" },
|
|
handleRecordUpdated
|
|
)
|
|
.on(
|
|
"postgres_changes",
|
|
{ event: "DELETE", schema: "public", table: "documents" },
|
|
handleRecordDeleted
|
|
)
|
|
.subscribe();
|
|
|
|
return () => {
|
|
subscription.unsubscribe();
|
|
};
|
|
}, [ogDocuments, supabase]);
|
|
|
|
return (
|
|
<SidebarGroup className="group-data-[collapsible=icon]:hidden">
|
|
<SidebarGroupLabel>Documents</SidebarGroupLabel>
|
|
<SidebarMenu>
|
|
{documents.map((item) => (
|
|
<SidebarMenuItem key={item.id} aria-disabled={item.disabled}>
|
|
<SidebarMenuButton asChild disabled={item.disabled}>
|
|
<a href={item.url} title={item.name}>
|
|
{item.disabled ? (
|
|
<LoaderCircle className="animate-spin text-muted-foreground" />
|
|
) : (
|
|
<span>{item.emoji ? item.emoji : <FileText />}</span>
|
|
)}
|
|
<span>{item.name}</span>
|
|
</a>
|
|
</SidebarMenuButton>
|
|
<DropdownMenu>
|
|
<DropdownMenuTrigger asChild>
|
|
<SidebarMenuAction showOnHover>
|
|
<MoreHorizontal />
|
|
<span className="sr-only">More</span>
|
|
</SidebarMenuAction>
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent
|
|
className="w-56 rounded-lg"
|
|
side={isMobile ? "bottom" : "right"}
|
|
align={isMobile ? "end" : "start"}
|
|
>
|
|
<DropdownMenuItem
|
|
onClick={async () => {
|
|
const data = new FormData();
|
|
|
|
const session = await supabase.auth.getSession();
|
|
if (!session.data.session) {
|
|
toast.error("You are not logged in");
|
|
return;
|
|
}
|
|
|
|
data.append("id", item.id);
|
|
data.append(
|
|
"access_token",
|
|
session.data.session.access_token
|
|
);
|
|
data.append(
|
|
"refresh_token",
|
|
session.data.session.refresh_token
|
|
);
|
|
|
|
const eventSource = new SSE(`/api/process-document`, {
|
|
payload: data,
|
|
headers: {
|
|
apikey: process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!,
|
|
Authorization: `Bearer ${process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY}`,
|
|
},
|
|
method: "POST",
|
|
});
|
|
|
|
toast.loading("Reprocessing document...");
|
|
|
|
eventSource.onmessage = (event) => {
|
|
const message = JSON.parse(event.data);
|
|
if (message.status === "success") {
|
|
toast.success("Document reprocessed successfully");
|
|
eventSource.close();
|
|
} else if (message.status === "error") {
|
|
toast.error("Failed to reprocess document");
|
|
eventSource.close();
|
|
}
|
|
};
|
|
|
|
eventSource.onerror = (err) => {
|
|
console.error("SSE error:", err);
|
|
toast.error("An error occurred while reprocessing");
|
|
eventSource.close();
|
|
};
|
|
}}
|
|
>
|
|
<RefreshCw className="text-muted-foreground" />
|
|
<span>Reprocess Document</span>
|
|
</DropdownMenuItem>
|
|
<DropdownMenuSeparator />
|
|
<DropdownMenuItem>
|
|
<Trash2 className="text-muted-foreground" />
|
|
<span>Delete</span>
|
|
</DropdownMenuItem>
|
|
</DropdownMenuContent>
|
|
</DropdownMenu>
|
|
</SidebarMenuItem>
|
|
))}
|
|
</SidebarMenu>
|
|
</SidebarGroup>
|
|
);
|
|
}
|