import { Box, Text } from '@hermes/ink' import { memo } from 'react' import { LONG_MSG, ROLE } from '../constants.js' import { compactPreview, hasAnsi, isPasteBackedText, stripAnsi, userDisplay } from '../lib/text.js' import type { Theme } from '../theme.js' import type { Msg } from '../types.js' import { Md } from './markdown.js' import { ToolTrail } from './thinking.js' export const MessageLine = memo(function MessageLine({ cols, compact, msg, t }: { cols: number compact?: boolean msg: Msg t: Theme }) { if (msg.role === 'tool') { const preview = compactPreview(hasAnsi(msg.text) ? stripAnsi(msg.text) : msg.text, Math.max(24, cols - 14)) return ( {preview || '(empty tool result)'} ) } const { body, glyph, prefix } = ROLE[msg.role](t) const content = (() => { if (msg.kind === 'slash') { return {msg.text} } if (msg.role !== 'user' && hasAnsi(msg.text)) { return {msg.text} } if (msg.role === 'assistant') { return } if (msg.role === 'user' && msg.text.length > LONG_MSG && isPasteBackedText(msg.text)) { const [head, ...rest] = userDisplay(msg.text).split('[long message]') return ( {head} [long message] {rest.join('')} ) } return {msg.text} })() return ( {msg.thinking && ( 💭 {msg.thinking.replace(/\n/g, ' ').slice(0, 200)} )} {msg.tools?.length ? ( ) : null} {glyph}{' '} {content} ) })