import { useEffect, useMemo, useState } from 'react'; import { CallbackGeneratedChunk, useAppContext } from '../utils/app.context'; import ChatMessage from './ChatMessage'; import { CanvasType, Message, PendingMessage } from '../utils/types'; import { classNames, throttle } from '../utils/misc'; import CanvasPyInterpreter from './CanvasPyInterpreter'; import StorageUtils from '../utils/storage'; /** * A message display is a message node with additional information for rendering. * For example, siblings of the message node are stored as their last node (aka leaf node). */ export interface MessageDisplay { msg: Message | PendingMessage; siblingLeafNodeIds: Message['id'][]; siblingCurrIdx: number; isPending?: boolean; } function getListMessageDisplay( msgs: Readonly, leafNodeId: Message['id'] ): MessageDisplay[] { const currNodes = StorageUtils.filterByLeafNodeId(msgs, leafNodeId, true); const res: MessageDisplay[] = []; const nodeMap = new Map(); for (const msg of msgs) { nodeMap.set(msg.id, msg); } // find leaf node from a message node const findLeafNode = (msgId: Message['id']): Message['id'] => { let currNode: Message | undefined = nodeMap.get(msgId); while (currNode) { if (currNode.children.length === 0) break; currNode = nodeMap.get(currNode.children.at(-1) ?? -1); } return currNode?.id ?? -1; }; // traverse the current nodes for (const msg of currNodes) { const parentNode = nodeMap.get(msg.parent ?? -1); if (!parentNode) continue; const siblings = parentNode.children; if (msg.type !== 'root') { res.push({ msg, siblingLeafNodeIds: siblings.map(findLeafNode), siblingCurrIdx: siblings.indexOf(msg.id), }); } } return res; } const scrollToBottom = throttle( (requiresNearBottom: boolean, delay: number = 80) => { const mainScrollElem = document.getElementById('main-scroll'); if (!mainScrollElem) return; const spaceToBottom = mainScrollElem.scrollHeight - mainScrollElem.scrollTop - mainScrollElem.clientHeight; if (!requiresNearBottom || spaceToBottom < 50) { setTimeout( () => mainScrollElem.scrollTo({ top: mainScrollElem.scrollHeight }), delay ); } }, 80 ); export default function ChatScreen() { const { viewingChat, sendMessage, isGenerating, stopGenerating, pendingMessages, canvasData, replaceMessageAndGenerate, } = useAppContext(); const [inputMsg, setInputMsg] = useState(''); // keep track of leaf node for rendering const [currNodeId, setCurrNodeId] = useState(-1); const messages: MessageDisplay[] = useMemo(() => { if (!viewingChat) return []; else return getListMessageDisplay(viewingChat.messages, currNodeId); }, [currNodeId, viewingChat]); const currConvId = viewingChat?.conv.id ?? null; const pendingMsg: PendingMessage | undefined = pendingMessages[currConvId ?? '']; useEffect(() => { // reset to latest node when conversation changes setCurrNodeId(-1); // scroll to bottom when conversation changes scrollToBottom(false, 1); }, [currConvId]); const onChunk: CallbackGeneratedChunk = (currLeafNodeId?: Message['id']) => { if (currLeafNodeId) { setCurrNodeId(currLeafNodeId); } scrollToBottom(true); }; const sendNewMessage = async () => { if (inputMsg.trim().length === 0 || isGenerating(currConvId ?? '')) return; const lastInpMsg = inputMsg; setInputMsg(''); scrollToBottom(false); setCurrNodeId(-1); // get the last message node const lastMsgNodeId = messages.at(-1)?.msg.id ?? null; if (!(await sendMessage(currConvId, lastMsgNodeId, inputMsg, onChunk))) { // restore the input message if failed setInputMsg(lastInpMsg); } }; const handleEditMessage = async (msg: Message, content: string) => { if (!viewingChat) return; setCurrNodeId(msg.id); scrollToBottom(false); await replaceMessageAndGenerate( viewingChat.conv.id, msg.parent, content, onChunk ); setCurrNodeId(-1); scrollToBottom(false); }; const handleRegenerateMessage = async (msg: Message) => { if (!viewingChat) return; setCurrNodeId(msg.parent); scrollToBottom(false); await replaceMessageAndGenerate( viewingChat.conv.id, msg.parent, null, onChunk ); setCurrNodeId(-1); scrollToBottom(false); }; const hasCanvas = !!canvasData; // due to some timing issues of StorageUtils.appendMsg(), we need to make sure the pendingMsg is not duplicated upon rendering (i.e. appears once in the saved conversation and once in the pendingMsg) const pendingMsgDisplay: MessageDisplay[] = pendingMsg && messages.at(-1)?.msg.id !== pendingMsg.id ? [ { msg: pendingMsg, siblingLeafNodeIds: [], siblingCurrIdx: 0, isPending: true, }, ] : []; return (
{/* chat messages */}
{/* placeholder to shift the message to the bottom */} {viewingChat ? '' : 'Send a message to start'}
{[...messages, ...pendingMsgDisplay].map((msg) => ( ))}
{/* chat input */}
{isGenerating(currConvId ?? '') ? ( ) : ( )}
{canvasData?.type === CanvasType.PY_INTERPRETER && ( )}
); }