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}
)
})