Files
SteelCheck/components/ReportDisplay.tsx

214 lines
13 KiB
TypeScript

import React from 'react';
import type { ReportData, ComplianceItem } from '../types';
import { CheckCircleIcon, XCircleIcon, InfoIcon } from './Icons';
interface ReportDisplayProps {
report: ReportData;
}
const statusClass = (status: 'OK' | 'FALHA' | 'CONFORME' | 'NÃO CONFORME') => {
switch (status) {
case 'OK':
case 'CONFORME':
return 'text-emerald-600 bg-emerald-50 dark:bg-emerald-900/20 dark:text-emerald-400 border border-emerald-200 dark:border-emerald-800';
case 'FALHA':
case 'NÃO CONFORME':
return 'text-red-600 bg-red-50 dark:bg-red-900/20 dark:text-red-400 border border-red-200 dark:border-red-800';
default:
return 'text-slate-600 bg-slate-50 dark:bg-slate-800/50 dark:text-slate-400 border border-slate-200 dark:border-slate-700';
}
};
const statusIcon = (status: 'OK' | 'FALHA') => {
return status === 'OK'
? <CheckCircleIcon className="w-4 h-4" />
: <XCircleIcon className="w-4 h-4" />;
};
const ComplianceTableRow: React.FC<{ item: ComplianceItem; headerLabel: string }> = ({ item, headerLabel }) => (
<tr className="block md:table-row mb-4 md:mb-0 bg-white/50 dark:bg-slate-800/50 md:bg-transparent rounded-xl md:rounded-none shadow-sm md:shadow-none border border-slate-100 dark:border-slate-700 md:border-b md:border-slate-200/50 md:dark:border-slate-700/50 last:border-0 hover:bg-white/80 dark:hover:bg-slate-800/80 transition-colors">
<td className="p-4 md:py-4 md:px-6 flex justify-between items-center border-b border-slate-100 dark:border-slate-700 md:border-0 md:table-cell">
<span className="text-xs font-bold uppercase tracking-wider text-slate-500 dark:text-slate-400 md:hidden">{headerLabel}</span>
<span className="font-medium text-slate-700 dark:text-slate-300">{item.property || item.element || item.test}</span>
</td>
<td className="p-4 md:py-4 md:px-6 flex justify-between items-center border-b border-slate-100 dark:border-slate-700 md:border-0 md:table-cell">
<span className="text-xs font-bold uppercase tracking-wider text-slate-500 dark:text-slate-400 md:hidden">Norma</span>
<span className="text-slate-500 dark:text-slate-400 text-sm font-mono">{item.norm}</span>
</td>
<td className="p-4 md:py-4 md:px-6 flex justify-between items-center border-b border-slate-100 dark:border-slate-700 md:border-0 md:table-cell">
<span className="text-xs font-bold uppercase tracking-wider text-slate-500 dark:text-slate-400 md:hidden">Certificado</span>
<span className="text-slate-800 dark:text-slate-200 font-semibold font-mono">{item.certificate}</span>
</td>
<td className="p-4 md:py-4 md:px-6 flex justify-between items-center md:table-cell">
<span className="text-xs font-bold uppercase tracking-wider text-slate-500 dark:text-slate-400 md:hidden">Status</span>
<span className={`inline-flex items-center gap-1.5 px-2.5 py-1 rounded-full text-xs font-bold uppercase tracking-wide ${statusClass(item.status)}`}>
{statusIcon(item.status)}
{item.status}
</span>
</td>
</tr>
);
export const ReportDisplay: React.FC<ReportDisplayProps> = ({ report }) => {
const { identification, compliance, overPerformance, equivalents, confidence } = report;
return (
<div className="space-y-8 animate-fade-in pb-12">
<div className="text-center space-y-2">
<h2 className="text-3xl md:text-4xl font-display font-bold text-slate-900 dark:text-white">Relatório Técnico de Qualidade</h2>
<p className="text-slate-500 dark:text-slate-400">Análise de conformidade normativa assistida por IA</p>
</div>
<div className="bg-white/70 dark:bg-slate-800/60 backdrop-blur-md border border-white/20 dark:border-slate-700/50 shadow-xl p-8 rounded-2xl text-center relative overflow-hidden">
<div className="absolute top-0 left-0 w-full h-1 bg-gradient-to-r from-blue-500 via-indigo-500 to-purple-500"></div>
<h3 className="text-lg font-medium text-slate-500 dark:text-slate-400 mb-2 uppercase tracking-wide">Grau de Confiança da Análise</h3>
<div className="flex justify-center items-end gap-2">
<span className="text-6xl sm:text-7xl font-display font-bold bg-clip-text text-transparent bg-gradient-to-br from-blue-600 to-indigo-600 dark:from-blue-400 dark:to-indigo-400">{confidence}</span>
<span className="text-3xl font-bold text-slate-400 mb-2">%</span>
</div>
</div>
{/* Identification Section */}
<div className="bg-white/70 dark:bg-slate-800/60 backdrop-blur-md border border-white/20 dark:border-slate-700/50 shadow-xl rounded-2xl overflow-hidden">
<div className="px-6 py-4 bg-slate-50/50 dark:bg-slate-800/50 border-b border-slate-200/50 dark:border-slate-700/50 backdrop-blur-sm">
<h3 className="text-xl font-display font-bold text-slate-800 dark:text-white">1. Dados de Identificação</h3>
</div>
<div className="p-6">
<ul className="grid grid-cols-1 md:grid-cols-2 gap-x-8 gap-y-4">
{Object.entries(identification).map(([key, value]) => (
<li key={key} className="flex flex-col border-b border-slate-100 dark:border-slate-800/50 pb-3 last:border-0 last:pb-0">
<span className="text-xs font-bold text-slate-400 uppercase tracking-wider mb-1">{key.replace(/([A-Z])/g, ' $1')}</span>
<span className="text-lg font-medium text-slate-800 dark:text-slate-200">{value}</span>
</li>
))}
</ul>
</div>
</div>
{/* Compliance Section */}
<div className="bg-white/70 dark:bg-slate-800/60 backdrop-blur-md border border-white/20 dark:border-slate-700/50 shadow-xl rounded-2xl overflow-hidden">
<div className="px-6 py-4 bg-slate-50/50 dark:bg-slate-800/50 border-b border-slate-200/50 dark:border-slate-700/50 backdrop-blur-sm flex flex-col md:flex-row md:items-center justify-between gap-4">
<h3 className="text-xl font-display font-bold text-slate-800 dark:text-white">2. Verificação de Conformidade</h3>
<div className={`px-4 py-1.5 rounded-full font-bold text-sm uppercase tracking-wide border shadow-sm ${statusClass(compliance.status)}`}>
{compliance.status}
</div>
</div>
<div className="p-6 space-y-8">
<div>
<h4 className="flex items-center gap-2 font-display font-semibold text-lg mb-4 text-slate-800 dark:text-slate-200">
<span className="w-1.5 h-6 bg-blue-500 rounded-full"></span>
Análise de Propriedades Mecânicas
</h4>
<div className="overflow-x-auto">
<table className="w-full text-left border-collapse">
<thead className="hidden md:table-header-group text-xs uppercase tracking-wider text-slate-500 dark:text-slate-400 border-b-2 border-slate-100 dark:border-slate-700">
<tr>
<th className="py-3 px-6 font-semibold">Propriedade</th>
<th className="py-3 px-6 font-semibold">Norma</th>
<th className="py-3 px-6 font-semibold">Certificado</th>
<th className="py-3 px-6 font-semibold w-32">Status</th>
</tr>
</thead>
<tbody className="md:divide-y md:divide-slate-100 dark:md:divide-slate-700/50">
{compliance.mechanical.map((item, i) => <ComplianceTableRow key={i} item={item} headerLabel="Propriedade" />)}
</tbody>
</table>
</div>
</div>
<div>
<h4 className="flex items-center gap-2 font-display font-semibold text-lg mb-4 text-slate-800 dark:text-slate-200">
<span className="w-1.5 h-6 bg-purple-500 rounded-full"></span>
Análise de Composição Química (%)
</h4>
<div className="overflow-x-auto">
<table className="w-full text-left border-collapse">
<thead className="hidden md:table-header-group text-xs uppercase tracking-wider text-slate-500 dark:text-slate-400 border-b-2 border-slate-100 dark:border-slate-700">
<tr>
<th className="py-3 px-6 font-semibold">Elemento</th>
<th className="py-3 px-6 font-semibold">Norma</th>
<th className="py-3 px-6 font-semibold">Certificado</th>
<th className="py-3 px-6 font-semibold w-32">Status</th>
</tr>
</thead>
<tbody className="md:divide-y md:divide-slate-100 dark:md:divide-slate-700/50">
{compliance.chemical.map((item, i) => <ComplianceTableRow key={i} item={item} headerLabel="Elemento" />)}
</tbody>
</table>
</div>
</div>
{compliance.otherTests.length > 0 && (
<div>
<h4 className="flex items-center gap-2 font-display font-semibold text-lg mb-4 text-slate-800 dark:text-slate-200">
<span className="w-1.5 h-6 bg-emerald-500 rounded-full"></span>
Outros Testes
</h4>
<div className="overflow-x-auto">
<table className="w-full text-left border-collapse">
<thead className="hidden md:table-header-group text-xs uppercase tracking-wider text-slate-500 dark:text-slate-400 border-b-2 border-slate-100 dark:border-slate-700">
<tr>
<th className="py-3 px-6 font-semibold">Teste</th>
<th className="py-3 px-6 font-semibold">Norma</th>
<th className="py-3 px-6 font-semibold">Certificado</th>
<th className="py-3 px-6 font-semibold w-32">Status</th>
</tr>
</thead>
<tbody className="md:divide-y md:divide-slate-100 dark:md:divide-slate-700/50">
{compliance.otherTests.map((item, i) => <ComplianceTableRow key={i} item={item} headerLabel="Teste" />)}
</tbody>
</table>
</div>
</div>
)}
</div>
</div>
{/* Over-Performance Section */}
{compliance.status === 'CONFORME' && overPerformance.length > 0 && (
<div className="bg-white/70 dark:bg-slate-800/60 backdrop-blur-md border border-white/20 dark:border-slate-700/50 shadow-xl rounded-2xl overflow-hidden border-l-4 border-l-green-500">
<div className="px-6 py-4 bg-green-50/50 dark:bg-green-900/10 border-b border-green-100 dark:border-green-900/30">
<h3 className="text-xl font-display font-bold text-green-800 dark:text-green-300">3. Destaques de Desempenho</h3>
</div>
<div className="p-6">
<p className="text-sm text-slate-600 dark:text-slate-400 mb-6">Este material excede os requisitos mínimos normativos nos seguintes aspectos:</p>
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
{overPerformance.map((item, i) => (
<div key={i} className="flex items-start gap-4 p-4 rounded-xl bg-white/50 dark:bg-slate-800/50 border border-green-100 dark:border-green-900/30 shadow-sm">
<div className="p-2 bg-green-100 dark:bg-green-900/30 rounded-full text-green-600 dark:text-green-400 shrink-0">
<CheckCircleIcon className="w-5 h-5" />
</div>
<div>
<span className="block font-semibold text-slate-800 dark:text-slate-200">{item.property}</span>
<span className="text-green-600 dark:text-green-400 text-sm font-medium">{item.value} superior ao mínimo.</span>
</div>
</div>
))}
</div>
</div>
</div>
)}
{/* Equivalents Section */}
<div className="glass-panel rounded-2xl overflow-hidden">
<div className="px-6 py-4 bg-slate-50/50 dark:bg-slate-800/50 border-b border-slate-200/50 dark:border-slate-700/50 backdrop-blur-sm">
<h3 className="text-xl font-display font-bold text-slate-800 dark:text-white">4. Normas Equivalentes</h3>
</div>
<div className="p-6">
<p className="text-sm text-slate-500 dark:text-slate-400 mb-6">Equivalências internacionais para <span className="font-semibold text-slate-700 dark:text-slate-300">{identification.standards}</span>:</p>
<div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-4">
{equivalents.map((item, i) => (
<div key={i} className="group p-4 rounded-xl bg-gradient-to-br from-slate-50 to-white dark:from-slate-800 dark:to-slate-800/50 border border-slate-200 dark:border-slate-700 shadow-sm hover:shadow-md transition-all hover:-translate-y-1">
<p className="text-xs font-semibold text-slate-400 dark:text-slate-500 uppercase tracking-wider mb-1 group-hover:text-blue-500 transition-colors">{item.system}</p>
<p className="font-bold text-slate-800 dark:text-slate-200 font-mono text-lg">{item.norm}</p>
</div>
))}
</div>
</div>
</div>
</div>
);
};