Files
itinova-podcasts/client/src/pages/AdminDashboard.tsx
manus-admin aab11c8308 Initial commit: itinova-podcasts v1
Stack: Node.js/Express + React/Vite + tRPC + MySQL (Drizzle ORM)
Features: Gestion de podcasts, établissements, mots-clés, upload audio S3
Migrations: 0000-0002 (users, etablissements, mots_cles, podcasts, podcast_mots_cles)
2026-04-12 18:34:56 -04:00

192 lines
8.1 KiB
TypeScript

import AdminLayout from "@/components/AdminLayout";
import { trpc } from "@/lib/trpc";
import { Building2, Mic2, Radio, FileText, TrendingUp, ArrowRight } from "lucide-react";
import { Link } from "wouter";
import { Button } from "@/components/ui/button";
const ARTWORK_COLORS = [
"from-blue-500 to-indigo-600",
"from-violet-500 to-purple-700",
"from-emerald-500 to-teal-600",
"from-orange-500 to-red-600",
"from-cyan-500 to-blue-600",
"from-rose-500 to-pink-600",
"from-amber-500 to-orange-600",
"from-indigo-500 to-blue-700",
];
export default function AdminDashboard() {
const { data: stats, isLoading } = trpc.podcasts.stats.useQuery();
const { data: recentPodcasts } = trpc.podcasts.list.useQuery({ statut: undefined });
const statCards = [
{
title: "Total podcasts",
value: stats?.total ?? 0,
icon: Mic2,
gradient: "from-blue-500 to-indigo-600",
textColor: "text-blue-600",
bgLight: "bg-blue-50",
change: null,
},
{
title: "Publiés",
value: stats?.publies ?? 0,
icon: Radio,
gradient: "from-emerald-500 to-teal-600",
textColor: "text-emerald-600",
bgLight: "bg-emerald-50",
change: null,
},
{
title: "Brouillons",
value: stats?.brouillons ?? 0,
icon: FileText,
gradient: "from-amber-500 to-orange-500",
textColor: "text-amber-600",
bgLight: "bg-amber-50",
change: null,
},
{
title: "Établissements",
value: stats?.etablissements ?? 0,
icon: Building2,
gradient: "from-violet-500 to-purple-600",
textColor: "text-violet-600",
bgLight: "bg-violet-50",
change: null,
},
];
return (
<AdminLayout>
<div className="space-y-7 max-w-6xl">
{/* ── Header ── */}
<div className="flex items-start justify-between">
<div>
<h1 className="text-2xl font-bold text-foreground tracking-tight">Tableau de bord</h1>
<p className="text-muted-foreground text-sm mt-1">
Vue d'ensemble de la plateforme Itinova Podcasts
</p>
</div>
<div className="hidden sm:flex items-center gap-2 px-3 py-1.5 bg-emerald-50 border border-emerald-200 rounded-full">
<span className="w-2 h-2 rounded-full bg-emerald-500 animate-pulse" />
<span className="text-xs font-medium text-emerald-700">Plateforme active</span>
</div>
</div>
{/* ── Stat cards ── */}
<div className="grid grid-cols-2 lg:grid-cols-4 gap-4">
{statCards.map((card) => (
<div
key={card.title}
className="relative bg-card rounded-2xl p-5 border border-border shadow-sm hover:shadow-md transition-all duration-200 overflow-hidden group"
>
{/* Gradient accent top */}
<div className={`absolute top-0 left-0 right-0 h-1 bg-gradient-to-r ${card.gradient}`} />
<div className="flex items-start justify-between">
<div>
<p className="text-xs font-medium text-muted-foreground uppercase tracking-wider">{card.title}</p>
<p className="text-3xl font-bold text-foreground mt-2 tabular-nums">
{isLoading ? (
<span className="inline-block w-10 h-8 bg-muted rounded-lg animate-pulse" />
) : card.value}
</p>
</div>
<div className={`w-11 h-11 rounded-xl flex items-center justify-center bg-gradient-to-br ${card.gradient} shadow-md group-hover:scale-110 transition-transform`}>
<card.icon className="w-5 h-5 text-white" />
</div>
</div>
</div>
))}
</div>
{/* ── Taux de publication ── */}
{stats && stats.total > 0 && (
<div className="bg-card rounded-2xl p-5 border border-border shadow-sm">
<div className="flex items-center justify-between mb-3">
<div className="flex items-center gap-2">
<TrendingUp className="w-4 h-4 text-primary" />
<span className="text-sm font-semibold text-foreground">Taux de publication</span>
</div>
<span className="text-sm font-bold text-primary">
{Math.round((stats.publies / stats.total) * 100)}%
</span>
</div>
<div className="h-2.5 bg-muted rounded-full overflow-hidden">
<div
className="h-full bg-gradient-to-r from-blue-500 to-indigo-600 rounded-full transition-all duration-700"
style={{ width: `${Math.round((stats.publies / stats.total) * 100)}%` }}
/>
</div>
<div className="flex justify-between mt-2 text-xs text-muted-foreground">
<span>{stats.publies} publié{stats.publies !== 1 ? "s" : ""}</span>
<span>{stats.brouillons} brouillon{stats.brouillons !== 1 ? "s" : ""}</span>
</div>
</div>
)}
{/* ── Podcasts récents ── */}
<div className="bg-card rounded-2xl border border-border shadow-sm overflow-hidden">
<div className="flex items-center justify-between px-6 py-4 border-b border-border">
<h2 className="font-semibold text-foreground">Podcasts récents</h2>
<Link href="/admin/podcasts">
<a className="flex items-center gap-1 text-sm text-primary hover:text-primary/80 font-medium transition-colors">
Voir tout
<ArrowRight className="w-3.5 h-3.5" />
</a>
</Link>
</div>
{!recentPodcasts || recentPodcasts.length === 0 ? (
<div className="text-center py-16 px-6">
<div className="w-16 h-16 rounded-2xl bg-muted flex items-center justify-center mx-auto mb-4">
<Mic2 className="w-8 h-8 text-muted-foreground/40" />
</div>
<p className="font-semibold text-foreground">Aucun podcast pour l'instant</p>
<p className="text-sm text-muted-foreground mt-1">Commencez par créer votre premier podcast</p>
<Link href="/admin/podcasts">
<Button variant="outline" size="sm" className="mt-4">
Créer le premier podcast
</Button>
</Link>
</div>
) : (
<div className="divide-y divide-border">
{recentPodcasts.slice(0, 6).map((p) => {
const colorClass = ARTWORK_COLORS[p.id % ARTWORK_COLORS.length];
return (
<div
key={p.id}
className="flex items-center gap-4 px-6 py-3.5 hover:bg-muted/30 transition-colors group"
>
{/* Artwork */}
<div className={`w-10 h-10 rounded-xl bg-gradient-to-br ${colorClass} flex items-center justify-center shrink-0 shadow-sm`}>
<Mic2 className="w-4 h-4 text-white" />
</div>
{/* Info */}
<div className="flex-1 min-w-0">
<p className="font-medium text-sm text-foreground truncate leading-tight">{p.titre}</p>
<p className="text-xs text-muted-foreground truncate mt-0.5">{p.etablissementNom}</p>
</div>
{/* Badge statut */}
<span className={p.statut === "publie" ? "badge-publie" : "badge-brouillon"}>
{p.statut === "publie" ? "Publié" : "Brouillon"}
</span>
{/* Action */}
<Link href="/admin/podcasts">
<a className="opacity-0 group-hover:opacity-100 transition-opacity text-xs text-primary hover:underline font-medium flex items-center gap-1">
Éditer <ArrowRight className="w-3 h-3" />
</a>
</Link>
</div>
);
})}
</div>
)}
</div>
</div>
</AdminLayout>
);
}