fix-laudo-editor

This commit is contained in:
João Gustavo 2025-11-10 18:10:50 -03:00
parent ef133fc9c0
commit 96df6c7289
4 changed files with 748 additions and 3 deletions

View File

@ -0,0 +1,733 @@
'use client';
import React, { useState, useRef, useEffect } from 'react';
import { useRouter } from 'next/navigation';
import ProtectedRoute from '@/components/shared/ProtectedRoute';
import { useAuth } from '@/hooks/useAuth';
import { useToast } from '@/hooks/use-toast';
import { listarPacientes } from '@/lib/api';
import { useReports } from '@/hooks/useReports';
import { Button } from '@/components/ui/button';
import { Input } from '@/components/ui/input';
import { Label } from '@/components/ui/label';
import { Textarea } from '@/components/ui/textarea';
import { Select, SelectTrigger, SelectContent, SelectItem, SelectValue } from '@/components/ui/select';
import { FileText, Upload, Settings, Eye, ArrowLeft } from 'lucide-react';
// Helpers para normalizar dados
const getPatientName = (p: any) => p?.full_name ?? p?.nome ?? '';
const getPatientCpf = (p: any) => p?.cpf ?? '';
const getPatientSex = (p: any) => p?.sex ?? p?.sexo ?? '';
const getPatientAge = (p: any) => {
if (!p) return '';
const bd = p?.birth_date ?? p?.data_nascimento ?? p?.birthDate;
if (bd) {
const d = new Date(bd);
if (!isNaN(d.getTime())) {
const age = Math.floor((Date.now() - d.getTime()) / (1000 * 60 * 60 * 24 * 365.25));
return `${age}`;
}
}
return p?.idade ?? p?.age ?? '';
};
export default function LaudosEditorPage() {
const router = useRouter();
const { user, token } = useAuth();
const { toast } = useToast();
const { createNewReport } = useReports();
// Estados principais
const [pacienteSelecionado, setPacienteSelecionado] = useState<any>(null);
const [listaPacientes, setListaPacientes] = useState<any[]>([]);
const [content, setContent] = useState('');
const [activeTab, setActiveTab] = useState('editor');
const [showPreview, setShowPreview] = useState(false);
// Estados para solicitante e prazo
const [solicitanteId, setSolicitanteId] = useState<string>(user?.id || '');
const displaySolicitante = user?.name || '';
const [prazoDate, setPrazoDate] = useState<string>('');
const [prazoTime, setPrazoTime] = useState<string>('');
// Campos do laudo
const [campos, setCampos] = useState({
cid: '',
diagnostico: '',
conclusao: '',
exame: '',
especialidade: '',
mostrarData: true,
mostrarAssinatura: true,
});
// Imagens
const [imagens, setImagens] = useState<any[]>([]);
const [templates] = useState([
'Exame normal, sem alterações significativas',
'Paciente em acompanhamento ambulatorial',
'Recomenda-se retorno em 30 dias',
'Alterações compatíveis com processo inflamatório',
'Resultado dentro dos parâmetros de normalidade',
'Recomendo seguimento com especialista',
]);
// Histórico
const [history, setHistory] = useState<string[]>([]);
const [historyIndex, setHistoryIndex] = useState(-1);
// Carregar pacientes ao montar
useEffect(() => {
async function fetchPacientes() {
try {
if (!token) {
setListaPacientes([]);
return;
}
const pacientes = await listarPacientes();
setListaPacientes(pacientes || []);
} catch (err) {
console.warn('Erro ao carregar pacientes:', err);
setListaPacientes([]);
}
}
fetchPacientes();
}, [token]);
// Atualizar histórico
useEffect(() => {
if (history[historyIndex] !== content) {
const newHistory = history.slice(0, historyIndex + 1);
setHistory([...newHistory, content]);
setHistoryIndex(newHistory.length);
}
}, [content]);
// Desfazer
const handleUndo = () => {
if (historyIndex > 0) {
setContent(history[historyIndex - 1]);
setHistoryIndex(historyIndex - 1);
}
};
// Formatação de texto
const formatText = (type: string, value?: any) => {
const textarea = document.querySelector('textarea') as HTMLTextAreaElement;
if (!textarea) return;
const start = textarea.selectionStart;
const end = textarea.selectionEnd;
const selectedText = textarea.value.substring(start, end);
let formattedText = '';
switch (type) {
case 'bold':
formattedText = selectedText ? `**${selectedText}**` : '**texto em negrito**';
break;
case 'italic':
formattedText = selectedText ? `*${selectedText}*` : '*texto em itálico*';
break;
case 'underline':
formattedText = selectedText ? `__${selectedText}__` : '__texto sublinhado__';
break;
case 'list-ul':
formattedText = selectedText ? selectedText.split('\n').map((l) => `${l}`).join('\n') : '• item da lista';
break;
case 'list-ol':
formattedText = selectedText ? selectedText.split('\n').map((l, i) => `${i + 1}. ${l}`).join('\n') : '1. item da lista';
break;
case 'indent':
formattedText = selectedText ? selectedText.split('\n').map((l) => ` ${l}`).join('\n') : ' ';
break;
case 'outdent':
formattedText = selectedText ? selectedText.split('\n').map((l) => l.replace(/^\s{1,4}/, '')).join('\n') : '';
break;
case 'align-left':
formattedText = selectedText ? `[left]${selectedText}[/left]` : '[left]Texto à esquerda[/left]';
break;
case 'align-center':
formattedText = selectedText ? `[center]${selectedText}[/center]` : '[center]Texto centralizado[/center]';
break;
case 'align-right':
formattedText = selectedText ? `[right]${selectedText}[/right]` : '[right]Texto à direita[/right]';
break;
case 'align-justify':
formattedText = selectedText ? `[justify]${selectedText}[/justify]` : '[justify]Texto justificado[/justify]';
break;
case 'font-size':
formattedText = selectedText ? `[size=${value}]${selectedText}[/size]` : `[size=${value}]Texto tamanho ${value}[/size]`;
break;
case 'font-family':
formattedText = selectedText ? `[font=${value}]${selectedText}[/font]` : `[font=${value}]${value}[/font]`;
break;
case 'font-color':
formattedText = selectedText ? `[color=${value}]${selectedText}[/color]` : `[color=${value}]${value}[/color]`;
break;
default:
return;
}
const newText = textarea.value.substring(0, start) + formattedText + textarea.value.substring(end);
setContent(newText);
};
const insertTemplate = (template: string) => {
setContent((prev: string) => (prev ? `${prev}\n\n${template}` : template));
};
const handleImageUpload = (e: React.ChangeEvent<HTMLInputElement>) => {
const files = Array.from(e.target.files || []);
files.forEach((file) => {
const reader = new FileReader();
reader.onload = (e) => {
setImagens((prev) => [
...prev,
{
id: Date.now() + Math.random(),
name: file.name,
url: e.target?.result,
type: file.type,
},
]);
};
reader.readAsDataURL(file);
});
};
const processContent = (content: string) => {
return content
.replace(/\*\*(.*?)\*\*/g, '<strong>$1</strong>')
.replace(/\*(.*?)\*/g, '<em>$1</em>')
.replace(/__(.*?)__/g, '<u>$1</u>')
.replace(/\[left\]([\s\S]*?)\[\/left\]/g, '<div style="text-align:left">$1</div>')
.replace(/\[center\]([\s\S]*?)\[\/center\]/g, '<div style="text-align:center">$1</div>')
.replace(/\[right\]([\s\S]*?)\[\/right\]/g, '<div style="text-align:right">$1</div>')
.replace(/\[justify\]([\s\S]*?)\[\/justify\]/g, '<div style="text-align:justify">$1</div>')
.replace(/\[size=(\d+)\]([\s\S]*?)\[\/size\]/g, '<span style="font-size:$1px">$2</span>')
.replace(/\[font=([^\]]+)\]([\s\S]*?)\[\/font\]/g, '<span style="font-family:$1">$2</span>')
.replace(/\[color=([^\]]+)\]([\s\S]*?)\[\/color\]/g, '<span style="color:$1">$2</span>')
.replace(/{{diagnostico}}/g, campos.diagnostico || '[DIAGNÓSTICO]')
.replace(/{{conclusao}}/g, campos.conclusao || '[CONCLUSÃO]')
.replace(/\n/g, '<br>');
};
const handleSave = async () => {
try {
if (!pacienteSelecionado?.id) {
toast({
title: 'Erro',
description: 'Selecione um paciente para continuar.',
variant: 'destructive',
});
return;
}
const userId = user?.id || '00000000-0000-0000-0000-000000000001';
let composedDueAt = undefined;
if (prazoDate) {
const t = prazoTime || '23:59';
composedDueAt = new Date(`${prazoDate}T${t}:00`).toISOString();
}
const payload = {
patient_id: pacienteSelecionado?.id,
order_number: '',
exam: campos.exame || '',
diagnosis: campos.diagnostico || '',
conclusion: campos.conclusao || '',
cid_code: campos.cid || '',
content_html: content,
content_json: {},
requested_by: solicitanteId || userId,
due_at: composedDueAt ?? new Date().toISOString(),
hide_date: !campos.mostrarData,
hide_signature: !campos.mostrarAssinatura,
};
if (createNewReport) {
await createNewReport(payload as any);
toast({
title: 'Laudo criado com sucesso!',
description: 'O laudo foi liberado e salvo.',
variant: 'default',
});
// Redirecionar para profissional
router.push('/profissional');
}
} catch (err) {
toast({
title: 'Erro ao criar laudo',
description: (err && typeof err === 'object' && 'message' in err) ? (err as any).message : String(err) || 'Tente novamente.',
variant: 'destructive',
});
}
};
return (
<ProtectedRoute>
<div className="min-h-screen bg-background flex flex-col">
{/* Header */}
<div className="border-b border-border bg-card shadow-sm sticky top-0 z-10">
<div className="px-2 sm:px-4 md:px-6 py-3 sm:py-4 flex items-center justify-between gap-2 sm:gap-4">
<div className="flex items-center gap-2 sm:gap-4 flex-1 min-w-0">
<Button
variant="ghost"
size="sm"
onClick={() => router.push('/profissional')}
className="p-0 h-auto flex-shrink-0"
>
<ArrowLeft className="w-4 sm:w-5 h-4 sm:h-5" />
</Button>
<div className="min-w-0">
<h1 className="text-lg sm:text-2xl font-bold truncate">Novo Laudo Médico</h1>
<p className="text-xs sm:text-sm text-muted-foreground truncate">Crie um novo laudo selecionando um paciente</p>
</div>
</div>
</div>
</div>
{/* Main Content */}
<div className="flex-1 overflow-hidden flex flex-col">
{/* Seleção de Paciente */}
<div className="border-b border-border bg-card px-2 sm:px-4 md:px-6 py-3 sm:py-4 flex-shrink-0 overflow-y-auto max-h-48 sm:max-h-56">
{!pacienteSelecionado ? (
<div className="bg-muted border border-border rounded-lg p-2 sm:p-4">
<Label htmlFor="select-paciente" className="text-xs sm:text-sm font-medium mb-2 block">
Selecionar Paciente *
</Label>
<Select
onValueChange={(value) => {
const paciente = listaPacientes.find((p) => p.id === value);
if (paciente) setPacienteSelecionado(paciente);
}}
>
<SelectTrigger className="w-full text-xs sm:text-sm">
<SelectValue placeholder="Escolha um paciente para criar o laudo" />
</SelectTrigger>
<SelectContent>
{listaPacientes.map((paciente) => (
<SelectItem key={paciente.id} value={paciente.id}>
<span className="text-xs sm:text-sm">
{paciente.full_name} {paciente.cpf ? `- CPF: ${paciente.cpf}` : ''}
</span>
</SelectItem>
))}
</SelectContent>
</Select>
</div>
) : (
<div className="flex flex-col sm:flex-row sm:items-center gap-2 sm:gap-4">
<div className="flex-1 min-w-0">
<div className="font-semibold text-primary text-sm sm:text-lg truncate">{getPatientName(pacienteSelecionado)}</div>
<div className="text-xs sm:text-sm text-muted-foreground line-clamp-2">
{getPatientCpf(pacienteSelecionado) ? `CPF: ${getPatientCpf(pacienteSelecionado)} | ` : ''}
{pacienteSelecionado?.birth_date ? `Nascimento: ${pacienteSelecionado.birth_date}` : getPatientAge(pacienteSelecionado) ? `Idade: ${getPatientAge(pacienteSelecionado)} anos` : ''}
{getPatientSex(pacienteSelecionado) ? ` | Sexo: ${getPatientSex(pacienteSelecionado)}` : ''}
</div>
</div>
<Button
variant="outline"
size="sm"
onClick={() => setPacienteSelecionado(null)}
className="text-xs sm:text-sm flex-shrink-0"
>
Trocar
</Button>
</div>
)}
{/* Solicitante e Prazo */}
{pacienteSelecionado && (
<div className="mt-3 sm:mt-4 grid grid-cols-1 sm:grid-cols-2 gap-2 sm:gap-3">
<div>
<Label htmlFor="solicitante" className="text-xs sm:text-sm">
Solicitante
</Label>
<Input id="solicitante" value={displaySolicitante} readOnly disabled className="text-xs sm:text-sm mt-1 h-8 sm:h-10" />
</div>
<div>
<Label htmlFor="prazoDate" className="text-xs sm:text-sm">
Prazo do Laudo
</Label>
<div className="flex gap-1 sm:gap-2 mt-1">
<Input
id="prazoDate"
type="date"
value={prazoDate}
onChange={(e) => setPrazoDate(e.target.value)}
className="text-xs sm:text-sm h-8 sm:h-10 flex-1"
/>
<Input
id="prazoTime"
type="time"
value={prazoTime}
onChange={(e) => setPrazoTime(e.target.value)}
className="text-xs sm:text-sm h-8 sm:h-10 flex-1"
/>
</div>
<p className="text-xs text-muted-foreground mt-1">Defina a data e hora (opcional).</p>
</div>
</div>
)}
</div>
{/* Tabs */}
<div className="flex border-b border-border bg-card overflow-x-auto flex-shrink-0">
<button
onClick={() => setActiveTab('editor')}
className={`px-2 sm:px-4 py-2 text-xs sm:text-sm font-medium border-b-2 transition-colors whitespace-nowrap ${
activeTab === 'editor'
? 'border-blue-500 text-blue-600'
: 'border-transparent text-gray-600 dark:text-muted-foreground'
}`}
>
<FileText className="w-3 sm:w-4 h-3 sm:h-4 inline mr-1" />
Editor
</button>
<button
onClick={() => setActiveTab('imagens')}
className={`px-2 sm:px-4 py-2 text-xs sm:text-sm font-medium border-b-2 transition-colors whitespace-nowrap ${
activeTab === 'imagens'
? 'border-blue-500 text-blue-600'
: 'border-transparent text-gray-600 dark:text-muted-foreground'
}`}
>
<Upload className="w-3 sm:w-4 h-3 sm:h-4 inline mr-1" />
Imagens ({imagens.length})
</button>
<button
onClick={() => setActiveTab('campos')}
className={`px-2 sm:px-4 py-2 text-xs sm:text-sm font-medium border-b-2 transition-colors whitespace-nowrap ${
activeTab === 'campos'
? 'border-blue-500 text-blue-600'
: 'border-transparent text-gray-600 dark:text-muted-foreground'
}`}
>
<Settings className="w-3 sm:w-4 h-3 sm:h-4 inline mr-1" />
Campos
</button>
<button
onClick={() => setShowPreview(!showPreview)}
className={`px-2 sm:px-4 py-2 text-xs sm:text-sm font-medium border-b-2 transition-colors whitespace-nowrap ${
showPreview ? 'border-green-500 text-green-600' : 'border-transparent text-gray-600 dark:text-muted-foreground'
}`}
>
<Eye className="w-3 sm:w-4 h-3 sm:h-4 inline mr-1" />
<span className="hidden sm:inline">{showPreview ? 'Ocultar' : 'Pré-visualização'}</span>
<span className="sm:hidden">{showPreview ? 'Ocultar' : 'Preview'}</span>
</button>
</div>
{/* Content */}
<div className="flex-1 overflow-hidden flex flex-col md:flex-row bg-background">
{/* Left Panel */}
<div className={`flex flex-col overflow-hidden transition-all ${showPreview ? 'w-full md:w-3/5 h-1/2 md:h-full' : 'w-full'}`}>
{/* Editor Tab */}
{activeTab === 'editor' && (
<div className="flex-1 flex flex-col overflow-hidden">
{/* Toolbar */}
<div className="p-1.5 sm:p-2 md:p-3 border-b border-border overflow-x-auto bg-card flex-shrink-0">
<div className="flex flex-wrap gap-1 sm:gap-2 items-center">
<label className="text-xs mr-0.5 sm:mr-1 whitespace-nowrap">Tam</label>
<input
type="number"
min={8}
max={32}
defaultValue={14}
onBlur={(e) => formatText('font-size', e.target.value)}
className="w-10 sm:w-12 border rounded px-1 py-0.5 text-xs"
title="Tamanho da fonte"
/>
<label className="text-xs mr-0.5 sm:mr-1 whitespace-nowrap hidden sm:inline">Fonte</label>
<select
defaultValue={'Arial'}
onBlur={(e) => formatText('font-family', e.target.value)}
className="border rounded px-1 py-0.5 text-xs bg-background text-foreground hidden sm:block"
style={{ minWidth: 80 }}
>
<option value="Arial">Arial</option>
<option value="Helvetica">Helvetica</option>
<option value="Times New Roman">Times New Roman</option>
<option value="Courier New">Courier New</option>
<option value="Verdana">Verdana</option>
<option value="Georgia">Georgia</option>
</select>
<label className="text-xs mr-1 hidden sm:inline">Cor</label>
<input
type="color"
defaultValue="#222222"
onBlur={(e) => formatText('font-color', e.target.value)}
className="w-7 sm:w-8 h-7 sm:h-8 border rounded hidden sm:block"
title="Cor da fonte"
/>
<Button variant="outline" size="sm" onClick={() => formatText('align-left')} title="Alinhar à esquerda" className="px-1.5 text-xs h-8">
</Button>
<Button variant="outline" size="sm" onClick={() => formatText('align-center')} title="Centralizar" className="px-1.5 text-xs h-8">
·
</Button>
<Button variant="outline" size="sm" onClick={() => formatText('align-right')} title="Alinhar à direita" className="px-1.5 text-xs h-8">
</Button>
<Button variant="outline" size="sm" onClick={() => formatText('list-ol')} title="Lista numerada" className="px-1.5 text-xs h-8">
1.
</Button>
<Button variant="outline" size="sm" onClick={() => formatText('list-ul')} title="Lista com marcadores" className="px-1.5 text-xs h-8">
</Button>
<Button variant="outline" size="sm" onClick={handleUndo} title="Desfazer" className="px-1.5 text-xs h-8">
</Button>
<div className="hidden md:flex flex-wrap gap-1">
{templates.map((template, idx) => (
<Button
key={idx}
variant="ghost"
size="sm"
className="text-xs h-auto p-1 px-1.5"
onClick={() => insertTemplate(template)}
title={template}
>
{template.substring(0, 15)}...
</Button>
))}
</div>
</div>
</div>
{/* Editor Textarea */}
<div className="flex-1 overflow-hidden p-2 sm:p-3 md:p-4">
<Textarea
value={content}
onChange={(e) => setContent(e.target.value)}
placeholder="Digite o conteúdo do laudo aqui. Use ** para negrito, * para itálico."
className="w-full h-full resize-none text-xs sm:text-sm"
/>
</div>
</div>
)}
{/* Imagens Tab */}
{activeTab === 'imagens' && (
<div className="flex-1 p-2 sm:p-3 md:p-4 overflow-y-auto">
<div className="mb-3 sm:mb-4">
<Label htmlFor="upload-images" className="text-xs sm:text-sm">
Upload de Imagens
</Label>
<Input
id="upload-images"
type="file"
multiple
accept="image/*,.pdf"
onChange={handleImageUpload}
className="mt-1 sm:mt-2 text-xs"
/>
</div>
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-2 lg:grid-cols-3 gap-2 sm:gap-3 md:gap-4">
{imagens.map((img) => (
<div key={img.id} className="border border-border rounded-lg p-1.5 sm:p-2">
{img.type.startsWith('image/') ? (
<img src={img.url} alt={img.name} className="w-full h-20 sm:h-24 md:h-28 object-cover rounded" />
) : (
<div className="w-full h-20 sm:h-24 md:h-28 bg-muted rounded flex items-center justify-center">
<FileText className="w-6 sm:w-8 h-6 sm:h-8 text-muted-foreground" />
</div>
)}
<p className="text-xs text-muted-foreground mt-1 truncate">{img.name}</p>
<Button
variant="destructive"
size="sm"
className="w-full mt-1 text-xs h-8"
onClick={() => setImagens((prev) => prev.filter((i) => i.id !== img.id))}
>
Remover
</Button>
</div>
))}
</div>
</div>
)}
{/* Campos Tab */}
{activeTab === 'campos' && (
<div className="flex-1 p-2 sm:p-3 md:p-4 space-y-2 sm:space-y-3 md:space-y-4 overflow-y-auto">
<div>
<Label htmlFor="cid" className="text-xs sm:text-sm">
CID
</Label>
<Input
id="cid"
value={campos.cid}
onChange={(e) => setCampos((prev) => ({ ...prev, cid: e.target.value }))}
placeholder="Ex: M25.5, I10, etc."
className="text-xs sm:text-sm mt-1 h-8 sm:h-10"
/>
</div>
<div>
<Label htmlFor="exame" className="text-xs sm:text-sm">
Exame
</Label>
<Input
id="exame"
value={campos.exame}
onChange={(e) => setCampos((prev) => ({ ...prev, exame: e.target.value }))}
placeholder="Exame realizado"
className="text-xs sm:text-sm mt-1 h-8 sm:h-10"
/>
</div>
<div>
<Label htmlFor="diagnostico" className="text-xs sm:text-sm">
Diagnóstico
</Label>
<Textarea
id="diagnostico"
value={campos.diagnostico}
onChange={(e) => setCampos((prev) => ({ ...prev, diagnostico: e.target.value }))}
placeholder="Diagnóstico principal"
rows={2}
className="text-xs sm:text-sm mt-1"
/>
</div>
<div>
<Label htmlFor="conclusao" className="text-xs sm:text-sm">
Conclusão
</Label>
<Textarea
id="conclusao"
value={campos.conclusao}
onChange={(e) => setCampos((prev) => ({ ...prev, conclusao: e.target.value }))}
placeholder="Conclusão do laudo"
rows={2}
className="text-xs sm:text-sm mt-1"
/>
</div>
<div className="space-y-1 sm:space-y-2">
<div className="flex items-center gap-2">
<input
type="checkbox"
id="mostrar-data"
checked={campos.mostrarData}
onChange={(e) => setCampos((prev) => ({ ...prev, mostrarData: e.target.checked }))}
className="w-4 h-4"
/>
<Label htmlFor="mostrar-data" className="text-xs sm:text-sm">
Mostrar data no laudo
</Label>
</div>
<div className="flex items-center gap-2">
<input
type="checkbox"
id="mostrar-assinatura"
checked={campos.mostrarAssinatura}
onChange={(e) => setCampos((prev) => ({ ...prev, mostrarAssinatura: e.target.checked }))}
className="w-4 h-4"
/>
<Label htmlFor="mostrar-assinatura" className="text-xs sm:text-sm">
Mostrar assinatura no laudo
</Label>
</div>
</div>
</div>
)}
</div>
{/* Preview Panel */}
{showPreview && (
<div className="w-full md:w-2/5 h-1/2 md:h-full border-t md:border-l md:border-t-0 border-border bg-muted/20 flex flex-col overflow-hidden">
<div className="p-2 sm:p-2.5 md:p-3 border-b border-border flex-shrink-0 bg-card">
<h3 className="font-semibold text-xs sm:text-sm text-foreground truncate">Pré-visualização</h3>
</div>
<div className="flex-1 overflow-y-auto p-2 sm:p-2.5 md:p-3">
<div className="bg-background border border-border rounded p-2 sm:p-2.5 md:p-3 text-xs space-y-1.5 sm:space-y-2">
{/* Header */}
<div className="text-center mb-2 pb-2 border-b border-border/40">
<h2 className="text-xs sm:text-sm font-bold leading-tight line-clamp-2">
LAUDO {campos.especialidade ? `- ${campos.especialidade.toUpperCase().substring(0, 12)}` : ''}
</h2>
{campos.exame && <p className="text-xs font-semibold mt-1 line-clamp-2">{campos.exame}</p>}
{campos.mostrarData && (
<p className="text-xs text-muted-foreground mt-1">{new Date().toLocaleDateString('pt-BR')}</p>
)}
</div>
{/* Paciente */}
{pacienteSelecionado && (
<div className="mb-1.5 pb-1.5 border-b border-border/40 space-y-0.5">
<div className="text-xs line-clamp-1">
<span className="font-semibold">Paciente:</span> {getPatientName(pacienteSelecionado).substring(0, 20)}
</div>
<div className="text-xs line-clamp-1">
<span className="font-semibold">CPF:</span> {getPatientCpf(pacienteSelecionado).substring(0, 14)}
</div>
</div>
)}
{/* Informações Clínicas */}
<div className="mb-1.5 pb-1.5 border-b border-border/40 space-y-0.5">
{campos.cid && (
<div className="text-xs line-clamp-1">
<span className="font-semibold">CID:</span>{' '}
<span className="text-blue-600 dark:text-blue-400 font-semibold">{campos.cid}</span>
</div>
)}
</div>
{/* Diagnóstico - Completo */}
{campos.diagnostico && (
<div className="mb-1.5 pb-1.5 border-b border-border/40">
<div className="text-xs font-semibold mb-0.5">Diagnóstico:</div>
<div className="text-xs line-clamp-3 text-muted-foreground">
{campos.diagnostico}
</div>
</div>
)}
{/* Conteúdo */}
{content && (
<div className="mb-1.5 pb-1.5 border-b border-border/40">
<div className="text-xs font-semibold mb-0.5">Conteúdo:</div>
<div
className="text-xs leading-tight whitespace-pre-wrap line-clamp-3 md:line-clamp-4 text-muted-foreground"
dangerouslySetInnerHTML={{
__html: processContent(content).substring(0, 150),
}}
/>
</div>
)}
{/* Conclusão - Completa */}
{campos.conclusao && (
<div className="mb-1.5 pb-1.5 border-b border-border/40">
<div className="text-xs font-semibold mb-0.5">Conclusão:</div>
<div className="text-xs line-clamp-3 text-muted-foreground">
{campos.conclusao}
</div>
</div>
)}
</div>
</div>
</div>
)}
</div>
</div>
{/* Footer */}
<div className="p-2 sm:p-3 md:p-4 border-t border-border bg-card flex-shrink-0">
<div className="flex flex-col sm:flex-row items-stretch sm:items-center justify-between gap-2 sm:gap-4">
<div className="text-xs text-muted-foreground hidden md:block">
Editor de relatórios com formatação de texto rica.
</div>
<div className="flex gap-2 w-full sm:w-auto">
<Button variant="outline" onClick={() => router.push('/profissional')} className="flex-1 sm:flex-none text-xs sm:text-sm h-8 sm:h-10">
Cancelar
</Button>
<Button onClick={handleSave} className="flex-1 sm:flex-none text-xs sm:text-sm h-8 sm:h-10">
Liberar Laudo
</Button>
</div>
</div>
</div>
</div>
</ProtectedRoute>
);
}

View File

@ -3,6 +3,7 @@ import React, { useState, useRef, useEffect } from "react";
import Image from "next/image";
import SignatureCanvas from "react-signature-canvas";
import Link from "next/link";
import { useRouter } from "next/navigation";
import ProtectedRoute from "@/components/shared/ProtectedRoute";
import { useAuth } from "@/hooks/useAuth";
import { useToast } from "@/hooks/use-toast";
@ -116,6 +117,7 @@ const colorsByType = {
};
const ProfissionalPage = () => {
const router = useRouter();
const { logout, user, token } = useAuth();
const { toast } = useToast();
const [activeSection, setActiveSection] = useState('calendario');
@ -1367,7 +1369,7 @@ const ProfissionalPage = () => {
<p className="text-muted-foreground">Nesta seção você pode gerenciar todos os laudos gerados.</p>
</div>
<Button
onClick={() => setIsCreatingNew(true)}
onClick={() => router.push('/laudos-editor')}
className="flex items-center gap-2"
>
<Plus className="w-4 h-4" />

View File

@ -142,10 +142,15 @@ export function AuthProvider({ children }: { children: ReactNode }) {
} catch (e) {
console.warn('[AUTH] Falha ao buscar /auth/v1/user durante restauração de sessão:', e)
}
// Tentar buscar profile consolidado (user-info) e mesclar
// Tentar buscar profile consolidado (user-info) e mesclar
try {
const info = await getUserInfo()
if (info?.profile) {
// Atualizar nome se vem no profile
if (info.profile.full_name) {
userData.name = info.profile.full_name
console.log('[AUTH] Nome atualizado de profile na restauração:', info.profile.full_name)
}
const mapped = {
cpf: (info.profile as any).cpf ?? userData.profile?.cpf,
crm: (info.profile as any).crm ?? userData.profile?.crm,
@ -241,6 +246,11 @@ export function AuthProvider({ children }: { children: ReactNode }) {
try {
const info = await getUserInfo()
if (info?.profile && response.user) {
// Atualizar nome se vem no profile
if (info.profile.full_name) {
response.user.name = info.profile.full_name
console.log('[AUTH] Nome atualizado de profile:', info.profile.full_name)
}
const mapped = {
cpf: (info.profile as any).cpf ?? response.user.profile?.cpf,
crm: (info.profile as any).crm ?? response.user.profile?.crm,

View File

@ -156,7 +156,7 @@ export async function loginUser(
user: {
id: data.user?.id || data.id || "1",
email: email,
name: data.user?.name || data.name || email.split('@')[0],
name: data.user?.name || data.name || data.user?.full_name || data.full_name || email.split('@')[0],
userType: userType,
profile: data.user?.profile || data.profile || {}
}