mirror of
https://github.com/MODSetter/SurfSense.git
synced 2025-09-02 10:39:13 +00:00
309 lines
No EOL
9.8 KiB
TypeScript
309 lines
No EOL
9.8 KiB
TypeScript
'use client';
|
|
|
|
import { useEffect, useState } from 'react';
|
|
import { AppSidebar } from '@/components/sidebar/app-sidebar';
|
|
import { iconMap } from '@/components/sidebar/app-sidebar';
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogDescription,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
} from "@/components/ui/dialog";
|
|
import { Button } from "@/components/ui/button";
|
|
import { Trash2 } from "lucide-react";
|
|
import { apiClient } from '@/lib/api'; // Import the API client
|
|
|
|
interface Chat {
|
|
created_at: string;
|
|
id: number;
|
|
type: string;
|
|
title: string;
|
|
messages: string[];
|
|
search_space_id: number;
|
|
}
|
|
|
|
interface SearchSpace {
|
|
created_at: string;
|
|
id: number;
|
|
name: string;
|
|
description: string;
|
|
user_id: string;
|
|
}
|
|
|
|
interface User {
|
|
id: string;
|
|
email: string;
|
|
is_active: boolean;
|
|
is_superuser: boolean;
|
|
is_verified: boolean;
|
|
}
|
|
|
|
interface AppSidebarProviderProps {
|
|
searchSpaceId: string;
|
|
navSecondary: {
|
|
title: string;
|
|
url: string;
|
|
icon: string;
|
|
}[];
|
|
navMain: {
|
|
title: string;
|
|
url: string;
|
|
icon: string;
|
|
isActive?: boolean;
|
|
items?: {
|
|
title: string;
|
|
url: string;
|
|
}[];
|
|
}[];
|
|
}
|
|
|
|
export function AppSidebarProvider({
|
|
searchSpaceId,
|
|
navSecondary,
|
|
navMain
|
|
}: AppSidebarProviderProps) {
|
|
const [recentChats, setRecentChats] = useState<{ name: string; url: string; icon: string; id: number; search_space_id: number; actions: { name: string; icon: string; onClick: () => void }[] }[]>([]);
|
|
const [searchSpace, setSearchSpace] = useState<SearchSpace | null>(null);
|
|
const [user, setUser] = useState<User | null>(null);
|
|
const [isLoadingChats, setIsLoadingChats] = useState(true);
|
|
const [isLoadingSearchSpace, setIsLoadingSearchSpace] = useState(true);
|
|
const [isLoadingUser, setIsLoadingUser] = useState(true);
|
|
const [chatError, setChatError] = useState<string | null>(null);
|
|
const [searchSpaceError, setSearchSpaceError] = useState<string | null>(null);
|
|
const [userError, setUserError] = useState<string | null>(null);
|
|
const [showDeleteDialog, setShowDeleteDialog] = useState(false);
|
|
const [chatToDelete, setChatToDelete] = useState<{ id: number, name: string } | null>(null);
|
|
const [isDeleting, setIsDeleting] = useState(false);
|
|
const [isClient, setIsClient] = useState(false);
|
|
|
|
// Set isClient to true when component mounts on the client
|
|
useEffect(() => {
|
|
setIsClient(true);
|
|
}, []);
|
|
|
|
// Fetch user details
|
|
useEffect(() => {
|
|
const fetchUser = async () => {
|
|
try {
|
|
// Only run on client-side
|
|
if (typeof window === 'undefined') return;
|
|
|
|
try {
|
|
// Use the API client instead of direct fetch
|
|
const userData = await apiClient.get<User>('users/me');
|
|
setUser(userData);
|
|
setUserError(null);
|
|
} catch (error) {
|
|
console.error('Error fetching user:', error);
|
|
setUserError(error instanceof Error ? error.message : 'Unknown error occurred');
|
|
} finally {
|
|
setIsLoadingUser(false);
|
|
}
|
|
} catch (error) {
|
|
console.error('Error in fetchUser:', error);
|
|
setIsLoadingUser(false);
|
|
}
|
|
};
|
|
|
|
fetchUser();
|
|
}, []);
|
|
|
|
// Fetch recent chats
|
|
useEffect(() => {
|
|
const fetchRecentChats = async () => {
|
|
try {
|
|
// Only run on client-side
|
|
if (typeof window === 'undefined') return;
|
|
|
|
try {
|
|
// Use the API client instead of direct fetch - filter by current search space ID
|
|
const chats: Chat[] = await apiClient.get<Chat[]>(`api/v1/chats/?limit=5&skip=0&search_space_id=${searchSpaceId}`);
|
|
|
|
// Transform API response to the format expected by AppSidebar
|
|
const formattedChats = chats.map(chat => ({
|
|
name: chat.title || `Chat ${chat.id}`, // Fallback if title is empty
|
|
url: `/dashboard/${chat.search_space_id}/researcher/${chat.id}`,
|
|
icon: 'MessageCircleMore',
|
|
id: chat.id,
|
|
search_space_id: chat.search_space_id,
|
|
actions: [
|
|
{
|
|
name: 'View Details',
|
|
icon: 'ExternalLink',
|
|
onClick: () => {
|
|
window.location.href = `/dashboard/${chat.search_space_id}/researcher/${chat.id}`;
|
|
}
|
|
},
|
|
{
|
|
name: 'Delete',
|
|
icon: 'Trash2',
|
|
onClick: () => {
|
|
setChatToDelete({ id: chat.id, name: chat.title || `Chat ${chat.id}` });
|
|
setShowDeleteDialog(true);
|
|
}
|
|
}
|
|
]
|
|
}));
|
|
|
|
setRecentChats(formattedChats);
|
|
setChatError(null);
|
|
} catch (error) {
|
|
console.error('Error fetching chats:', error);
|
|
setChatError(error instanceof Error ? error.message : 'Unknown error occurred');
|
|
// Provide empty array to ensure UI still renders
|
|
setRecentChats([]);
|
|
} finally {
|
|
setIsLoadingChats(false);
|
|
}
|
|
} catch (error) {
|
|
console.error('Error in fetchRecentChats:', error);
|
|
setIsLoadingChats(false);
|
|
}
|
|
};
|
|
|
|
fetchRecentChats();
|
|
|
|
// Set up a refresh interval (every 5 minutes)
|
|
const intervalId = setInterval(fetchRecentChats, 5 * 60 * 1000);
|
|
|
|
// Clean up interval on component unmount
|
|
return () => clearInterval(intervalId);
|
|
}, [searchSpaceId]);
|
|
|
|
// Handle delete chat
|
|
const handleDeleteChat = async () => {
|
|
if (!chatToDelete) return;
|
|
|
|
try {
|
|
setIsDeleting(true);
|
|
|
|
// Use the API client instead of direct fetch
|
|
await apiClient.delete(`api/v1/chats/${chatToDelete.id}`);
|
|
|
|
// Close dialog and refresh chats
|
|
setRecentChats(recentChats.filter(chat => chat.id !== chatToDelete.id));
|
|
|
|
} catch (error) {
|
|
console.error('Error deleting chat:', error);
|
|
} finally {
|
|
setIsDeleting(false);
|
|
setShowDeleteDialog(false);
|
|
setChatToDelete(null);
|
|
}
|
|
};
|
|
|
|
// Fetch search space details
|
|
useEffect(() => {
|
|
const fetchSearchSpace = async () => {
|
|
try {
|
|
// Only run on client-side
|
|
if (typeof window === 'undefined') return;
|
|
|
|
try {
|
|
// Use the API client instead of direct fetch
|
|
const data: SearchSpace = await apiClient.get<SearchSpace>(`api/v1/searchspaces/${searchSpaceId}`);
|
|
setSearchSpace(data);
|
|
setSearchSpaceError(null);
|
|
} catch (error) {
|
|
console.error('Error fetching search space:', error);
|
|
setSearchSpaceError(error instanceof Error ? error.message : 'Unknown error occurred');
|
|
} finally {
|
|
setIsLoadingSearchSpace(false);
|
|
}
|
|
} catch (error) {
|
|
console.error('Error in fetchSearchSpace:', error);
|
|
setIsLoadingSearchSpace(false);
|
|
}
|
|
};
|
|
|
|
fetchSearchSpace();
|
|
}, [searchSpaceId]);
|
|
|
|
// Create a fallback chat if there's an error or no chats
|
|
const fallbackChats = chatError || (!isLoadingChats && recentChats.length === 0)
|
|
? [{
|
|
name: chatError ? "Error loading chats" : "No recent chats",
|
|
url: "#",
|
|
icon: chatError ? "AlertCircle" : "MessageCircleMore",
|
|
id: 0,
|
|
search_space_id: Number(searchSpaceId),
|
|
actions: []
|
|
}]
|
|
: [];
|
|
|
|
// Use fallback chats if there's an error or no chats
|
|
const displayChats = recentChats.length > 0 ? recentChats : fallbackChats;
|
|
|
|
// Update the first item in navSecondary to show the search space name
|
|
const updatedNavSecondary = [...navSecondary];
|
|
if (updatedNavSecondary.length > 0 && isClient) {
|
|
updatedNavSecondary[0] = {
|
|
...updatedNavSecondary[0],
|
|
title: searchSpace?.name || (isLoadingSearchSpace ? 'Loading...' : searchSpaceError ? 'Error loading search space' : 'Unknown Search Space'),
|
|
};
|
|
}
|
|
|
|
// Create user object for AppSidebar
|
|
const customUser = {
|
|
name: isClient && user?.email ? user.email.split('@')[0] : 'User',
|
|
email: isClient ? (user?.email || (isLoadingUser ? 'Loading...' : userError ? 'Error loading user' : 'Unknown User')) : 'Loading...',
|
|
avatar: '/icon-128.png', // Default avatar
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<AppSidebar
|
|
user={customUser}
|
|
navSecondary={updatedNavSecondary}
|
|
navMain={navMain}
|
|
RecentChats={isClient ? displayChats : []}
|
|
/>
|
|
|
|
{/* Delete Confirmation Dialog - Only render on client */}
|
|
{isClient && (
|
|
<Dialog open={showDeleteDialog} onOpenChange={setShowDeleteDialog}>
|
|
<DialogContent className="sm:max-w-md">
|
|
<DialogHeader>
|
|
<DialogTitle className="flex items-center gap-2">
|
|
<Trash2 className="h-5 w-5 text-destructive" />
|
|
<span>Delete Chat</span>
|
|
</DialogTitle>
|
|
<DialogDescription>
|
|
Are you sure you want to delete <span className="font-medium">{chatToDelete?.name}</span>? This action cannot be undone.
|
|
</DialogDescription>
|
|
</DialogHeader>
|
|
<DialogFooter className="flex gap-2 sm:justify-end">
|
|
<Button
|
|
variant="outline"
|
|
onClick={() => setShowDeleteDialog(false)}
|
|
disabled={isDeleting}
|
|
>
|
|
Cancel
|
|
</Button>
|
|
<Button
|
|
variant="destructive"
|
|
onClick={handleDeleteChat}
|
|
disabled={isDeleting}
|
|
className="gap-2"
|
|
>
|
|
{isDeleting ? (
|
|
<>
|
|
<span className="h-4 w-4 animate-spin rounded-full border-2 border-current border-t-transparent" />
|
|
Deleting...
|
|
</>
|
|
) : (
|
|
<>
|
|
<Trash2 className="h-4 w-4" />
|
|
Delete
|
|
</>
|
|
)}
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
)}
|
|
</>
|
|
);
|
|
}
|