Files
Anonyme/app/components/AnonymizationInterface.tsx
2025-07-28 20:55:11 +02:00

215 lines
6.6 KiB
TypeScript

import { CheckCircle } from "lucide-react";
interface AnonymizationInterfaceProps {
isProcessing: boolean;
outputText?: string;
sourceText?: string;
}
export const AnonymizationInterface = ({
isProcessing,
outputText,
sourceText,
}: AnonymizationInterfaceProps) => {
// Fonction pour détecter quels types de données ont été anonymisés
const getAnonymizedDataTypes = () => {
if (!outputText || !sourceText) return new Set();
const anonymizedTypes = new Set<string>();
// ✅ NOUVEAUX PATTERNS PRESIDIO
// Noms (PERSON)
if (outputText.includes("<PERSON>")) {
anonymizedTypes.add("Prénoms");
anonymizedTypes.add("Noms de famille");
anonymizedTypes.add("Noms complets");
}
// Emails (EMAIL_ADDRESS)
if (outputText.includes("<EMAIL_ADDRESS>")) {
anonymizedTypes.add("Adresses e-mail");
}
// Téléphones (PHONE_NUMBER)
if (outputText.includes("<PHONE_NUMBER>")) {
anonymizedTypes.add("Numéros de téléphone");
}
// Adresses (LOCATION)
if (outputText.includes("<LOCATION>")) {
anonymizedTypes.add("Adresses");
}
// IBAN (IBAN)
if (outputText.includes("<IBAN>")) {
anonymizedTypes.add("Numéros d'ID"); // Ou créer une nouvelle catégorie "IBAN"
}
// Organisations (ORGANIZATION)
if (outputText.includes("<ORGANIZATION>")) {
anonymizedTypes.add("Noms de domaine"); // Ou adapter selon vos besoins
}
// Dates personnalisées (CUSTOM_DATE)
if (outputText.includes("<CUSTOM_DATE>")) {
anonymizedTypes.add("Dates");
}
// Numéros d'entreprise belges (BE_ENTERPRISE_NUMBER)
if (outputText.includes("<BE_ENTERPRISE_NUMBER>")) {
anonymizedTypes.add("Numéros d'ID");
}
// ✅ ANCIENS PATTERNS (pour compatibilité)
// Noms (anciens patterns [Nom1], [Nom2]...)
if (outputText.includes("[Nom1]") || outputText.includes("[Nom")) {
anonymizedTypes.add("Prénoms");
anonymizedTypes.add("Noms de famille");
anonymizedTypes.add("Noms complets");
}
// Emails (anciens patterns)
if (outputText.includes("[Email1]") || outputText.includes("[Email")) {
anonymizedTypes.add("Adresses e-mail");
}
// Téléphones (anciens patterns)
if (outputText.includes("[Téléphone1]") || outputText.includes("[Téléphone")) {
anonymizedTypes.add("Numéros de téléphone");
}
// Adresses (anciens patterns)
if (outputText.includes("[Adresse1]") || outputText.includes("[Adresse")) {
anonymizedTypes.add("Adresses");
}
// Numéros d'ID / Sécurité sociale (anciens patterns)
if (
outputText.includes("[NuméroSS1]") ||
outputText.includes("[NuméroSS") ||
outputText.includes("[ID")
) {
anonymizedTypes.add("Numéros d'ID");
}
// Valeurs monétaires
if (outputText.includes("[Montant") || /\[\d+[€$]\]/.test(outputText)) {
anonymizedTypes.add("Valeurs monétaires");
}
// Noms de domaine
if (outputText.includes("[Domaine") || /\[.*\.com\]/.test(outputText)) {
anonymizedTypes.add("Noms de domaine");
}
// Valeurs numériques
if (
/\[\d+\]/.test(outputText) &&
!outputText.includes("[Téléphone") &&
!outputText.includes("[Montant")
) {
anonymizedTypes.add("Valeurs numériques");
}
// Texte personnalisé (si du texte a été modifié mais pas avec les patterns spécifiques)
if (sourceText !== outputText && anonymizedTypes.size === 0) {
anonymizedTypes.add("Texte personnalisé");
}
return anonymizedTypes;
};
// Structure exacte de SupportedDataTypes (récupérée dynamiquement)
const supportedDataStructure = [
{
items: ["Prénoms", "Numéros de téléphone", "Noms de domaine"],
},
{
items: ["Noms de famille", "Adresses", "Dates"],
},
{
items: ["Noms complets", "Numéros d'ID", "Valeurs numériques"],
},
{
items: ["Adresses e-mail", "Valeurs monétaires", "Texte personnalisé"],
},
];
if (isProcessing) {
return (
<div className="bg-gray-50 border border-gray-200 rounded-xl p-6">
<div className="flex items-center justify-center space-x-3 mb-4">
<div className="animate-spin rounded-full h-6 w-6 border-b-2 border-gray-500"></div>
<h4 className="text-sm font-semibold text-gray-700">
Anonymisation en cours...
</h4>
</div>
<div className="space-y-3">
<div className="flex items-center space-x-2">
<div className="w-2 h-2 bg-gray-500 rounded-full animate-pulse"></div>
<span className="text-xs text-gray-600">Analyse du contenu</span>
</div>
<div className="flex items-center space-x-2">
<div
className="w-2 h-2 bg-gray-500 rounded-full animate-pulse"
style={{ animationDelay: "0.5s" }}
></div>
<span className="text-xs text-gray-600">
Détection des données sensibles
</span>
</div>
<div className="flex items-center space-x-2">
<div
className="w-2 h-2 bg-gray-500 rounded-full animate-pulse"
style={{ animationDelay: "1s" }}
></div>
<span className="text-xs text-gray-600">
Application de l&apos;anonymisation
</span>
</div>
</div>
</div>
);
}
if (outputText) {
const anonymizedTypes = getAnonymizedDataTypes();
return (
<div className="bg-green-50 border border-green-200 rounded-xl p-6">
<div className="flex items-center space-x-3 mb-4">
<CheckCircle className="h-5 w-5 text-green-600" />
<h4 className="text-sm font-semibold text-green-700">
Anonymisation terminée avec succès
</h4>
</div>
<div className="grid grid-cols-2 md:grid-cols-4 gap-4 text-xs">
{supportedDataStructure.map((column, columnIndex) => (
<div key={columnIndex} className="flex flex-col space-y-2">
{column.items.map((item, itemIndex) => {
const isAnonymized = anonymizedTypes.has(item);
return (
<span
key={itemIndex}
className={
isAnonymized
? "text-green-700 font-medium"
: "text-gray-400"
}
>
{isAnonymized ? "✓" : "•"} {item}
</span>
);
})}
</div>
))}
</div>
</div>
);
}
return null;
};