/* ┌──────────────────────────────────────────────────────────────────────────────┐ │ @author: Davidson Gomes │ │ @file: /app/chat/components/InlineDataAttachments.tsx │ │ Developed by: Davidson Gomes │ │ Creation date: August 29, 2025 │ │ Contact: contato@evolution-api.com │ ├──────────────────────────────────────────────────────────────────────────────┤ │ @copyright © Evolution API 2025. All rights reserved. │ │ Licensed under the Apache License, Version 2.0 │ │ │ │ You may not use this file except in compliance with the License. │ │ You may obtain a copy of the License at │ │ │ │ http://www.apache.org/licenses/LICENSE-2.0 │ │ │ │ Unless required by applicable law or agreed to in writing, software │ │ distributed under the License is distributed on an "AS IS" BASIS, │ │ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. │ │ See the License for the specific language governing permissions and │ │ limitations under the License. │ ├──────────────────────────────────────────────────────────────────────────────┤ │ @important │ │ For any future changes to the code in this file, it is recommended to │ │ include, together with the modification, the information of the developer │ │ who changed it and the date of modification. │ └──────────────────────────────────────────────────────────────────────────────┘ */ "use client"; import React, { useState, useEffect } from "react"; import { formatFileSize, isImageFile } from "@/lib/file-utils"; import { File, FileText, Download, Image } from "lucide-react"; import { ChatPart } from "@/services/sessionService"; interface InlineDataAttachmentsProps { parts: ChatPart[]; className?: string; sessionId?: string; } interface ProcessedFile { filename: string; content_type: string; data: string; size: number; preview_url?: string; } export function InlineDataAttachments({ parts, className = "", sessionId }: InlineDataAttachmentsProps) { const [processedFiles, setProcessedFiles] = useState([]); const [isProcessed, setIsProcessed] = useState(false); useEffect(() => { if (isProcessed) return; const validParts = parts.filter(part => part.inline_data && part.inline_data.data); if (validParts.length === 0) { setIsProcessed(true); return; } const files = validParts.map((part, index) => { const { mime_type, data } = part.inline_data!; const extension = mime_type.split('/')[1] || 'file'; let filename = ''; if (part.inline_data?.metadata?.filename) { filename = part.inline_data.metadata.filename; } else if (part.file_data?.filename) { filename = part.file_data.filename; } else { filename = `media_${index + 1}.${extension}`; } let preview_url = undefined; if (data && isImageFile(mime_type)) { preview_url = data.startsWith('data:') ? data : `data:${mime_type};base64,${data}`; } const fileData: ProcessedFile = { filename, content_type: mime_type, size: data.length, data, preview_url }; return fileData; }); setProcessedFiles(files); setIsProcessed(true); }, [parts, isProcessed]); if (processedFiles.length === 0) return null; const downloadFile = (file: ProcessedFile) => { try { const link = document.createElement("a"); const dataUrl = file.data.startsWith('data:') ? file.data : `data:${file.content_type};base64,${file.data}`; link.href = dataUrl; link.download = file.filename; document.body.appendChild(link); link.click(); document.body.removeChild(link); } catch (error) { console.error(`Error downloading file ${file.filename}:`, error); } }; const getFileUrl = (file: ProcessedFile) => { return file.preview_url || (file.data.startsWith('data:') ? file.data : `data:${file.content_type};base64,${file.data}`); }; return (
Attached files:
{processedFiles.map((file, index) => (
{isImageFile(file.content_type) && (
{file.filename} { console.error(`Error loading image ${file.filename}`); (e.target as HTMLImageElement).src = ""; }} />
)}
{isImageFile(file.content_type) ? ( ) : file.content_type === "application/pdf" ? ( ) : ( )}
{file.filename}
{formatFileSize(file.size)}
))}
); }