the-tip-top-frontend/app/admin/utilisateurs/[id]/page.tsx
soufiane dce1559a32 feat: improve user management and profile features
- Replace email verification status with active/inactive status
- Add user archiving (soft delete) instead of hard delete
- Add search by name/email in user management
- Add status filter (active/inactive) in user management
- Simplify actions to show only "Détails" button
- Add Google Maps with clickable marker on contact page
- Update button labels from "Désactiver" to "Supprimer"

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-28 13:59:52 +01:00

245 lines
9.5 KiB
TypeScript

'use client';
import { useState, useEffect, useCallback } from 'react';
import { useRouter, useParams } from 'next/navigation';
import { adminService } from '@/services/admin.service';
import { User } from '@/types';
import { ArrowLeft, Mail, Phone, MapPin, Calendar, Award, Ticket, CheckCircle, Clock } from 'lucide-react';
import toast from 'react-hot-toast';
export default function UserDetailsPage() {
const router = useRouter();
const params = useParams();
const userId = params.id as string;
const [user, setUser] = useState<User | null>(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const loadUserDetails = useCallback(async () => {
try {
setLoading(true);
setError(null);
const userData = await adminService.getUserById(userId);
setUser(userData);
} catch (err: any) {
setError(err.message || 'Erreur lors du chargement des détails');
toast.error('Erreur lors du chargement des détails de l\'utilisateur');
} finally {
setLoading(false);
}
}, [userId]);
useEffect(() => {
loadUserDetails();
}, [loadUserDetails]);
const getRoleBadgeColor = (role: string) => {
switch (role) {
case 'ADMIN':
return 'bg-red-100 text-red-800';
case 'EMPLOYEE':
return 'bg-blue-100 text-blue-800';
case 'CLIENT':
return 'bg-green-100 text-green-800';
default:
return 'bg-gray-100 text-gray-800';
}
};
if (loading) {
return (
<div className="p-8">
<div className="animate-pulse space-y-4">
<div className="h-8 bg-gray-200 rounded w-1/4"></div>
<div className="h-64 bg-gray-200 rounded"></div>
</div>
</div>
);
}
if (error || !user) {
return (
<div className="p-8">
<div className="bg-red-100 border border-red-400 text-red-700 px-4 py-3 rounded">
{error || 'Utilisateur non trouvé'}
</div>
<button
onClick={() => router.back()}
className="mt-4 flex items-center gap-2 text-blue-600 hover:text-blue-800"
>
<ArrowLeft className="w-4 h-4" />
Retour
</button>
</div>
);
}
return (
<div className="p-8 bg-gray-50 min-h-screen">
{/* Header */}
<div className="mb-6">
<button
onClick={() => router.back()}
className="flex items-center gap-2 text-gray-600 hover:text-gray-800 mb-4"
>
<ArrowLeft className="w-4 h-4" />
Retour à la liste
</button>
<div className="flex items-center justify-between">
<div>
<h1 className="text-3xl font-bold text-gray-900">
{user.firstName} {user.lastName}
</h1>
<p className="text-gray-600 mt-1">Détails de l'utilisateur</p>
</div>
<span className={`px-4 py-2 rounded-full text-sm font-semibold ${getRoleBadgeColor(user.role)}`}>
{user.role}
</span>
</div>
</div>
<div className="grid grid-cols-1 lg:grid-cols-3 gap-6">
{/* Informations principales */}
<div className="lg:col-span-2 space-y-6">
{/* Informations de contact */}
<div className="bg-white rounded-lg shadow-sm border border-gray-200 p-6">
<h2 className="text-lg font-semibold text-gray-900 mb-4">Informations de contact</h2>
<div className="space-y-4">
<div className="flex items-start gap-3">
<Mail className="w-5 h-5 text-gray-400 mt-0.5" />
<div>
<p className="text-sm font-medium text-gray-600">Email</p>
<p className="text-base text-gray-900">{user.email}</p>
</div>
</div>
{user.phone && (
<div className="flex items-start gap-3">
<Phone className="w-5 h-5 text-gray-400 mt-0.5" />
<div>
<p className="text-sm font-medium text-gray-600">Téléphone</p>
<p className="text-base text-gray-900">{user.phone}</p>
</div>
</div>
)}
{(user.address || user.city || user.postalCode) && (
<div className="flex items-start gap-3">
<MapPin className="w-5 h-5 text-gray-400 mt-0.5" />
<div>
<p className="text-sm font-medium text-gray-600">Adresse</p>
<p className="text-base text-gray-900">
{user.address && <>{user.address}<br /></>}
{user.postalCode && user.city && `${user.postalCode} ${user.city}`}
</p>
</div>
</div>
)}
</div>
</div>
{/* Informations personnelles */}
{(user.dateOfBirth || user.gender) && (
<div className="bg-white rounded-lg shadow-sm border border-gray-200 p-6">
<h2 className="text-lg font-semibold text-gray-900 mb-4">Informations personnelles</h2>
<div className="space-y-4">
{user.dateOfBirth && (
<div>
<p className="text-sm font-medium text-gray-600">Date de naissance</p>
<p className="text-base text-gray-900">
{new Date(user.dateOfBirth).toLocaleDateString('fr-FR', {
day: 'numeric',
month: 'long',
year: 'numeric'
})}
</p>
</div>
)}
{user.gender && (
<div>
<p className="text-sm font-medium text-gray-600">Genre</p>
<p className="text-base text-gray-900">
{user.gender === 'MALE' ? 'Homme' :
user.gender === 'FEMALE' ? 'Femme' :
user.gender === 'OTHER' ? 'Autre' : 'Non spécifié'}
</p>
</div>
)}
</div>
</div>
)}
{/* Statistiques de tickets */}
{user.role === 'CLIENT' && (
<div className="bg-white rounded-lg shadow-sm border border-gray-200 p-6">
<h2 className="text-lg font-semibold text-gray-900 mb-4">Activité des tickets</h2>
<div className="grid grid-cols-3 gap-4">
<div className="text-center p-4 bg-blue-50 rounded-lg">
<Ticket className="w-8 h-8 text-blue-600 mx-auto mb-2" />
<p className="text-2xl font-bold text-blue-600">{user.ticketsCount || 0}</p>
<p className="text-sm text-gray-600">Total tickets</p>
</div>
<div className="text-center p-4 bg-yellow-50 rounded-lg">
<Clock className="w-8 h-8 text-yellow-600 mx-auto mb-2" />
<p className="text-2xl font-bold text-yellow-600">{user.pendingTickets || 0}</p>
<p className="text-sm text-gray-600">En attente</p>
</div>
<div className="text-center p-4 bg-green-50 rounded-lg">
<CheckCircle className="w-8 h-8 text-green-600 mx-auto mb-2" />
<p className="text-2xl font-bold text-green-600">{user.claimedTickets || 0}</p>
<p className="text-sm text-gray-600">Réclamés</p>
</div>
</div>
</div>
)}
</div>
{/* Sidebar */}
<div className="space-y-6">
{/* Statut du compte */}
<div className="bg-white rounded-lg shadow-sm border border-gray-200 p-6">
<h2 className="text-lg font-semibold text-gray-900 mb-4">Statut du compte</h2>
<div className="space-y-3">
<div>
<p className="text-sm font-medium text-gray-600 mb-1">Statut</p>
<span className={`inline-flex items-center px-3 py-1 rounded-full text-sm font-semibold ${
user.isActive !== false
? 'bg-green-100 text-green-800'
: 'bg-red-100 text-red-800'
}`}>
{user.isActive !== false ? 'Actif' : 'Inactif'}
</span>
</div>
<div>
<p className="text-sm font-medium text-gray-600 mb-1">Membre depuis</p>
<div className="flex items-center gap-2 text-gray-900">
<Calendar className="w-4 h-4 text-gray-400" />
<span className="text-sm">
{new Date(user.createdAt).toLocaleDateString('fr-FR', {
day: 'numeric',
month: 'long',
year: 'numeric'
})}
</span>
</div>
</div>
</div>
</div>
{/* Actions rapides */}
<div className="bg-white rounded-lg shadow-sm border border-gray-200 p-6">
<h2 className="text-lg font-semibold text-gray-900 mb-4">Actions</h2>
<div className="space-y-2">
<button
onClick={() => router.push(`/admin/utilisateurs?edit=${user.id}`)}
className="w-full px-4 py-2 bg-blue-600 text-white rounded-lg hover:bg-blue-700 transition-colors"
>
Modifier l'utilisateur
</button>
</div>
</div>
</div>
</div>
</div>
);
}