/ src / components / MessageRow.tsx
MessageRow.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import * as React from 'react';
  3  import type { Command } from '../commands.js';
  4  import { Box } from '../ink.js';
  5  import type { Screen } from '../screens/REPL.js';
  6  import type { Tools } from '../Tool.js';
  7  import type { RenderableMessage } from '../types/message.js';
  8  import { getDisplayMessageFromCollapsed, getToolSearchOrReadInfo, getToolUseIdsFromCollapsedGroup, hasAnyToolInProgress } from '../utils/collapseReadSearch.js';
  9  import { type buildMessageLookups, EMPTY_STRING_SET, getProgressMessagesFromLookup, getSiblingToolUseIDsFromLookup, getToolUseID } from '../utils/messages.js';
 10  import { hasThinkingContent, Message } from './Message.js';
 11  import { MessageModel } from './MessageModel.js';
 12  import { shouldRenderStatically } from './Messages.js';
 13  import { MessageTimestamp } from './MessageTimestamp.js';
 14  import { OffscreenFreeze } from './OffscreenFreeze.js';
 15  export type Props = {
 16    message: RenderableMessage;
 17    /** Whether the previous message in renderableMessages is also a user message. */
 18    isUserContinuation: boolean;
 19    /**
 20     * Whether there is non-skippable content after this message in renderableMessages.
 21     * Only needs to be accurate for `collapsed_read_search` messages — used to decide
 22     * if the collapsed group spinner should stay active. Pass `false` otherwise.
 23     */
 24    hasContentAfter: boolean;
 25    tools: Tools;
 26    commands: Command[];
 27    verbose: boolean;
 28    inProgressToolUseIDs: Set<string>;
 29    streamingToolUseIDs: Set<string>;
 30    screen: Screen;
 31    canAnimate: boolean;
 32    onOpenRateLimitOptions?: () => void;
 33    lastThinkingBlockId: string | null;
 34    latestBashOutputUUID: string | null;
 35    columns: number;
 36    isLoading: boolean;
 37    lookups: ReturnType<typeof buildMessageLookups>;
 38  };
 39  
 40  /**
 41   * Scans forward from `index+1` to check if any "real" content follows. Used to
 42   * decide whether a collapsed read/search group should stay in its active
 43   * (grey dot, present-tense "Reading…") state while the query is still loading.
 44   *
 45   * Exported so Messages.tsx can compute this once per message and pass the
 46   * result as a boolean prop — avoids passing the full `renderableMessages` array
 47   * to each MessageRow (which React Compiler would pin in the fiber's memoCache,
 48   * accumulating every historical version of the array ≈ 1-2MB over a 7-turn session).
 49   */
 50  export function hasContentAfterIndex(messages: RenderableMessage[], index: number, tools: Tools, streamingToolUseIDs: Set<string>): boolean {
 51    for (let i = index + 1; i < messages.length; i++) {
 52      const msg = messages[i];
 53      if (msg?.type === 'assistant') {
 54        const content = msg.message.content[0];
 55        if (content?.type === 'thinking' || content?.type === 'redacted_thinking') {
 56          continue;
 57        }
 58        if (content?.type === 'tool_use') {
 59          if (getToolSearchOrReadInfo(content.name, content.input, tools).isCollapsible) {
 60            continue;
 61          }
 62          // Non-collapsible tool uses appear in syntheticStreamingToolUseMessages
 63          // before their ID is added to inProgressToolUseIDs. Skip while streaming
 64          // to avoid briefly finalizing the read group.
 65          if (streamingToolUseIDs.has(content.id)) {
 66            continue;
 67          }
 68        }
 69        return true;
 70      }
 71      if (msg?.type === 'system' || msg?.type === 'attachment') {
 72        continue;
 73      }
 74      // Tool results arrive while the collapsed group is still being built
 75      if (msg?.type === 'user') {
 76        const content = msg.message.content[0];
 77        if (content?.type === 'tool_result') {
 78          continue;
 79        }
 80      }
 81      // Collapsible grouped_tool_use messages arrive transiently before being
 82      // merged into the current collapsed group on the next render cycle
 83      if (msg?.type === 'grouped_tool_use') {
 84        const firstInput = msg.messages[0]?.message.content[0]?.input;
 85        if (getToolSearchOrReadInfo(msg.toolName, firstInput, tools).isCollapsible) {
 86          continue;
 87        }
 88      }
 89      return true;
 90    }
 91    return false;
 92  }
 93  function MessageRowImpl(t0) {
 94    const $ = _c(64);
 95    const {
 96      message: msg,
 97      isUserContinuation,
 98      hasContentAfter,
 99      tools,
100      commands,
101      verbose,
102      inProgressToolUseIDs,
103      streamingToolUseIDs,
104      screen,
105      canAnimate,
106      onOpenRateLimitOptions,
107      lastThinkingBlockId,
108      latestBashOutputUUID,
109      columns,
110      isLoading,
111      lookups
112    } = t0;
113    const isTranscriptMode = screen === "transcript";
114    const isGrouped = msg.type === "grouped_tool_use";
115    const isCollapsed = msg.type === "collapsed_read_search";
116    let t1;
117    if ($[0] !== hasContentAfter || $[1] !== inProgressToolUseIDs || $[2] !== isCollapsed || $[3] !== isLoading || $[4] !== msg) {
118      t1 = isCollapsed && (hasAnyToolInProgress(msg, inProgressToolUseIDs) || isLoading && !hasContentAfter);
119      $[0] = hasContentAfter;
120      $[1] = inProgressToolUseIDs;
121      $[2] = isCollapsed;
122      $[3] = isLoading;
123      $[4] = msg;
124      $[5] = t1;
125    } else {
126      t1 = $[5];
127    }
128    const isActiveCollapsedGroup = t1;
129    let t2;
130    if ($[6] !== isCollapsed || $[7] !== isGrouped || $[8] !== msg) {
131      t2 = isGrouped ? msg.displayMessage : isCollapsed ? getDisplayMessageFromCollapsed(msg) : msg;
132      $[6] = isCollapsed;
133      $[7] = isGrouped;
134      $[8] = msg;
135      $[9] = t2;
136    } else {
137      t2 = $[9];
138    }
139    const displayMsg = t2;
140    let t3;
141    if ($[10] !== isCollapsed || $[11] !== isGrouped || $[12] !== lookups || $[13] !== msg) {
142      t3 = isGrouped || isCollapsed ? [] : getProgressMessagesFromLookup(msg, lookups);
143      $[10] = isCollapsed;
144      $[11] = isGrouped;
145      $[12] = lookups;
146      $[13] = msg;
147      $[14] = t3;
148    } else {
149      t3 = $[14];
150    }
151    const progressMessagesForMessage = t3;
152    let t4;
153    if ($[15] !== inProgressToolUseIDs || $[16] !== isCollapsed || $[17] !== isGrouped || $[18] !== lookups || $[19] !== msg || $[20] !== screen || $[21] !== streamingToolUseIDs) {
154      const siblingToolUseIDs = isGrouped || isCollapsed ? EMPTY_STRING_SET : getSiblingToolUseIDsFromLookup(msg, lookups);
155      t4 = shouldRenderStatically(msg, streamingToolUseIDs, inProgressToolUseIDs, siblingToolUseIDs, screen, lookups);
156      $[15] = inProgressToolUseIDs;
157      $[16] = isCollapsed;
158      $[17] = isGrouped;
159      $[18] = lookups;
160      $[19] = msg;
161      $[20] = screen;
162      $[21] = streamingToolUseIDs;
163      $[22] = t4;
164    } else {
165      t4 = $[22];
166    }
167    const isStatic = t4;
168    let shouldAnimate = false;
169    if (canAnimate) {
170      if (isGrouped) {
171        let t5;
172        if ($[23] !== inProgressToolUseIDs || $[24] !== msg.messages) {
173          let t6;
174          if ($[26] !== inProgressToolUseIDs) {
175            t6 = m => {
176              const content = m.message.content[0];
177              return content?.type === "tool_use" && inProgressToolUseIDs.has(content.id);
178            };
179            $[26] = inProgressToolUseIDs;
180            $[27] = t6;
181          } else {
182            t6 = $[27];
183          }
184          t5 = msg.messages.some(t6);
185          $[23] = inProgressToolUseIDs;
186          $[24] = msg.messages;
187          $[25] = t5;
188        } else {
189          t5 = $[25];
190        }
191        shouldAnimate = t5;
192      } else {
193        if (isCollapsed) {
194          let t5;
195          if ($[28] !== inProgressToolUseIDs || $[29] !== msg) {
196            t5 = hasAnyToolInProgress(msg, inProgressToolUseIDs);
197            $[28] = inProgressToolUseIDs;
198            $[29] = msg;
199            $[30] = t5;
200          } else {
201            t5 = $[30];
202          }
203          shouldAnimate = t5;
204        } else {
205          let t5;
206          if ($[31] !== inProgressToolUseIDs || $[32] !== msg) {
207            const toolUseID = getToolUseID(msg);
208            t5 = !toolUseID || inProgressToolUseIDs.has(toolUseID);
209            $[31] = inProgressToolUseIDs;
210            $[32] = msg;
211            $[33] = t5;
212          } else {
213            t5 = $[33];
214          }
215          shouldAnimate = t5;
216        }
217      }
218    }
219    let t5;
220    if ($[34] !== displayMsg || $[35] !== isTranscriptMode) {
221      t5 = isTranscriptMode && displayMsg.type === "assistant" && displayMsg.message.content.some(_temp) && (displayMsg.timestamp || displayMsg.message.model);
222      $[34] = displayMsg;
223      $[35] = isTranscriptMode;
224      $[36] = t5;
225    } else {
226      t5 = $[36];
227    }
228    const hasMetadata = t5;
229    const t6 = !hasMetadata;
230    const t7 = hasMetadata ? undefined : columns;
231    let t8;
232    if ($[37] !== commands || $[38] !== inProgressToolUseIDs || $[39] !== isActiveCollapsedGroup || $[40] !== isStatic || $[41] !== isTranscriptMode || $[42] !== isUserContinuation || $[43] !== lastThinkingBlockId || $[44] !== latestBashOutputUUID || $[45] !== lookups || $[46] !== msg || $[47] !== onOpenRateLimitOptions || $[48] !== progressMessagesForMessage || $[49] !== shouldAnimate || $[50] !== t6 || $[51] !== t7 || $[52] !== tools || $[53] !== verbose) {
233      t8 = <Message message={msg} lookups={lookups} addMargin={t6} containerWidth={t7} tools={tools} commands={commands} verbose={verbose} inProgressToolUseIDs={inProgressToolUseIDs} progressMessagesForMessage={progressMessagesForMessage} shouldAnimate={shouldAnimate} shouldShowDot={true} isTranscriptMode={isTranscriptMode} isStatic={isStatic} onOpenRateLimitOptions={onOpenRateLimitOptions} isActiveCollapsedGroup={isActiveCollapsedGroup} isUserContinuation={isUserContinuation} lastThinkingBlockId={lastThinkingBlockId} latestBashOutputUUID={latestBashOutputUUID} />;
234      $[37] = commands;
235      $[38] = inProgressToolUseIDs;
236      $[39] = isActiveCollapsedGroup;
237      $[40] = isStatic;
238      $[41] = isTranscriptMode;
239      $[42] = isUserContinuation;
240      $[43] = lastThinkingBlockId;
241      $[44] = latestBashOutputUUID;
242      $[45] = lookups;
243      $[46] = msg;
244      $[47] = onOpenRateLimitOptions;
245      $[48] = progressMessagesForMessage;
246      $[49] = shouldAnimate;
247      $[50] = t6;
248      $[51] = t7;
249      $[52] = tools;
250      $[53] = verbose;
251      $[54] = t8;
252    } else {
253      t8 = $[54];
254    }
255    const messageEl = t8;
256    if (!hasMetadata) {
257      let t9;
258      if ($[55] !== messageEl) {
259        t9 = <OffscreenFreeze>{messageEl}</OffscreenFreeze>;
260        $[55] = messageEl;
261        $[56] = t9;
262      } else {
263        t9 = $[56];
264      }
265      return t9;
266    }
267    let t9;
268    if ($[57] !== displayMsg || $[58] !== isTranscriptMode) {
269      t9 = <Box flexDirection="row" justifyContent="flex-end" gap={1} marginTop={1}><MessageTimestamp message={displayMsg} isTranscriptMode={isTranscriptMode} /><MessageModel message={displayMsg} isTranscriptMode={isTranscriptMode} /></Box>;
270      $[57] = displayMsg;
271      $[58] = isTranscriptMode;
272      $[59] = t9;
273    } else {
274      t9 = $[59];
275    }
276    let t10;
277    if ($[60] !== columns || $[61] !== messageEl || $[62] !== t9) {
278      t10 = <OffscreenFreeze><Box width={columns} flexDirection="column">{t9}{messageEl}</Box></OffscreenFreeze>;
279      $[60] = columns;
280      $[61] = messageEl;
281      $[62] = t9;
282      $[63] = t10;
283    } else {
284      t10 = $[63];
285    }
286    return t10;
287  }
288  
289  /**
290   * Checks if a message is "streaming" - i.e., its content may still be changing.
291   * Exported for testing.
292   */
293  function _temp(c) {
294    return c.type === "text";
295  }
296  export function isMessageStreaming(msg: RenderableMessage, streamingToolUseIDs: Set<string>): boolean {
297    if (msg.type === 'grouped_tool_use') {
298      return msg.messages.some(m => {
299        const content = m.message.content[0];
300        return content?.type === 'tool_use' && streamingToolUseIDs.has(content.id);
301      });
302    }
303    if (msg.type === 'collapsed_read_search') {
304      const toolIds = getToolUseIdsFromCollapsedGroup(msg);
305      return toolIds.some(id => streamingToolUseIDs.has(id));
306    }
307    const toolUseID = getToolUseID(msg);
308    return !!toolUseID && streamingToolUseIDs.has(toolUseID);
309  }
310  
311  /**
312   * Checks if all tools in a message are resolved.
313   * Exported for testing.
314   */
315  export function allToolsResolved(msg: RenderableMessage, resolvedToolUseIDs: Set<string>): boolean {
316    if (msg.type === 'grouped_tool_use') {
317      return msg.messages.every(m => {
318        const content = m.message.content[0];
319        return content?.type === 'tool_use' && resolvedToolUseIDs.has(content.id);
320      });
321    }
322    if (msg.type === 'collapsed_read_search') {
323      const toolIds = getToolUseIdsFromCollapsedGroup(msg);
324      return toolIds.every(id => resolvedToolUseIDs.has(id));
325    }
326    if (msg.type === 'assistant') {
327      const block = msg.message.content[0];
328      if (block?.type === 'server_tool_use') {
329        return resolvedToolUseIDs.has(block.id);
330      }
331    }
332    const toolUseID = getToolUseID(msg);
333    return !toolUseID || resolvedToolUseIDs.has(toolUseID);
334  }
335  
336  /**
337   * Conservative memo comparator that only bails out when we're CERTAIN
338   * the message won't change. Fails safe by re-rendering when uncertain.
339   *
340   * Exported for testing.
341   */
342  export function areMessageRowPropsEqual(prev: Props, next: Props): boolean {
343    // Different message reference = content may have changed, must re-render
344    if (prev.message !== next.message) return false;
345  
346    // Screen mode change = re-render
347    if (prev.screen !== next.screen) return false;
348  
349    // Verbose toggle changes thinking block visibility
350    if (prev.verbose !== next.verbose) return false;
351  
352    // collapsed_read_search is never static in prompt mode (matches shouldRenderStatically)
353    if (prev.message.type === 'collapsed_read_search' && next.screen !== 'transcript') {
354      return false;
355    }
356  
357    // Width change affects Box layout
358    if (prev.columns !== next.columns) return false;
359  
360    // latestBashOutputUUID affects rendering (full vs truncated output)
361    const prevIsLatestBash = prev.latestBashOutputUUID === prev.message.uuid;
362    const nextIsLatestBash = next.latestBashOutputUUID === next.message.uuid;
363    if (prevIsLatestBash !== nextIsLatestBash) return false;
364  
365    // lastThinkingBlockId affects thinking block visibility — but only for
366    // messages that HAVE thinking content. Checking unconditionally busts the
367    // memo for every scrollback message whenever thinking starts/stops (CC-941).
368    if (prev.lastThinkingBlockId !== next.lastThinkingBlockId && hasThinkingContent(next.message)) {
369      return false;
370    }
371  
372    // Check if this message is still "in flight"
373    const isStreaming = isMessageStreaming(prev.message, prev.streamingToolUseIDs);
374    const isResolved = allToolsResolved(prev.message, prev.lookups.resolvedToolUseIDs);
375  
376    // Only bail out for truly static messages
377    if (isStreaming || !isResolved) return false;
378  
379    // Static message - safe to skip re-render
380    return true;
381  }
382  export const MessageRow = React.memo(MessageRowImpl, areMessageRowPropsEqual);
383  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","Command","Box","Screen","Tools","RenderableMessage","getDisplayMessageFromCollapsed","getToolSearchOrReadInfo","getToolUseIdsFromCollapsedGroup","hasAnyToolInProgress","buildMessageLookups","EMPTY_STRING_SET","getProgressMessagesFromLookup","getSiblingToolUseIDsFromLookup","getToolUseID","hasThinkingContent","Message","MessageModel","shouldRenderStatically","MessageTimestamp","OffscreenFreeze","Props","message","isUserContinuation","hasContentAfter","tools","commands","verbose","inProgressToolUseIDs","Set","streamingToolUseIDs","screen","canAnimate","onOpenRateLimitOptions","lastThinkingBlockId","latestBashOutputUUID","columns","isLoading","lookups","ReturnType","hasContentAfterIndex","messages","index","i","length","msg","type","content","name","input","isCollapsible","has","id","firstInput","toolName","MessageRowImpl","t0","$","_c","isTranscriptMode","isGrouped","isCollapsed","t1","isActiveCollapsedGroup","t2","displayMessage","displayMsg","t3","progressMessagesForMessage","t4","siblingToolUseIDs","isStatic","shouldAnimate","t5","t6","m","some","toolUseID","_temp","timestamp","model","hasMetadata","t7","undefined","t8","messageEl","t9","t10","c","isMessageStreaming","toolIds","allToolsResolved","resolvedToolUseIDs","every","block","areMessageRowPropsEqual","prev","next","prevIsLatestBash","uuid","nextIsLatestBash","isStreaming","isResolved","MessageRow","memo"],"sources":["MessageRow.tsx"],"sourcesContent":["import * as React from 'react'\nimport type { Command } from '../commands.js'\nimport { Box } from '../ink.js'\nimport type { Screen } from '../screens/REPL.js'\nimport type { Tools } from '../Tool.js'\nimport type { RenderableMessage } from '../types/message.js'\nimport {\n  getDisplayMessageFromCollapsed,\n  getToolSearchOrReadInfo,\n  getToolUseIdsFromCollapsedGroup,\n  hasAnyToolInProgress,\n} from '../utils/collapseReadSearch.js'\nimport {\n  type buildMessageLookups,\n  EMPTY_STRING_SET,\n  getProgressMessagesFromLookup,\n  getSiblingToolUseIDsFromLookup,\n  getToolUseID,\n} from '../utils/messages.js'\nimport { hasThinkingContent, Message } from './Message.js'\nimport { MessageModel } from './MessageModel.js'\nimport { shouldRenderStatically } from './Messages.js'\nimport { MessageTimestamp } from './MessageTimestamp.js'\nimport { OffscreenFreeze } from './OffscreenFreeze.js'\n\nexport type Props = {\n  message: RenderableMessage\n  /** Whether the previous message in renderableMessages is also a user message. */\n  isUserContinuation: boolean\n  /**\n   * Whether there is non-skippable content after this message in renderableMessages.\n   * Only needs to be accurate for `collapsed_read_search` messages — used to decide\n   * if the collapsed group spinner should stay active. Pass `false` otherwise.\n   */\n  hasContentAfter: boolean\n  tools: Tools\n  commands: Command[]\n  verbose: boolean\n  inProgressToolUseIDs: Set<string>\n  streamingToolUseIDs: Set<string>\n  screen: Screen\n  canAnimate: boolean\n  onOpenRateLimitOptions?: () => void\n  lastThinkingBlockId: string | null\n  latestBashOutputUUID: string | null\n  columns: number\n  isLoading: boolean\n  lookups: ReturnType<typeof buildMessageLookups>\n}\n\n/**\n * Scans forward from `index+1` to check if any \"real\" content follows. Used to\n * decide whether a collapsed read/search group should stay in its active\n * (grey dot, present-tense \"Reading…\") state while the query is still loading.\n *\n * Exported so Messages.tsx can compute this once per message and pass the\n * result as a boolean prop — avoids passing the full `renderableMessages` array\n * to each MessageRow (which React Compiler would pin in the fiber's memoCache,\n * accumulating every historical version of the array ≈ 1-2MB over a 7-turn session).\n */\nexport function hasContentAfterIndex(\n  messages: RenderableMessage[],\n  index: number,\n  tools: Tools,\n  streamingToolUseIDs: Set<string>,\n): boolean {\n  for (let i = index + 1; i < messages.length; i++) {\n    const msg = messages[i]\n    if (msg?.type === 'assistant') {\n      const content = msg.message.content[0]\n      if (\n        content?.type === 'thinking' ||\n        content?.type === 'redacted_thinking'\n      ) {\n        continue\n      }\n      if (content?.type === 'tool_use') {\n        if (\n          getToolSearchOrReadInfo(content.name, content.input, tools)\n            .isCollapsible\n        ) {\n          continue\n        }\n        // Non-collapsible tool uses appear in syntheticStreamingToolUseMessages\n        // before their ID is added to inProgressToolUseIDs. Skip while streaming\n        // to avoid briefly finalizing the read group.\n        if (streamingToolUseIDs.has(content.id)) {\n          continue\n        }\n      }\n      return true\n    }\n    if (msg?.type === 'system' || msg?.type === 'attachment') {\n      continue\n    }\n    // Tool results arrive while the collapsed group is still being built\n    if (msg?.type === 'user') {\n      const content = msg.message.content[0]\n      if (content?.type === 'tool_result') {\n        continue\n      }\n    }\n    // Collapsible grouped_tool_use messages arrive transiently before being\n    // merged into the current collapsed group on the next render cycle\n    if (msg?.type === 'grouped_tool_use') {\n      const firstInput = msg.messages[0]?.message.content[0]?.input\n      if (\n        getToolSearchOrReadInfo(msg.toolName, firstInput, tools).isCollapsible\n      ) {\n        continue\n      }\n    }\n    return true\n  }\n  return false\n}\n\nfunction MessageRowImpl({\n  message: msg,\n  isUserContinuation,\n  hasContentAfter,\n  tools,\n  commands,\n  verbose,\n  inProgressToolUseIDs,\n  streamingToolUseIDs,\n  screen,\n  canAnimate,\n  onOpenRateLimitOptions,\n  lastThinkingBlockId,\n  latestBashOutputUUID,\n  columns,\n  isLoading,\n  lookups,\n}: Props): React.ReactNode {\n  const isTranscriptMode = screen === 'transcript'\n  const isGrouped = msg.type === 'grouped_tool_use'\n  const isCollapsed = msg.type === 'collapsed_read_search'\n\n  // A collapsed group is \"active\" (grey dot, present tense \"Reading…\") when its tools\n  // are still executing OR when the overall query is still running with nothing after it.\n  // hasAnyToolInProgress takes priority: if tools are running, always show active regardless\n  // of what else is in the message list (avoids false finalization during parallel execution).\n  const isActiveCollapsedGroup =\n    isCollapsed &&\n    (hasAnyToolInProgress(msg, inProgressToolUseIDs) ||\n      (isLoading && !hasContentAfter))\n\n  const displayMsg = isGrouped\n    ? msg.displayMessage\n    : isCollapsed\n      ? getDisplayMessageFromCollapsed(msg)\n      : msg\n\n  const progressMessagesForMessage =\n    isGrouped || isCollapsed ? [] : getProgressMessagesFromLookup(msg, lookups)\n\n  const siblingToolUseIDs =\n    isGrouped || isCollapsed\n      ? EMPTY_STRING_SET\n      : getSiblingToolUseIDsFromLookup(msg, lookups)\n\n  const isStatic = shouldRenderStatically(\n    msg,\n    streamingToolUseIDs,\n    inProgressToolUseIDs,\n    siblingToolUseIDs,\n    screen,\n    lookups,\n  )\n\n  let shouldAnimate = false\n  if (canAnimate) {\n    if (isGrouped) {\n      shouldAnimate = msg.messages.some(m => {\n        const content = m.message.content[0]\n        return (\n          content?.type === 'tool_use' && inProgressToolUseIDs.has(content.id)\n        )\n      })\n    } else if (isCollapsed) {\n      shouldAnimate = hasAnyToolInProgress(msg, inProgressToolUseIDs)\n    } else {\n      const toolUseID = getToolUseID(msg)\n      shouldAnimate = !toolUseID || inProgressToolUseIDs.has(toolUseID)\n    }\n  }\n\n  const hasMetadata =\n    isTranscriptMode &&\n    displayMsg.type === 'assistant' &&\n    displayMsg.message.content.some(c => c.type === 'text') &&\n    (displayMsg.timestamp || displayMsg.message.model)\n\n  const messageEl = (\n    <Message\n      message={msg}\n      lookups={lookups}\n      addMargin={!hasMetadata}\n      containerWidth={hasMetadata ? undefined : columns}\n      tools={tools}\n      commands={commands}\n      verbose={verbose}\n      inProgressToolUseIDs={inProgressToolUseIDs}\n      progressMessagesForMessage={progressMessagesForMessage}\n      shouldAnimate={shouldAnimate}\n      shouldShowDot={true}\n      isTranscriptMode={isTranscriptMode}\n      isStatic={isStatic}\n      onOpenRateLimitOptions={onOpenRateLimitOptions}\n      isActiveCollapsedGroup={isActiveCollapsedGroup}\n      isUserContinuation={isUserContinuation}\n      lastThinkingBlockId={lastThinkingBlockId}\n      latestBashOutputUUID={latestBashOutputUUID}\n    />\n  )\n  // OffscreenFreeze: the outer React.memo already bails for static messages,\n  // so this only wraps rows that DO re-render — in-progress tools, collapsed\n  // read/search spinners, bash elapsed timers. When those rows have scrolled\n  // into terminal scrollback (non-fullscreen external builds), any content\n  // change forces log-update.ts into a full terminal reset per tick. Freezing\n  // returns the cached element ref so React bails and produces zero diff.\n  if (!hasMetadata) {\n    return <OffscreenFreeze>{messageEl}</OffscreenFreeze>\n  }\n  // Margin on children, not here — else null items (hook_success etc.) get phantom 1-row spacing.\n  return (\n    <OffscreenFreeze>\n      <Box width={columns} flexDirection=\"column\">\n        <Box\n          flexDirection=\"row\"\n          justifyContent=\"flex-end\"\n          gap={1}\n          marginTop={1}\n        >\n          <MessageTimestamp\n            message={displayMsg}\n            isTranscriptMode={isTranscriptMode}\n          />\n          <MessageModel\n            message={displayMsg}\n            isTranscriptMode={isTranscriptMode}\n          />\n        </Box>\n        {messageEl}\n      </Box>\n    </OffscreenFreeze>\n  )\n}\n\n/**\n * Checks if a message is \"streaming\" - i.e., its content may still be changing.\n * Exported for testing.\n */\nexport function isMessageStreaming(\n  msg: RenderableMessage,\n  streamingToolUseIDs: Set<string>,\n): boolean {\n  if (msg.type === 'grouped_tool_use') {\n    return msg.messages.some(m => {\n      const content = m.message.content[0]\n      return content?.type === 'tool_use' && streamingToolUseIDs.has(content.id)\n    })\n  }\n  if (msg.type === 'collapsed_read_search') {\n    const toolIds = getToolUseIdsFromCollapsedGroup(msg)\n    return toolIds.some(id => streamingToolUseIDs.has(id))\n  }\n  const toolUseID = getToolUseID(msg)\n  return !!toolUseID && streamingToolUseIDs.has(toolUseID)\n}\n\n/**\n * Checks if all tools in a message are resolved.\n * Exported for testing.\n */\nexport function allToolsResolved(\n  msg: RenderableMessage,\n  resolvedToolUseIDs: Set<string>,\n): boolean {\n  if (msg.type === 'grouped_tool_use') {\n    return msg.messages.every(m => {\n      const content = m.message.content[0]\n      return content?.type === 'tool_use' && resolvedToolUseIDs.has(content.id)\n    })\n  }\n  if (msg.type === 'collapsed_read_search') {\n    const toolIds = getToolUseIdsFromCollapsedGroup(msg)\n    return toolIds.every(id => resolvedToolUseIDs.has(id))\n  }\n  if (msg.type === 'assistant') {\n    const block = msg.message.content[0]\n    if (block?.type === 'server_tool_use') {\n      return resolvedToolUseIDs.has(block.id)\n    }\n  }\n  const toolUseID = getToolUseID(msg)\n  return !toolUseID || resolvedToolUseIDs.has(toolUseID)\n}\n\n/**\n * Conservative memo comparator that only bails out when we're CERTAIN\n * the message won't change. Fails safe by re-rendering when uncertain.\n *\n * Exported for testing.\n */\nexport function areMessageRowPropsEqual(prev: Props, next: Props): boolean {\n  // Different message reference = content may have changed, must re-render\n  if (prev.message !== next.message) return false\n\n  // Screen mode change = re-render\n  if (prev.screen !== next.screen) return false\n\n  // Verbose toggle changes thinking block visibility\n  if (prev.verbose !== next.verbose) return false\n\n  // collapsed_read_search is never static in prompt mode (matches shouldRenderStatically)\n  if (\n    prev.message.type === 'collapsed_read_search' &&\n    next.screen !== 'transcript'\n  ) {\n    return false\n  }\n\n  // Width change affects Box layout\n  if (prev.columns !== next.columns) return false\n\n  // latestBashOutputUUID affects rendering (full vs truncated output)\n  const prevIsLatestBash = prev.latestBashOutputUUID === prev.message.uuid\n  const nextIsLatestBash = next.latestBashOutputUUID === next.message.uuid\n  if (prevIsLatestBash !== nextIsLatestBash) return false\n\n  // lastThinkingBlockId affects thinking block visibility — but only for\n  // messages that HAVE thinking content. Checking unconditionally busts the\n  // memo for every scrollback message whenever thinking starts/stops (CC-941).\n  if (\n    prev.lastThinkingBlockId !== next.lastThinkingBlockId &&\n    hasThinkingContent(next.message)\n  ) {\n    return false\n  }\n\n  // Check if this message is still \"in flight\"\n  const isStreaming = isMessageStreaming(prev.message, prev.streamingToolUseIDs)\n  const isResolved = allToolsResolved(\n    prev.message,\n    prev.lookups.resolvedToolUseIDs,\n  )\n\n  // Only bail out for truly static messages\n  if (isStreaming || !isResolved) return false\n\n  // Static message - safe to skip re-render\n  return true\n}\n\nexport const MessageRow = React.memo(MessageRowImpl, areMessageRowPropsEqual)\n"],"mappings":"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","ignoreList":[]}