veza/apps/web/src/features/dashboard/pages/DashboardPage.tsx

212 lines
7.2 KiB
TypeScript

import { useEffect } from 'react';
import { useAuthStore } from '@/features/auth/store/authStore';
import { useLibraryItems, useLibraryActions, useLibraryStatus } from '@/utils/storeSelectors';
import {
Card,
CardContent,
CardDescription,
CardHeader,
CardTitle,
} from '@/components/ui/card';
import { Button } from '@/components/ui/button';
import { Music, MessageSquare, Library, Users, Heart } from 'lucide-react';
function DashboardPage() {
const { user } = useAuthStore();
// FE-STATE-009: Use selector that returns denormalized array
const items = useLibraryItems();
const { fetchItems } = useLibraryActions();
const { isLoading } = useLibraryStatus();
useEffect(() => {
fetchItems({ limit: 5 });
}, [fetchItems]);
const stats = [
{
title: 'Pistes écoutées',
value: '1,234',
change: '+12%',
icon: Music,
color: 'text-blue-600',
},
{
title: 'Messages envoyés',
value: '567',
change: '+8%',
icon: MessageSquare,
color: 'text-green-600',
},
{
title: 'Favoris',
value: '89',
change: '+23%',
icon: Heart,
color: 'text-red-600',
},
{
title: 'Amis actifs',
value: '45',
change: '+5%',
icon: Users,
color: 'text-purple-600',
},
];
return (
<div className="space-y-6">
{/* En-tête */}
<div>
<h1 className="text-3xl font-bold tracking-tight">
Bonjour, {user?.first_name || user?.username} !
</h1>
<p className="text-muted-foreground">
Voici un aperçu de votre activité sur Veza
</p>
</div>
{/* Statistiques */}
<div className="grid gap-4 md:grid-cols-2 lg:grid-cols-4">
{stats.map((stat) => (
<Card key={stat.title}>
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium">
{stat.title}
</CardTitle>
<stat.icon className={`h-4 w-4 ${stat.color}`} />
</CardHeader>
<CardContent>
<div className="text-2xl font-bold">{stat.value}</div>
<p className="text-xs text-muted-foreground">
<span className="text-green-600">{stat.change}</span> par
rapport au mois dernier
</p>
</CardContent>
</Card>
))}
</div>
<div className="grid gap-6 md:grid-cols-2 lg:grid-cols-3">
{/* Activité récente */}
<Card className="md:col-span-2">
<CardHeader>
<CardTitle>Activité récente</CardTitle>
<CardDescription>
Vos dernières interactions sur la plateforme
</CardDescription>
</CardHeader>
<CardContent>
<div className="space-y-4">
<div className="flex items-center space-x-4">
<div className="w-2 h-2 bg-blue-600 rounded-full"></div>
<div className="flex-1 space-y-1">
<p className="text-sm font-medium">Nouvelle piste ajoutée</p>
<p className="text-xs text-muted-foreground">
Il y a 2 heures
</p>
</div>
</div>
<div className="flex items-center space-x-4">
<div className="w-2 h-2 bg-green-600 rounded-full"></div>
<div className="flex-1 space-y-1">
<p className="text-sm font-medium">Message reçu de @alice</p>
<p className="text-xs text-muted-foreground">
Il y a 4 heures
</p>
</div>
</div>
<div className="flex items-center space-x-4">
<div className="w-2 h-2 bg-purple-600 rounded-full"></div>
<div className="flex-1 space-y-1">
<p className="text-sm font-medium">Nouveau favori ajouté</p>
<p className="text-xs text-muted-foreground">
Il y a 6 heures
</p>
</div>
</div>
</div>
</CardContent>
</Card>
{/* Pistes récentes */}
<Card>
<CardHeader>
<CardTitle>Pistes récentes</CardTitle>
<CardDescription>
Dernières pistes de votre bibliothèque
</CardDescription>
</CardHeader>
<CardContent>
{isLoading ? (
<div className="space-y-2">
{[...Array(3)].map((_, i) => (
<div key={i} className="flex items-center space-x-3">
<div className="w-10 h-10 bg-muted rounded"></div>
<div className="flex-1 space-y-1">
<div className="h-4 bg-muted rounded w-3/4"></div>
<div className="h-3 bg-muted rounded w-1/2"></div>
</div>
</div>
))}
</div>
) : (
<div className="space-y-3">
{items.slice(0, 3).map((item) => (
<div key={item.id} className="flex items-center space-x-3">
<div className="w-10 h-10 bg-muted rounded flex items-center justify-center">
<Music className="h-4 w-4 text-muted-foreground" />
</div>
<div className="flex-1 min-w-0">
<p className="text-sm font-medium truncate">
{item.title}
</p>
<p className="text-xs text-muted-foreground truncate">
{item.description || 'Aucune description'}
</p>
</div>
</div>
))}
{items.length === 0 && (
<p className="text-sm text-muted-foreground text-center py-4">
Aucune piste dans votre bibliothèque
</p>
)}
</div>
)}
</CardContent>
</Card>
</div>
{/* Actions rapides */}
<Card>
<CardHeader>
<CardTitle>Actions rapides</CardTitle>
<CardDescription>
Accédez rapidement aux fonctionnalités principales
</CardDescription>
</CardHeader>
<CardContent>
<div className="grid gap-4 md:grid-cols-2 lg:grid-cols-4">
<Button variant="outline" className="h-20 flex-col space-y-2">
<Music className="h-6 w-6" />
<span>Nouvelle piste</span>
</Button>
<Button variant="outline" className="h-20 flex-col space-y-2">
<MessageSquare className="h-6 w-6" />
<span>Nouveau chat</span>
</Button>
<Button variant="outline" className="h-20 flex-col space-y-2">
<Library className="h-6 w-6" />
<span>Bibliothèque</span>
</Button>
<Button variant="outline" className="h-20 flex-col space-y-2">
<Users className="h-6 w-6" />
<span>Inviter des amis</span>
</Button>
</div>
</CardContent>
</Card>
</div>
);
}
export default DashboardPage;