Se prepara chat asignaturas para ia
This commit is contained in:
@@ -1,4 +1,5 @@
|
|||||||
import { useParams, useRouterState } from '@tanstack/react-router'
|
import { useQueryClient } from '@tanstack/react-query'
|
||||||
|
import { useParams } from '@tanstack/react-router'
|
||||||
import {
|
import {
|
||||||
Sparkles,
|
Sparkles,
|
||||||
Send,
|
Send,
|
||||||
@@ -10,13 +11,15 @@ import {
|
|||||||
BookOpen,
|
BookOpen,
|
||||||
Check,
|
Check,
|
||||||
X,
|
X,
|
||||||
|
MessageSquarePlus,
|
||||||
|
Archive,
|
||||||
|
History, // Agregado
|
||||||
} from 'lucide-react'
|
} from 'lucide-react'
|
||||||
import { useState, useEffect, useRef, useMemo } from 'react'
|
import { useState, useEffect, useRef, useMemo } from 'react'
|
||||||
|
|
||||||
import type { IASugerencia } from '@/types/asignatura'
|
import type { IASugerencia } from '@/types/asignatura'
|
||||||
|
|
||||||
import { Avatar, AvatarFallback } from '@/components/ui/avatar'
|
import { Avatar, AvatarFallback } from '@/components/ui/avatar'
|
||||||
import { Badge } from '@/components/ui/badge'
|
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
import { ScrollArea } from '@/components/ui/scroll-area'
|
import { ScrollArea } from '@/components/ui/scroll-area'
|
||||||
import { Textarea } from '@/components/ui/textarea'
|
import { Textarea } from '@/components/ui/textarea'
|
||||||
@@ -25,39 +28,10 @@ import {
|
|||||||
useConversationBySubject,
|
useConversationBySubject,
|
||||||
useMessagesBySubjectChat,
|
useMessagesBySubjectChat,
|
||||||
useSubject,
|
useSubject,
|
||||||
useUpdateSubjectRecommendation,
|
useUpdateSubjectConversationStatus,
|
||||||
} from '@/data'
|
} from '@/data'
|
||||||
import { cn } from '@/lib/utils'
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
// Tipos importados de tu archivo de asignatura
|
|
||||||
|
|
||||||
const PRESETS = [
|
|
||||||
{
|
|
||||||
id: 'mejorar-objetivo',
|
|
||||||
label: 'Mejorar objetivo',
|
|
||||||
icon: Target,
|
|
||||||
prompt: 'Mejora la redacción del objetivo de esta asignatura...',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'contenido-tematico',
|
|
||||||
label: 'Sugerir contenido',
|
|
||||||
icon: BookOpen,
|
|
||||||
prompt: 'Genera un desglose de temas para esta asignatura...',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'actividades',
|
|
||||||
label: 'Actividades de aprendizaje',
|
|
||||||
icon: GraduationCap,
|
|
||||||
prompt: 'Sugiere actividades prácticas para los temas seleccionados...',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'bibliografia',
|
|
||||||
label: 'Actualizar bibliografía',
|
|
||||||
icon: FileText,
|
|
||||||
prompt: 'Recomienda bibliografía reciente para esta asignatura...',
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
interface SelectedField {
|
interface SelectedField {
|
||||||
key: string
|
key: string
|
||||||
label: string
|
label: string
|
||||||
@@ -65,55 +39,74 @@ interface SelectedField {
|
|||||||
}
|
}
|
||||||
|
|
||||||
interface IAAsignaturaTabProps {
|
interface IAAsignaturaTabProps {
|
||||||
asignatura: Record<string, any>
|
asignatura?: Record<string, any>
|
||||||
|
|
||||||
onSendMessage: (message: string, campoId?: string) => void
|
|
||||||
onAcceptSuggestion: (sugerencia: IASugerencia) => void
|
onAcceptSuggestion: (sugerencia: IASugerencia) => void
|
||||||
onRejectSuggestion: (messageId: string) => void
|
onRejectSuggestion: (messageId: string) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export function IAAsignaturaTab({
|
export function IAAsignaturaTab({
|
||||||
onSendMessage,
|
|
||||||
onAcceptSuggestion,
|
onAcceptSuggestion,
|
||||||
onRejectSuggestion,
|
onRejectSuggestion,
|
||||||
}: IAAsignaturaTabProps) {
|
}: IAAsignaturaTabProps) {
|
||||||
const routerState = useRouterState()
|
const queryClient = useQueryClient()
|
||||||
const { asignaturaId } = useParams({
|
const { asignaturaId } = useParams({
|
||||||
from: '/planes/$planId/asignaturas/$asignaturaId',
|
from: '/planes/$planId/asignaturas/$asignaturaId',
|
||||||
})
|
})
|
||||||
|
|
||||||
const { data: datosGenerales, isLoading: loadingAsig } =
|
// --- ESTADOS ---
|
||||||
useSubject(asignaturaId)
|
const [activeChatId, setActiveChatId] = useState<string | undefined>(
|
||||||
// ESTADOS PRINCIPALES (Igual que en Planes)
|
undefined,
|
||||||
|
)
|
||||||
|
const [showArchived, setShowArchived] = useState(false)
|
||||||
const [input, setInput] = useState('')
|
const [input, setInput] = useState('')
|
||||||
const [selectedFields, setSelectedFields] = useState<Array<SelectedField>>([])
|
const [selectedFields, setSelectedFields] = useState<Array<SelectedField>>([])
|
||||||
const [showSuggestions, setShowSuggestions] = useState(false)
|
const [showSuggestions, setShowSuggestions] = useState(false)
|
||||||
const [isLoading, setIsLoading] = useState(false)
|
|
||||||
const scrollRef = useRef<HTMLDivElement>(null)
|
|
||||||
const [isSending, setIsSending] = useState(false)
|
const [isSending, setIsSending] = useState(false)
|
||||||
|
const scrollRef = useRef<HTMLDivElement>(null)
|
||||||
|
|
||||||
const { data: conversaciones } = useConversationBySubject(asignaturaId)
|
// --- DATA QUERIES ---
|
||||||
const activeConversationId = conversaciones?.[0]?.id
|
const { data: datosGenerales } = useSubject(asignaturaId)
|
||||||
|
const { data: todasConversaciones, isLoading: loadingConv } =
|
||||||
const { data: rawMessages } = useMessagesBySubjectChat(activeConversationId)
|
useConversationBySubject(asignaturaId)
|
||||||
|
const { data: rawMessages } = useMessagesBySubjectChat(activeChatId, {
|
||||||
|
enabled: !!activeChatId,
|
||||||
|
})
|
||||||
const { mutateAsync: sendMessage } = useAISubjectChat()
|
const { mutateAsync: sendMessage } = useAISubjectChat()
|
||||||
const { mutate: applySuggestion } = useUpdateSubjectRecommendation()
|
const { mutate: updateStatus } = useUpdateSubjectConversationStatus()
|
||||||
|
const [isCreatingNewChat, setIsCreatingNewChat] = useState(false)
|
||||||
|
const hasInitialSelected = useRef(false)
|
||||||
|
|
||||||
|
// --- AUTO-SCROLL ---
|
||||||
|
useEffect(() => {
|
||||||
|
const viewport = scrollRef.current?.querySelector(
|
||||||
|
'[data-radix-scroll-area-viewport]',
|
||||||
|
)
|
||||||
|
if (viewport) {
|
||||||
|
viewport.scrollTop = viewport.scrollHeight
|
||||||
|
}
|
||||||
|
}, [rawMessages, isSending])
|
||||||
|
|
||||||
|
// --- FILTRADO DE CHATS ---
|
||||||
|
const { activeChats, archivedChats } = useMemo(() => {
|
||||||
|
const chats = todasConversaciones || []
|
||||||
|
return {
|
||||||
|
activeChats: chats.filter((c: any) => c.estado === 'ACTIVA'),
|
||||||
|
archivedChats: chats.filter((c: any) => c.estado === 'ARCHIVADA'),
|
||||||
|
}
|
||||||
|
}, [todasConversaciones])
|
||||||
|
|
||||||
|
// --- PROCESAMIENTO DE MENSAJES ---
|
||||||
const messages = useMemo(() => {
|
const messages = useMemo(() => {
|
||||||
return rawMessages
|
if (!rawMessages) return []
|
||||||
?.map((m) => ({
|
return rawMessages.flatMap((m) => {
|
||||||
id: m.id,
|
const msgs = []
|
||||||
role: 'user', // El mensaje del usuario
|
msgs.push({ id: `${m.id}-user`, role: 'user', content: m.mensaje })
|
||||||
content: m.mensaje,
|
if (m.respuesta) {
|
||||||
sugerencia: null,
|
msgs.push({
|
||||||
}))
|
|
||||||
.concat(
|
|
||||||
rawMessages?.map((m) => ({
|
|
||||||
id: `${m.id}-ai`,
|
id: `${m.id}-ai`,
|
||||||
role: 'assistant',
|
role: 'assistant',
|
||||||
content: m.respuesta,
|
content: m.respuesta,
|
||||||
sugerencia:
|
sugerencia: m.propuesta?.recommendations?.[0]
|
||||||
m.propuesta?.recommendations?.length > 0
|
|
||||||
? {
|
? {
|
||||||
id: m.id,
|
id: m.id,
|
||||||
campoKey: m.propuesta.recommendations[0].campo_afectado,
|
campoKey: m.propuesta.recommendations[0].campo_afectado,
|
||||||
@@ -126,148 +119,218 @@ export function IAAsignaturaTab({
|
|||||||
aceptada: m.propuesta.recommendations[0].aplicada,
|
aceptada: m.propuesta.recommendations[0].aplicada,
|
||||||
}
|
}
|
||||||
: null,
|
: null,
|
||||||
})),
|
})
|
||||||
)
|
}
|
||||||
.sort((a, b) => (a.id > b.id ? 1 : -1)) // Unir y ordenar (simplificado)
|
return msgs
|
||||||
|
})
|
||||||
// NOTA: En producción, es mejor que tu query devuelva los mensajes ya intercalados
|
|
||||||
// o usar el campo 'conversacion_json' de la tabla padre para mayor fidelidad.
|
|
||||||
}, [rawMessages])
|
}, [rawMessages])
|
||||||
|
|
||||||
// 1. Transformar datos de la asignatura para el menú
|
// Auto-selección inicial
|
||||||
const availableFields = useMemo(() => {
|
|
||||||
if (!datosGenerales?.datos) return []
|
|
||||||
|
|
||||||
const estructuraProps =
|
|
||||||
datosGenerales?.estructuras_asignatura?.definicion?.properties || {}
|
|
||||||
|
|
||||||
return Object.keys(datosGenerales.datos).map((key) => {
|
|
||||||
const estructuraCampo = estructuraProps[key]
|
|
||||||
|
|
||||||
const labelAmigable =
|
|
||||||
estructuraCampo?.title ||
|
|
||||||
key.replace(/_/g, ' ').replace(/\b\w/g, (l) => l.toUpperCase())
|
|
||||||
|
|
||||||
return {
|
|
||||||
key,
|
|
||||||
label: labelAmigable,
|
|
||||||
value: String(datosGenerales.datos[key] || ''),
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}, [datosGenerales])
|
|
||||||
|
|
||||||
// 2. Manejar el estado inicial si viene de "Datos de Asignatura" (Prefill)
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const state = routerState.location.state as any
|
// Si ya hay un chat, o si el usuario ya interactuó (hasInitialSelected), abortamos.
|
||||||
|
if (activeChatId || hasInitialSelected.current) return
|
||||||
|
|
||||||
if (state?.prefillCampo && availableFields.length > 0) {
|
if (activeChats.length > 0 && !loadingConv) {
|
||||||
console.log(state?.prefillCampo)
|
setActiveChatId(activeChats[0].id)
|
||||||
console.log(availableFields)
|
hasInitialSelected.current = true
|
||||||
|
|
||||||
const field = availableFields.find((f) => f.key === state.prefillCampo)
|
|
||||||
|
|
||||||
if (field && !selectedFields.find((sf) => sf.key === field.key)) {
|
|
||||||
setSelectedFields([field])
|
|
||||||
// Sincronizamos el texto inicial con el campo pre-seleccionado
|
|
||||||
setInput(`Mejora el campo ${field.label}: `)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
}, [availableFields])
|
|
||||||
|
|
||||||
// Scroll automático
|
|
||||||
useEffect(() => {
|
|
||||||
if (scrollRef.current) {
|
|
||||||
scrollRef.current.scrollTop = scrollRef.current.scrollHeight
|
|
||||||
}
|
|
||||||
}, [messages, isLoading])
|
|
||||||
|
|
||||||
// 3. Lógica para el disparador ":"
|
|
||||||
const handleInputChange = (e: React.ChangeEvent<HTMLTextAreaElement>) => {
|
|
||||||
const val = e.target.value
|
|
||||||
setInput(val)
|
|
||||||
setShowSuggestions(val.endsWith(':'))
|
|
||||||
}
|
|
||||||
|
|
||||||
const toggleField = (field: SelectedField) => {
|
|
||||||
setSelectedFields((prev) => {
|
|
||||||
const isSelected = prev.find((f) => f.key === field.key)
|
|
||||||
|
|
||||||
// 1. Si ya está seleccionado, lo quitamos (Toggle OFF)
|
|
||||||
if (isSelected) {
|
|
||||||
return prev.filter((f) => f.key !== field.key)
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. Si no está, lo agregamos a la lista (Toggle ON)
|
|
||||||
const newSelected = [...prev, field]
|
|
||||||
|
|
||||||
// 3. Actualizamos el texto del input para reflejar los títulos (labels)
|
|
||||||
setInput((prevText) => {
|
|
||||||
// Separamos lo que el usuario escribió antes del disparador ":"
|
|
||||||
// y lo que viene después (posibles keys/labels previos)
|
|
||||||
const parts = prevText.split(':')
|
|
||||||
const beforeColon = parts[0]
|
|
||||||
|
|
||||||
// Creamos un string con los labels de todos los campos seleccionados
|
|
||||||
const labelsPath = newSelected.map((f) => f.label).join(', ')
|
|
||||||
|
|
||||||
return `${beforeColon.trim()}: ${labelsPath} `
|
|
||||||
})
|
|
||||||
|
|
||||||
return newSelected
|
|
||||||
})
|
|
||||||
|
|
||||||
// Opcional: mantener abierto si quieres que el usuario elija varios seguidos
|
|
||||||
// setShowSuggestions(false)
|
|
||||||
}
|
|
||||||
|
|
||||||
const buildPrompt = (userInput: string) => {
|
|
||||||
if (selectedFields.length === 0) return userInput
|
|
||||||
const fieldsText = selectedFields
|
|
||||||
.map((f) => `- ${f.label}: ${f.value || '(vacio)'}`)
|
|
||||||
.join('\n')
|
|
||||||
|
|
||||||
return `${userInput}\n\nCampos a analizar:\n${fieldsText}`.trim()
|
|
||||||
}
|
}
|
||||||
|
}, [activeChats, loadingConv])
|
||||||
|
|
||||||
const handleSend = async (promptOverride?: string) => {
|
const handleSend = async (promptOverride?: string) => {
|
||||||
const rawText = promptOverride || input
|
const text = promptOverride || input
|
||||||
if (!rawText.trim() && selectedFields.length === 0) return
|
if (!text.trim() && selectedFields.length === 0) return
|
||||||
|
|
||||||
setIsSending(true)
|
setIsSending(true)
|
||||||
try {
|
try {
|
||||||
await sendMessage({
|
const response = await sendMessage({
|
||||||
subjectId: asignaturaId as any,
|
subjectId: asignaturaId as any, // Importante: se usa para crear la conv si activeChatId es undefined
|
||||||
content: rawText,
|
content: text,
|
||||||
campos: selectedFields.map((f) => f.key),
|
campos: selectedFields.map((f) => f.key),
|
||||||
conversacionId: activeConversationId,
|
conversacionId: activeChatId, // Si es undefined, la mutación crea el chat automáticamente
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// IMPORTANTE: Después de la respuesta, actualizamos el ID activo con el que creó el backend
|
||||||
|
if (response.conversacionId) {
|
||||||
|
setActiveChatId(response.conversacionId)
|
||||||
|
}
|
||||||
|
|
||||||
setInput('')
|
setInput('')
|
||||||
setSelectedFields([])
|
setSelectedFields([])
|
||||||
|
|
||||||
|
// Invalidamos la lista de conversaciones para que el nuevo chat aparezca en el historial (panel izquierdo)
|
||||||
|
queryClient.invalidateQueries({
|
||||||
|
queryKey: ['conversation-by-subject', asignaturaId],
|
||||||
|
})
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error al enviar mensaje:', error)
|
||||||
} finally {
|
} finally {
|
||||||
setIsSending(false)
|
setIsSending(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const toggleField = (field: SelectedField) => {
|
||||||
|
setSelectedFields((prev) =>
|
||||||
|
prev.find((f) => f.key === field.key)
|
||||||
|
? prev.filter((f) => f.key !== field.key)
|
||||||
|
: [...prev, field],
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const availableFields = useMemo(() => {
|
||||||
|
if (!datosGenerales?.datos) return []
|
||||||
|
const estructuraProps =
|
||||||
|
datosGenerales?.estructuras_asignatura?.definicion?.properties || {}
|
||||||
|
return Object.keys(datosGenerales.datos).map((key) => ({
|
||||||
|
key,
|
||||||
|
label:
|
||||||
|
estructuraProps[key]?.title || key.replace(/_/g, ' ').toUpperCase(),
|
||||||
|
value: String(datosGenerales.datos[key] || ''),
|
||||||
|
}))
|
||||||
|
}, [datosGenerales])
|
||||||
|
|
||||||
|
const createNewChat = () => {
|
||||||
|
setActiveChatId(undefined) // Al ser undefined, el próximo mensaje creará la charla en el backend
|
||||||
|
setInput('')
|
||||||
|
setSelectedFields([])
|
||||||
|
// Opcional: podrías forzar el foco al textarea aquí con una ref
|
||||||
|
}
|
||||||
|
|
||||||
|
const PRESETS = [
|
||||||
|
{
|
||||||
|
id: 'mejorar-obj',
|
||||||
|
label: 'Mejorar objetivo',
|
||||||
|
icon: Target,
|
||||||
|
prompt: 'Mejora la redacción del objetivo...',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'sugerir-cont',
|
||||||
|
label: 'Sugerir contenido',
|
||||||
|
icon: BookOpen,
|
||||||
|
prompt: 'Genera un desglose de temas...',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'actividades',
|
||||||
|
label: 'Actividades',
|
||||||
|
icon: GraduationCap,
|
||||||
|
prompt: 'Sugiere actividades prácticas...',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex h-[calc(100vh-160px)] max-h-[calc(100vh-160px)] w-full gap-6 overflow-hidden p-4">
|
<div className="flex h-[calc(100vh-160px)] w-full gap-6 overflow-hidden p-4">
|
||||||
{/* PANEL DE CHAT PRINCIPAL */}
|
{/* PANEL IZQUIERDO */}
|
||||||
<div className="relative flex min-w-0 flex-[3] flex-col overflow-hidden rounded-xl border border-slate-200 bg-slate-50/50 shadow-sm">
|
<div className="flex w-64 flex-col border-r pr-4">
|
||||||
{/* Barra superior */}
|
<div className="mb-4 flex items-center justify-between px-2">
|
||||||
<div className="shrink-0 border-b bg-white p-3">
|
<h2 className="flex items-center gap-2 text-xs font-bold text-slate-500 uppercase">
|
||||||
<div className="flex flex-wrap items-center gap-2">
|
<History size={14} /> Historial
|
||||||
<span className="text-[10px] font-bold tracking-wider text-slate-400 uppercase">
|
</h2>
|
||||||
IA de Asignatura
|
<Button
|
||||||
</span>
|
variant="ghost"
|
||||||
</div>
|
size="icon"
|
||||||
|
className={cn(
|
||||||
|
'h-8 w-8',
|
||||||
|
showArchived && 'bg-teal-50 text-teal-600',
|
||||||
|
)}
|
||||||
|
onClick={() => setShowArchived(!showArchived)}
|
||||||
|
>
|
||||||
|
<Archive size={16} />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
// 1. Limpiamos el ID
|
||||||
|
setActiveChatId(undefined)
|
||||||
|
// 2. Marcamos que ya hubo una "interacción inicial" para que el useEffect no actúe
|
||||||
|
hasInitialSelected.current = true
|
||||||
|
// 3. Limpiamos estados visuales
|
||||||
|
setIsCreatingNewChat(true)
|
||||||
|
setInput('')
|
||||||
|
setSelectedFields([])
|
||||||
|
|
||||||
|
// 4. Opcional: Limpiar el caché de mensajes actual para que la pantalla se vea vacía al instante
|
||||||
|
queryClient.setQueryData(['subject-messages', undefined], [])
|
||||||
|
}}
|
||||||
|
variant="outline"
|
||||||
|
className="mb-4 w-full justify-start gap-2 border-dashed border-slate-300 hover:border-teal-500"
|
||||||
|
>
|
||||||
|
<MessageSquarePlus size={18} /> Nuevo Chat
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<ScrollArea className="flex-1">
|
||||||
|
<div className="space-y-1 pr-3">
|
||||||
|
{(showArchived ? archivedChats : activeChats).map((chat: any) => (
|
||||||
|
<div
|
||||||
|
key={chat.id}
|
||||||
|
onClick={() => {
|
||||||
|
setActiveChatId(chat.id)
|
||||||
|
setIsCreatingNewChat(false) // <--- Volvemos al modo normal
|
||||||
|
}}
|
||||||
|
className={cn(
|
||||||
|
'group relative flex cursor-pointer items-center gap-3 rounded-lg px-3 py-2 text-sm transition-all',
|
||||||
|
activeChatId === chat.id
|
||||||
|
? 'bg-teal-50 font-medium text-teal-900'
|
||||||
|
: 'text-slate-600 hover:bg-slate-100',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<FileText size={14} className="shrink-0 opacity-50" />
|
||||||
|
<span className="flex-1 truncate">
|
||||||
|
{chat.titulo || 'Conversación'}
|
||||||
|
</span>
|
||||||
|
<button
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation()
|
||||||
|
updateStatus(
|
||||||
|
{
|
||||||
|
id: chat.id,
|
||||||
|
estado: showArchived ? 'ACTIVA' : 'ARCHIVADA',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onSuccess: () =>
|
||||||
|
queryClient.invalidateQueries({
|
||||||
|
queryKey: ['conversation-by-subject'],
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
className="rounded p-1 opacity-0 group-hover:opacity-100 hover:bg-slate-200"
|
||||||
|
>
|
||||||
|
<Archive size={12} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* PANEL CENTRAL */}
|
||||||
|
<div className="relative flex min-w-0 flex-[3] flex-col overflow-hidden rounded-xl border border-slate-200 bg-slate-50/50 shadow-sm">
|
||||||
|
<div className="shrink-0 border-b bg-white p-3">
|
||||||
|
<span className="text-[10px] font-bold tracking-wider text-slate-400 uppercase">
|
||||||
|
Asistente IA
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* CONTENIDO DEL CHAT */}
|
|
||||||
<div className="relative min-h-0 flex-1">
|
<div className="relative min-h-0 flex-1">
|
||||||
<ScrollArea ref={scrollRef} className="h-full w-full">
|
<ScrollArea ref={scrollRef} className="h-full w-full">
|
||||||
<div className="mx-auto max-w-3xl space-y-6 p-6">
|
<div className="mx-auto max-w-3xl space-y-6 p-6">
|
||||||
{messages?.map((msg) => (
|
{messages.length === 0 && !isSending && (
|
||||||
|
<div className="flex h-full flex-col items-center justify-center space-y-4 text-center opacity-60">
|
||||||
|
<div className="rounded-full bg-teal-100 p-4">
|
||||||
|
<Sparkles size={32} className="text-teal-600" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h3 className="font-bold text-slate-700">
|
||||||
|
Nueva Consultoría IA
|
||||||
|
</h3>
|
||||||
|
<p className="max-w-[250px] text-xs text-slate-500">
|
||||||
|
Selecciona campos con ":" o usa una acción rápida para
|
||||||
|
comenzar.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{messages.map((msg) => (
|
||||||
<div
|
<div
|
||||||
key={msg.id}
|
key={msg.id}
|
||||||
className={`flex ${msg.role === 'user' ? 'flex-row-reverse' : 'flex-row'} items-start gap-3`}
|
className={`flex ${msg.role === 'user' ? 'flex-row-reverse' : 'flex-row'} items-start gap-3`}
|
||||||
@@ -297,12 +360,11 @@ export function IAAsignaturaTab({
|
|||||||
{msg.content}
|
{msg.content}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Renderizado de Sugerencias (Homologado con lógica de Asignatura) */}
|
|
||||||
{msg.sugerencia && !msg.sugerencia.aceptada && (
|
{msg.sugerencia && !msg.sugerencia.aceptada && (
|
||||||
<div className="animate-in fade-in slide-in-from-top-1 mt-3 w-full">
|
<div className="animate-in fade-in slide-in-from-top-1 mt-3 w-full">
|
||||||
<div className="rounded-xl border border-teal-100 bg-white p-4 shadow-md">
|
<div className="rounded-xl border border-teal-100 bg-white p-4 shadow-md">
|
||||||
<p className="mb-2 text-[10px] font-bold text-slate-400 uppercase">
|
<p className="mb-2 text-[10px] font-bold text-slate-400 uppercase">
|
||||||
Propuesta para: {msg.sugerencia.campoNombre}
|
Propuesta: {msg.sugerencia.campoNombre}
|
||||||
</p>
|
</p>
|
||||||
<div className="mb-4 max-h-40 overflow-y-auto rounded-lg bg-slate-50 p-3 text-xs text-slate-600 italic">
|
<div className="mb-4 max-h-40 overflow-y-auto rounded-lg bg-slate-50 p-3 text-xs text-slate-600 italic">
|
||||||
{msg.sugerencia.valorSugerido}
|
{msg.sugerencia.valorSugerido}
|
||||||
@@ -310,10 +372,8 @@ export function IAAsignaturaTab({
|
|||||||
<div className="flex gap-2">
|
<div className="flex gap-2">
|
||||||
<Button
|
<Button
|
||||||
size="sm"
|
size="sm"
|
||||||
onClick={() =>
|
onClick={() => onAcceptSuggestion(msg.sugerencia)}
|
||||||
onAcceptSuggestion(msg.sugerencia!)
|
className="h-8 bg-teal-600 hover:bg-teal-700"
|
||||||
}
|
|
||||||
className="h-8 bg-teal-600 text-xs hover:bg-teal-700"
|
|
||||||
>
|
>
|
||||||
<Check size={14} className="mr-1" /> Aplicar
|
<Check size={14} className="mr-1" /> Aplicar
|
||||||
</Button>
|
</Button>
|
||||||
@@ -321,7 +381,7 @@ export function IAAsignaturaTab({
|
|||||||
size="sm"
|
size="sm"
|
||||||
variant="outline"
|
variant="outline"
|
||||||
onClick={() => onRejectSuggestion(msg.id)}
|
onClick={() => onRejectSuggestion(msg.id)}
|
||||||
className="h-8 text-xs"
|
className="h-8"
|
||||||
>
|
>
|
||||||
<X size={14} className="mr-1" /> Descartar
|
<X size={14} className="mr-1" /> Descartar
|
||||||
</Button>
|
</Button>
|
||||||
@@ -329,44 +389,36 @@ export function IAAsignaturaTab({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
{msg.sugerencia?.aceptada && (
|
|
||||||
<Badge className="mt-2 border-teal-200 bg-teal-100 text-teal-700 hover:bg-teal-100">
|
|
||||||
<Check className="mr-1 h-3 w-3" /> Sugerencia aplicada
|
|
||||||
</Badge>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
{isLoading && (
|
{isSending && (
|
||||||
<div className="flex gap-2 p-4">
|
<div className="flex animate-pulse gap-2 p-4">
|
||||||
<div className="h-2 w-2 animate-bounce rounded-full bg-teal-400" />
|
<div className="h-2 w-2 rounded-full bg-teal-400" />
|
||||||
<div className="h-2 w-2 animate-bounce rounded-full bg-teal-400 [animation-delay:0.2s]" />
|
<div className="h-2 w-2 rounded-full bg-teal-400" />
|
||||||
<div className="h-2 w-2 animate-bounce rounded-full bg-teal-400 [animation-delay:0.4s]" />
|
<div className="h-2 w-2 rounded-full bg-teal-400" />
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</ScrollArea>
|
</ScrollArea>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* INPUT FIJO AL FONDO */}
|
{/* INPUT */}
|
||||||
<div className="shrink-0 border-t bg-white p-4">
|
<div className="shrink-0 border-t bg-white p-4">
|
||||||
<div className="relative mx-auto max-w-4xl">
|
<div className="relative mx-auto max-w-4xl">
|
||||||
{/* MENÚ DE SUGERENCIAS FLOTANTE */}
|
|
||||||
{showSuggestions && (
|
{showSuggestions && (
|
||||||
<div className="animate-in slide-in-from-bottom-2 absolute bottom-full z-50 mb-2 w-72 overflow-hidden rounded-xl border bg-white shadow-2xl">
|
<div className="animate-in slide-in-from-bottom-2 absolute bottom-full z-50 mb-2 w-72 overflow-hidden rounded-xl border bg-white shadow-2xl">
|
||||||
<div className="border-b bg-slate-50 px-3 py-2 text-[10px] font-bold tracking-wider text-slate-500 uppercase">
|
<div className="border-b bg-slate-50 px-3 py-2 text-[10px] font-bold text-slate-500 uppercase">
|
||||||
Seleccionar campo de asignatura
|
Campos de Asignatura
|
||||||
</div>
|
</div>
|
||||||
<div className="max-h-64 overflow-y-auto p-1">
|
<div className="max-h-64 overflow-y-auto p-1">
|
||||||
{availableFields.map((field) => (
|
{availableFields.map((field) => (
|
||||||
<button
|
<button
|
||||||
key={field.key}
|
key={field.key}
|
||||||
onClick={() => toggleField(field)}
|
onClick={() => toggleField(field)}
|
||||||
className="group flex w-full items-center justify-between rounded-lg px-3 py-2 text-left text-sm transition-colors hover:bg-teal-50"
|
className="flex w-full items-center justify-between rounded-lg px-3 py-2 text-sm transition-colors hover:bg-teal-50"
|
||||||
>
|
>
|
||||||
<span className="text-slate-700 group-hover:text-teal-700">
|
<span className="text-slate-700">{field.label}</span>
|
||||||
{field.label}
|
|
||||||
</span>
|
|
||||||
{selectedFields.find((f) => f.key === field.key) && (
|
{selectedFields.find((f) => f.key === field.key) && (
|
||||||
<Check size={14} className="text-teal-600" />
|
<Check size={14} className="text-teal-600" />
|
||||||
)}
|
)}
|
||||||
@@ -376,9 +428,7 @@ export function IAAsignaturaTab({
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* CONTENEDOR DEL INPUT */}
|
|
||||||
<div className="flex flex-col gap-2 rounded-xl border bg-slate-50 p-2 transition-all focus-within:bg-white focus-within:ring-1 focus-within:ring-teal-500">
|
<div className="flex flex-col gap-2 rounded-xl border bg-slate-50 p-2 transition-all focus-within:bg-white focus-within:ring-1 focus-within:ring-teal-500">
|
||||||
{/* Visualización de Tags */}
|
|
||||||
{selectedFields.length > 0 && (
|
{selectedFields.length > 0 && (
|
||||||
<div className="flex flex-wrap gap-2 px-2 pt-1">
|
<div className="flex flex-wrap gap-2 px-2 pt-1">
|
||||||
{selectedFields.map((field) => (
|
{selectedFields.map((field) => (
|
||||||
@@ -386,10 +436,10 @@ export function IAAsignaturaTab({
|
|||||||
key={field.key}
|
key={field.key}
|
||||||
className="animate-in zoom-in-95 flex items-center gap-1 rounded-md border border-teal-200 bg-teal-100 px-2 py-0.5 text-[11px] font-semibold text-teal-800"
|
className="animate-in zoom-in-95 flex items-center gap-1 rounded-md border border-teal-200 bg-teal-100 px-2 py-0.5 text-[11px] font-semibold text-teal-800"
|
||||||
>
|
>
|
||||||
<span className="opacity-70">Campo:</span> {field.label}
|
{field.label}
|
||||||
<button
|
<button
|
||||||
onClick={() => toggleField(field)}
|
onClick={() => toggleField(field)}
|
||||||
className="ml-1 rounded-full p-0.5 transition-colors hover:bg-teal-200"
|
className="ml-1 rounded-full p-0.5 hover:bg-teal-200"
|
||||||
>
|
>
|
||||||
<X size={10} />
|
<X size={10} />
|
||||||
</button>
|
</button>
|
||||||
@@ -401,27 +451,28 @@ export function IAAsignaturaTab({
|
|||||||
<div className="flex items-end gap-2">
|
<div className="flex items-end gap-2">
|
||||||
<Textarea
|
<Textarea
|
||||||
value={input}
|
value={input}
|
||||||
onChange={handleInputChange}
|
onChange={(e) => {
|
||||||
|
setInput(e.target.value)
|
||||||
|
if (e.target.value.endsWith(':')) setShowSuggestions(true)
|
||||||
|
else if (showSuggestions && !e.target.value.includes(':'))
|
||||||
|
setShowSuggestions(false)
|
||||||
|
}}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key === 'Enter' && !e.shiftKey) {
|
if (e.key === 'Enter' && !e.shiftKey) {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
handleSend()
|
handleSend()
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
placeholder={
|
placeholder='Escribe ":" para referenciar un campo...'
|
||||||
selectedFields.length > 0
|
className="max-h-[120px] min-h-[40px] flex-1 resize-none border-none bg-transparent text-sm shadow-none focus-visible:ring-0"
|
||||||
? 'Instrucciones para los campos seleccionados...'
|
|
||||||
: 'Escribe tu solicitud o ":" para campos...'
|
|
||||||
}
|
|
||||||
className="max-h-[120px] min-h-[40px] flex-1 resize-none border-none bg-transparent py-2 text-sm shadow-none focus-visible:ring-0"
|
|
||||||
/>
|
/>
|
||||||
<Button
|
<Button
|
||||||
onClick={() => handleSend()}
|
onClick={() => handleSend()}
|
||||||
disabled={
|
disabled={
|
||||||
(!input.trim() && selectedFields.length === 0) || isLoading
|
(!input.trim() && selectedFields.length === 0) || isSending
|
||||||
}
|
}
|
||||||
size="icon"
|
size="icon"
|
||||||
className="mb-1 h-9 w-9 shrink-0 bg-teal-600 hover:bg-teal-700"
|
className="h-9 w-9 bg-teal-600 hover:bg-teal-700"
|
||||||
>
|
>
|
||||||
<Send size={16} className="text-white" />
|
<Send size={16} className="text-white" />
|
||||||
</Button>
|
</Button>
|
||||||
@@ -431,24 +482,22 @@ export function IAAsignaturaTab({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* PANEL LATERAL (ACCIONES RÁPIDAS) */}
|
{/* PANEL DERECHO ACCIONES */}
|
||||||
<div className="flex flex-[1] flex-col gap-4 overflow-y-auto pr-2">
|
<div className="flex flex-[1] flex-col gap-4 overflow-y-auto pr-2">
|
||||||
<h4 className="flex items-center gap-2 text-left text-sm font-bold text-slate-800">
|
<h4 className="flex items-center gap-2 text-sm font-bold text-slate-800">
|
||||||
<Lightbulb size={18} className="text-orange-500" /> Acciones rápidas
|
<Lightbulb size={18} className="text-orange-500" /> Atajos
|
||||||
</h4>
|
</h4>
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
{PRESETS.map((preset) => (
|
{PRESETS.map((preset) => (
|
||||||
<button
|
<button
|
||||||
key={preset.id}
|
key={preset.id}
|
||||||
onClick={() => handleSend(preset.prompt)}
|
onClick={() => handleSend(preset.prompt)}
|
||||||
className="group flex w-full items-center gap-3 rounded-xl border bg-white p-3 text-left text-sm shadow-sm transition-all hover:border-teal-500 hover:bg-teal-50"
|
className="group flex w-full items-center gap-3 rounded-xl border bg-white p-3 text-left text-sm transition-all hover:border-teal-500 hover:bg-teal-50"
|
||||||
>
|
>
|
||||||
<div className="rounded-lg bg-slate-100 p-2 text-slate-500 group-hover:bg-teal-100 group-hover:text-teal-600">
|
<div className="rounded-lg bg-slate-100 p-2 group-hover:bg-teal-100 group-hover:text-teal-600">
|
||||||
<preset.icon size={16} />
|
<preset.icon size={16} />
|
||||||
</div>
|
</div>
|
||||||
<span className="leading-tight font-medium text-slate-700">
|
<span className="font-medium text-slate-700">{preset.label}</span>
|
||||||
{preset.label}
|
|
||||||
</span>
|
|
||||||
</button>
|
</button>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user