import { useMemo, useState } from 'react'; import { MarkdownViewer } from '@renderer/components/chat/viewers/MarkdownViewer'; import { CARD_BG, CARD_BORDER_STYLE, CARD_ICON_MUTED, CARD_TEXT_LIGHT, } from '@renderer/constants/cssVariables'; import { getTeamColorSet } from '@renderer/constants/teamColors'; import { getMessageTypeLabel, getStructuredMessageSummary, parseMessageReply, parseStructuredAgentMessage, } from '@renderer/utils/agentMessageFormatting'; import { formatAgentRole } from '@renderer/utils/formatAgentRole'; import { createAgentBlockRegex } from '@shared/constants/agentBlocks'; import { Bot, ChevronRight, ListPlus, MessageSquare, Reply } from 'lucide-react'; import { ReplyQuoteBlock } from './ReplyQuoteBlock'; import type { TeamColorSet } from '@renderer/constants/teamColors'; import type { InboxMessage } from '@shared/types'; type StructuredMessage = Record; interface ActivityItemProps { message: InboxMessage; memberRole?: string; memberColor?: string; recipientColor?: string; onMemberNameClick?: (memberName: string) => void; onCreateTask?: (subject: string, description: string) => void; onReply?: (message: InboxMessage) => void; } function getStringField(obj: StructuredMessage, key: string): string | null { const value = obj[key]; return typeof value === 'string' && value.trim() !== '' ? value : null; } function getNoiseLabel(parsed: StructuredMessage): string | null { const type = getStringField(parsed, 'type'); if (type === 'idle_notification') { const reason = getStringField(parsed, 'idleReason'); return reason ? `Idle (${reason})` : 'Idle'; } if (type === 'shutdown_response') { return parsed.approve === true ? 'Shut down' : 'Rejected shutdown'; } if (type === 'shutdown_request') { return 'Shutdown requested'; } if (type === 'shutdown_approved' || type === 'teammate_terminated') { return type === 'shutdown_approved' ? 'Shutdown confirmed' : 'Terminated'; } if (type === 'task_completed') { const rawTaskId = parsed.taskId; const taskId = typeof rawTaskId === 'string' || typeof rawTaskId === 'number' ? rawTaskId : null; return taskId !== null ? `Completed task #${taskId}` : 'Completed a task'; } return null; } // --------------------------------------------------------------------------- // Compact noise row (idle, shutdown, terminated) — minimal dot + name + label // --------------------------------------------------------------------------- const NoiseRow = ({ name, label, colors, }: { name: string; label: string; colors: TeamColorSet; }): React.JSX.Element => (
{name} {label}
); // --------------------------------------------------------------------------- // Detect system/automated messages that should be collapsed by default. // These are generated by teamctl.js and contain tool instructions, not // human-written content, so showing them expanded adds visual noise. // --------------------------------------------------------------------------- const SYSTEM_MESSAGE_PATTERNS: { pattern: RegExp; label: string }[] = [ { pattern: /^New task assigned to you:/, label: 'Task assignment' }, { pattern: /^Task #\d+\s+approved/, label: 'Task approved' }, { pattern: /^Task #\d+\s+needs fixes/, label: 'Review changes requested' }, ]; function getSystemMessageLabel(text: string): string | null { for (const { pattern, label } of SYSTEM_MESSAGE_PATTERNS) { if (pattern.test(text)) return label; } return null; } /** Strip ```info_for_agent ... ``` blocks from text for UI display. */ function stripAgentBlocks(text: string): string { return text.replace(createAgentBlockRegex(), '').trim(); } // --------------------------------------------------------------------------- // Full message card — left colored border, name badge, collapsible content // --------------------------------------------------------------------------- export const ActivityItem = ({ message, memberRole, memberColor, recipientColor, onMemberNameClick, onCreateTask, onReply, }: ActivityItemProps): React.JSX.Element => { const colors = getTeamColorSet(memberColor ?? message.color ?? ''); const recipientColors = message.to && recipientColor ? getTeamColorSet(recipientColor) : null; const formattedRole = formatAgentRole(memberRole); const timestamp = Number.isNaN(Date.parse(message.timestamp)) ? message.timestamp : new Date(message.timestamp).toLocaleString(); const structured = parseStructuredAgentMessage(message.text); const noiseLabel = structured ? getNoiseLabel(structured) : null; // System/automated messages start collapsed const systemLabel = !structured ? getSystemMessageLabel(message.text) : null; const [isExpanded, setIsExpanded] = useState(!systemLabel); // Strip agent-only blocks from displayed text const displayText = useMemo( () => (structured ? null : stripAgentBlocks(message.text)), [structured, message.text] ); // Check if this is a reply message const parsedReply = useMemo( () => (displayText ? parseMessageReply(displayText) : null), [displayText] ); // Noise messages: minimal inline row if (noiseLabel) { return ; } const messageType = structured && typeof structured.type === 'string' ? getMessageTypeLabel(structured.type) : null; const autoSummary = structured ? getStructuredMessageSummary(structured) : null; const handleCreateTask = (): void => { const subject = message.summary || autoSummary || `Task from ${message.from}`; const plainText = structured ? JSON.stringify(structured, null, 2) : message.text; const description = `From: ${message.from}\nAt: ${timestamp}\n\n${plainText}`.slice(0, 2000); onCreateTask?.(subject, description); }; const summaryText = message.summary || autoSummary || ''; const HeaderTag = systemLabel ? 'button' : 'div'; return (
{/* Header — clickable when system message to toggle expand */} setIsExpanded((v) => !v) : undefined} onKeyDown={ systemLabel ? (e) => { if (e.key === 'Enter' || e.key === ' ') { e.preventDefault(); setIsExpanded((v) => !v); } } : undefined } > {/* Chevron for collapsible system messages */} {systemLabel ? ( ) : null} {message.source === 'lead_session' || message.source === 'lead_process' ? ( ) : ( )} {/* Name badge — clickable to open member popup */} {onMemberNameClick ? ( ) : ( {message.from} )} {/* Role */} {formattedRole ? ( {formattedRole} ) : null} {/* Message type label or system label */} {systemLabel ? ( {systemLabel} ) : messageType ? ( {messageType} ) : null} {/* Lead session marker */} {message.source === 'lead_session' ? ( session ) : message.source === 'lead_process' ? ( live ) : null} {/* Recipient — badge like sender, clickable to open member popup */} {message.to && message.to !== message.from && recipientColors ? ( {onMemberNameClick ? ( ) : ( {message.to} )} ) : message.to && message.to !== message.from ? ( {onMemberNameClick ? ( ) : ( {message.to} )} ) : null} {/* Summary */} {summaryText} {/* Timestamp + reply + create task */}
{onReply && ( )} {onCreateTask && ( )} {timestamp}
{/* Content — collapsed for system messages, expanded for others */} {isExpanded ? (
{structured ? (
{autoSummary && autoSummary !== messageType ? (

{autoSummary}

) : null}
Raw JSON
                  {JSON.stringify(structured, null, 2)}
                
) : parsedReply ? ( ) : ( )}
) : null}
); };