283 lines
8.4 KiB
TypeScript
283 lines
8.4 KiB
TypeScript
'use client';
|
|
|
|
import {
|
|
Plus,
|
|
RefreshCw,
|
|
Download,
|
|
Upload,
|
|
LayoutDashboard,
|
|
Layers,
|
|
List,
|
|
} from 'lucide-react';
|
|
import { useState, useEffect, useRef, useMemo } from 'react';
|
|
import { Responsive, WidthProvider } from 'react-grid-layout';
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardDescription,
|
|
CardFooter,
|
|
CardHeader,
|
|
CardTitle,
|
|
} from '@/components/ui/card';
|
|
import WidgetConfigModal from '@/components/dashboard/WidgetConfigModal';
|
|
import WidgetDisplay from '@/components/dashboard/WidgetDisplay';
|
|
import { useDashboard } from '@/lib/hooks/useDashboard';
|
|
import { Widget, WidgetConfig } from '@/lib/types/widget';
|
|
import { DASHBOARD_CONSTRAINTS } from '@/lib/constants/dashboard';
|
|
import { toast } from 'sonner';
|
|
|
|
const ResponsiveGridLayout = WidthProvider(Responsive);
|
|
|
|
const DashboardPage = () => {
|
|
const {
|
|
widgets,
|
|
isLoading,
|
|
addWidget,
|
|
updateWidget,
|
|
deleteWidget,
|
|
refreshWidget,
|
|
refreshAllWidgets,
|
|
exportDashboard,
|
|
importDashboard,
|
|
settings,
|
|
updateSettings,
|
|
getLayouts,
|
|
updateLayouts,
|
|
} = useDashboard();
|
|
|
|
const [showAddModal, setShowAddModal] = useState(false);
|
|
const [editingWidget, setEditingWidget] = useState<Widget | null>(null);
|
|
const hasAutoRefreshed = useRef(false);
|
|
|
|
// Memoize the ResponsiveGridLayout to prevent re-renders
|
|
const ResponsiveGrid = useMemo(() => ResponsiveGridLayout, []);
|
|
|
|
// Auto-refresh stale widgets when dashboard loads (only once)
|
|
useEffect(() => {
|
|
if (!isLoading && widgets.length > 0 && !hasAutoRefreshed.current) {
|
|
hasAutoRefreshed.current = true;
|
|
refreshAllWidgets();
|
|
}
|
|
}, [isLoading, widgets, refreshAllWidgets]);
|
|
|
|
const handleAddWidget = () => {
|
|
setEditingWidget(null);
|
|
setShowAddModal(true);
|
|
};
|
|
|
|
const handleEditWidget = (widget: Widget) => {
|
|
setEditingWidget(widget);
|
|
setShowAddModal(true);
|
|
};
|
|
|
|
const handleSaveWidget = (widgetConfig: WidgetConfig) => {
|
|
if (editingWidget) {
|
|
// Update existing widget
|
|
updateWidget(editingWidget.id, widgetConfig);
|
|
} else {
|
|
// Add new widget
|
|
addWidget(widgetConfig);
|
|
}
|
|
setShowAddModal(false);
|
|
setEditingWidget(null);
|
|
};
|
|
|
|
const handleCloseModal = () => {
|
|
setShowAddModal(false);
|
|
setEditingWidget(null);
|
|
};
|
|
|
|
const handleDeleteWidget = (widgetId: string) => {
|
|
deleteWidget(widgetId);
|
|
};
|
|
|
|
const handleRefreshWidget = (widgetId: string) => {
|
|
refreshWidget(widgetId, true); // Force refresh when manually triggered
|
|
};
|
|
|
|
const handleRefreshAll = () => {
|
|
refreshAllWidgets(true);
|
|
};
|
|
|
|
const handleExport = async () => {
|
|
try {
|
|
const configJson = await exportDashboard();
|
|
await navigator.clipboard.writeText(configJson);
|
|
toast.success('Dashboard configuration copied to clipboard');
|
|
console.log('Dashboard configuration copied to clipboard');
|
|
} catch (error) {
|
|
console.error('Export failed:', error);
|
|
toast.error('Failed to copy dashboard configuration');
|
|
}
|
|
};
|
|
|
|
const handleImport = async () => {
|
|
try {
|
|
const configJson = await navigator.clipboard.readText();
|
|
await importDashboard(configJson);
|
|
toast.success('Dashboard configuration imported successfully');
|
|
console.log('Dashboard configuration imported successfully');
|
|
} catch (error) {
|
|
console.error('Import failed:', error);
|
|
toast.error('Failed to import dashboard configuration');
|
|
}
|
|
};
|
|
|
|
const handleToggleProcessingMode = () => {
|
|
updateSettings({ parallelLoading: !settings.parallelLoading });
|
|
};
|
|
|
|
// Handle layout changes from react-grid-layout
|
|
const handleLayoutChange = (layout: any, layouts: any) => {
|
|
updateLayouts(layouts);
|
|
};
|
|
|
|
// Memoize grid children to prevent unnecessary re-renders
|
|
const gridChildren = useMemo(() => {
|
|
return widgets.map((widget) => (
|
|
<div key={widget.id}>
|
|
<WidgetDisplay
|
|
widget={widget}
|
|
onEdit={handleEditWidget}
|
|
onDelete={handleDeleteWidget}
|
|
onRefresh={handleRefreshWidget}
|
|
/>
|
|
</div>
|
|
));
|
|
}, [widgets]);
|
|
|
|
// Empty state component
|
|
const EmptyDashboard = () => (
|
|
<div className="col-span-2 flex justify-center items-center min-h-[400px]">
|
|
<Card className="w-96 text-center">
|
|
<CardHeader>
|
|
<CardTitle>Welcome to your Dashboard</CardTitle>
|
|
<CardDescription>
|
|
Create your first widget to get started with personalized
|
|
information
|
|
</CardDescription>
|
|
</CardHeader>
|
|
|
|
<CardContent>
|
|
<p className="text-sm text-fg/60 mb-4">
|
|
Widgets let you fetch content from any URL and process it with AI to
|
|
show exactly what you need.
|
|
</p>
|
|
</CardContent>
|
|
|
|
<CardFooter className="justify-center">
|
|
<button
|
|
onClick={handleAddWidget}
|
|
className="px-4 py-2 bg-accent text-white rounded hover:bg-accent-700 transition duration-200 flex items-center space-x-2"
|
|
>
|
|
<Plus size={16} />
|
|
<span>Create Your First Widget</span>
|
|
</button>
|
|
</CardFooter>
|
|
</Card>
|
|
</div>
|
|
);
|
|
|
|
return (
|
|
<div className="flex flex-col min-h-screen">
|
|
{/* Header matching other pages */}
|
|
<div className="flex flex-col pt-4">
|
|
<div className="flex items-center justify-between">
|
|
<div className="flex items-center">
|
|
<LayoutDashboard />
|
|
<h1 className="text-3xl font-medium p-2">Dashboard</h1>
|
|
</div>
|
|
|
|
<div className="flex items-center space-x-2">
|
|
<button
|
|
onClick={handleRefreshAll}
|
|
className="p-2 hover:bg-surface-2 rounded-lg transition duration-200"
|
|
title="Refresh All Widgets"
|
|
>
|
|
<RefreshCw size={18} />
|
|
</button>
|
|
|
|
<button
|
|
onClick={handleToggleProcessingMode}
|
|
className="p-2 hover:bg-surface-2 rounded-lg transition duration-200"
|
|
title={`Switch to ${settings.parallelLoading ? 'Sequential' : 'Parallel'} Processing`}
|
|
>
|
|
{settings.parallelLoading ? (
|
|
<Layers size={18} />
|
|
) : (
|
|
<List size={18} />
|
|
)}
|
|
</button>
|
|
|
|
<button
|
|
onClick={handleExport}
|
|
className="p-2 hover:bg-surface-2 rounded-lg transition duration-200"
|
|
title="Export Dashboard Configuration"
|
|
>
|
|
<Download size={18} />
|
|
</button>
|
|
|
|
<button
|
|
onClick={handleImport}
|
|
className="p-2 hover:bg-surface-2 rounded-lg transition duration-200"
|
|
title="Import Dashboard Configuration"
|
|
>
|
|
<Upload size={18} />
|
|
</button>
|
|
|
|
<button
|
|
onClick={handleAddWidget}
|
|
className="p-2 bg-accent hover:bg-accent-700 rounded-lg transition duration-200"
|
|
title="Add New Widget"
|
|
>
|
|
<Plus size={18} />
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<hr className="border-t my-4 w-full" />
|
|
</div>
|
|
|
|
{/* Main content area */}
|
|
<div className="flex-1 pb-20 lg:pb-2">
|
|
{isLoading ? (
|
|
<div className="flex items-center justify-center py-12">
|
|
<div className="text-center">
|
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 mx-auto mb-4"></div>
|
|
<p className="text-fg/60">Loading dashboard...</p>
|
|
</div>
|
|
</div>
|
|
) : widgets.length === 0 ? (
|
|
<EmptyDashboard />
|
|
) : (
|
|
<ResponsiveGrid
|
|
className="layout"
|
|
layouts={getLayouts()}
|
|
breakpoints={DASHBOARD_CONSTRAINTS.GRID_BREAKPOINTS}
|
|
cols={DASHBOARD_CONSTRAINTS.GRID_COLUMNS}
|
|
rowHeight={DASHBOARD_CONSTRAINTS.GRID_ROW_HEIGHT}
|
|
margin={DASHBOARD_CONSTRAINTS.GRID_MARGIN}
|
|
containerPadding={DASHBOARD_CONSTRAINTS.GRID_CONTAINER_PADDING}
|
|
onLayoutChange={handleLayoutChange}
|
|
isDraggable={true}
|
|
isResizable={true}
|
|
compactType="vertical"
|
|
preventCollision={false}
|
|
draggableHandle=".widget-drag-handle"
|
|
>
|
|
{gridChildren}
|
|
</ResponsiveGrid>
|
|
)}
|
|
</div>
|
|
|
|
{/* Widget Configuration Modal */}
|
|
<WidgetConfigModal
|
|
isOpen={showAddModal}
|
|
onClose={handleCloseModal}
|
|
onSave={handleSaveWidget}
|
|
editingWidget={editingWidget}
|
|
/>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default DashboardPage;
|