| import { marked } from "marked"; | |
| import DOMPurify from "dompurify"; | |
| import BotIcon from "./icons/BotIcon"; | |
| import UserIcon from "./icons/UserIcon"; | |
| import "./Chat.css"; | |
| import { useEffect } from "react"; | |
| function render(text) { | |
| return DOMPurify.sanitize(marked.parse(text)); | |
| } | |
| export default function Chat({ messages }) { | |
| const empty = messages.length === 0; | |
| useEffect(() => { | |
| window.MathJax.typeset(); | |
| }, [messages]); | |
| return ( | |
| <div | |
| className={`flex-1 p-6 max-w-[960px] w-full ${empty ? "flex flex-col items-center justify-end" : "space-y-4"}`} | |
| > | |
| {empty ? ( | |
| <div className="text-xl">Ready!</div> | |
| ) : ( | |
| messages.map((msg, i) => ( | |
| <div key={`message-${i}`} className="flex items-start space-x-4"> | |
| {msg.role === "assistant" ? ( | |
| <> | |
| <BotIcon className="h-6 w-6 min-h-6 min-w-6 my-3 text-gray-500 dark:text-gray-300" /> | |
| <div className="bg-gray-200 dark:bg-gray-700 rounded-lg p-4"> | |
| <p className="min-h-6 text-gray-800 dark:text-gray-200 overflow-wrap-anywhere"> | |
| {msg.content.length > 0 ? ( | |
| <span | |
| className="markdown" | |
| dangerouslySetInnerHTML={{ | |
| __html: render(msg.content), | |
| }} | |
| /> | |
| ) : ( | |
| <span className="h-6 flex items-center gap-1"> | |
| <span className="w-2.5 h-2.5 bg-gray-600 dark:bg-gray-300 rounded-full animate-pulse"></span> | |
| <span className="w-2.5 h-2.5 bg-gray-600 dark:bg-gray-300 rounded-full animate-pulse animation-delay-200"></span> | |
| <span className="w-2.5 h-2.5 bg-gray-600 dark:bg-gray-300 rounded-full animate-pulse animation-delay-400"></span> | |
| </span> | |
| )} | |
| </p> | |
| </div> | |
| </> | |
| ) : ( | |
| <> | |
| <UserIcon className="h-6 w-6 min-h-6 min-w-6 my-3 text-gray-500 dark:text-gray-300" /> | |
| <div className="bg-blue-500 text-white rounded-lg p-4"> | |
| <p className="min-h-6 overflow-wrap-anywhere"> | |
| {msg.content} | |
| </p> | |
| </div> | |
| </> | |
| )} | |
| </div> | |
| )) | |
| )} | |
| </div> | |
| ); | |
| } | |