claude-code/components/tasks/AsyncAgentDetailDialog.tsx

229 lines
29 KiB
TypeScript
Raw Permalink Normal View History

import { c as _c } from "react/compiler-runtime";
import React, { useMemo } from 'react';
import type { DeepImmutable } from 'src/types/utils.js';
import { useElapsedTime } from '../../hooks/useElapsedTime.js';
import type { KeyboardEvent } from '../../ink/events/keyboard-event.js';
import { Box, Text, useTheme } from '../../ink.js';
import { useKeybindings } from '../../keybindings/useKeybinding.js';
import { getEmptyToolPermissionContext } from '../../Tool.js';
import type { LocalAgentTaskState } from '../../tasks/LocalAgentTask/LocalAgentTask.js';
import { getTools } from '../../tools.js';
import { formatNumber } from '../../utils/format.js';
import { extractTag } from '../../utils/messages.js';
import { Byline } from '../design-system/Byline.js';
import { Dialog } from '../design-system/Dialog.js';
import { KeyboardShortcutHint } from '../design-system/KeyboardShortcutHint.js';
import { UserPlanMessage } from '../messages/UserPlanMessage.js';
import { renderToolActivity } from './renderToolActivity.js';
import { getTaskStatusColor, getTaskStatusIcon } from './taskStatusUtils.js';
type Props = {
agent: DeepImmutable<LocalAgentTaskState>;
onDone: () => void;
onKillAgent?: () => void;
onBack?: () => void;
};
export function AsyncAgentDetailDialog(t0) {
const $ = _c(54);
const {
agent,
onDone,
onKillAgent,
onBack
} = t0;
const [theme] = useTheme();
let t1;
if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
t1 = getTools(getEmptyToolPermissionContext());
$[0] = t1;
} else {
t1 = $[0];
}
const tools = t1;
const elapsedTime = useElapsedTime(agent.startTime, agent.status === "running", 1000, agent.totalPausedMs ?? 0);
let t2;
if ($[1] !== onDone) {
t2 = {
"confirm:yes": onDone
};
$[1] = onDone;
$[2] = t2;
} else {
t2 = $[2];
}
let t3;
if ($[3] === Symbol.for("react.memo_cache_sentinel")) {
t3 = {
context: "Confirmation"
};
$[3] = t3;
} else {
t3 = $[3];
}
useKeybindings(t2, t3);
let t4;
if ($[4] !== agent.status || $[5] !== onBack || $[6] !== onDone || $[7] !== onKillAgent) {
t4 = e => {
if (e.key === " ") {
e.preventDefault();
onDone();
} else {
if (e.key === "left" && onBack) {
e.preventDefault();
onBack();
} else {
if (e.key === "x" && agent.status === "running" && onKillAgent) {
e.preventDefault();
onKillAgent();
}
}
}
};
$[4] = agent.status;
$[5] = onBack;
$[6] = onDone;
$[7] = onKillAgent;
$[8] = t4;
} else {
t4 = $[8];
}
const handleKeyDown = t4;
let t5;
if ($[9] !== agent.prompt) {
t5 = extractTag(agent.prompt, "plan");
$[9] = agent.prompt;
$[10] = t5;
} else {
t5 = $[10];
}
const planContent = t5;
const displayPrompt = agent.prompt.length > 300 ? agent.prompt.substring(0, 297) + "\u2026" : agent.prompt;
const tokenCount = agent.result?.totalTokens ?? agent.progress?.tokenCount;
const toolUseCount = agent.result?.totalToolUseCount ?? agent.progress?.toolUseCount;
const t6 = agent.selectedAgent?.agentType ?? "agent";
const t7 = agent.description || "Async agent";
let t8;
if ($[11] !== t6 || $[12] !== t7) {
t8 = <Text>{t6} {" "}{t7}</Text>;
$[11] = t6;
$[12] = t7;
$[13] = t8;
} else {
t8 = $[13];
}
const title = t8;
let t9;
if ($[14] !== agent.status) {
t9 = agent.status !== "running" && <Text color={getTaskStatusColor(agent.status)}>{getTaskStatusIcon(agent.status)}{" "}{agent.status === "completed" ? "Completed" : agent.status === "failed" ? "Failed" : "Stopped"}{" \xB7 "}</Text>;
$[14] = agent.status;
$[15] = t9;
} else {
t9 = $[15];
}
let t10;
if ($[16] !== tokenCount) {
t10 = tokenCount !== undefined && tokenCount > 0 && <> · {formatNumber(tokenCount)} tokens</>;
$[16] = tokenCount;
$[17] = t10;
} else {
t10 = $[17];
}
let t11;
if ($[18] !== toolUseCount) {
t11 = toolUseCount !== undefined && toolUseCount > 0 && <>{" "}· {toolUseCount} {toolUseCount === 1 ? "tool" : "tools"}</>;
$[18] = toolUseCount;
$[19] = t11;
} else {
t11 = $[19];
}
let t12;
if ($[20] !== elapsedTime || $[21] !== t10 || $[22] !== t11) {
t12 = <Text dimColor={true}>{elapsedTime}{t10}{t11}</Text>;
$[20] = elapsedTime;
$[21] = t10;
$[22] = t11;
$[23] = t12;
} else {
t12 = $[23];
}
let t13;
if ($[24] !== t12 || $[25] !== t9) {
t13 = <Text>{t9}{t12}</Text>;
$[24] = t12;
$[25] = t9;
$[26] = t13;
} else {
t13 = $[26];
}
const subtitle = t13;
let t14;
if ($[27] !== agent.status || $[28] !== onBack || $[29] !== onKillAgent) {
t14 = exitState => exitState.pending ? <Text>Press {exitState.keyName} again to exit</Text> : <Byline>{onBack && <KeyboardShortcutHint shortcut={"\u2190"} action="go back" />}<KeyboardShortcutHint shortcut="Esc/Enter/Space" action="close" />{agent.status === "running" && onKillAgent && <KeyboardShortcutHint shortcut="x" action="stop" />}</Byline>;
$[27] = agent.status;
$[28] = onBack;
$[29] = onKillAgent;
$[30] = t14;
} else {
t14 = $[30];
}
let t15;
if ($[31] !== agent.progress || $[32] !== agent.status || $[33] !== theme) {
t15 = agent.status === "running" && agent.progress?.recentActivities && agent.progress.recentActivities.length > 0 && <Box flexDirection="column"><Text bold={true} dimColor={true}>Progress</Text>{agent.progress.recentActivities.map((activity, i) => <Text key={i} dimColor={i < agent.progress.recentActivities.length - 1} wrap="truncate-end">{i === agent.progress.recentActivities.length - 1 ? "\u203A " : " "}{renderToolActivity(activity, tools, theme)}</Text>)}</Box>;
$[31] = agent.progress;
$[32] = agent.status;
$[33] = theme;
$[34] = t15;
} else {
t15 = $[34];
}
let t16;
if ($[35] !== displayPrompt || $[36] !== planContent) {
t16 = planContent ? <Box marginTop={1}><UserPlanMessage addMargin={false} planContent={planContent} /></Box> : <Box flexDirection="column" marginTop={1}><Text bold={true} dimColor={true}>Prompt</Text><Text wrap="wrap">{displayPrompt}</Text></Box>;
$[35] = displayPrompt;
$[36] = planContent;
$[37] = t16;
} else {
t16 = $[37];
}
let t17;
if ($[38] !== agent.error || $[39] !== agent.status) {
t17 = agent.status === "failed" && agent.error && <Box flexDirection="column" marginTop={1}><Text bold={true} color="error">Error</Text><Text color="error" wrap="wrap">{agent.error}</Text></Box>;
$[38] = agent.error;
$[39] = agent.status;
$[40] = t17;
} else {
t17 = $[40];
}
let t18;
if ($[41] !== t15 || $[42] !== t16 || $[43] !== t17) {
t18 = <Box flexDirection="column">{t15}{t16}{t17}</Box>;
$[41] = t15;
$[42] = t16;
$[43] = t17;
$[44] = t18;
} else {
t18 = $[44];
}
let t19;
if ($[45] !== onDone || $[46] !== subtitle || $[47] !== t14 || $[48] !== t18 || $[49] !== title) {
t19 = <Dialog title={title} subtitle={subtitle} onCancel={onDone} color="background" inputGuide={t14}>{t18}</Dialog>;
$[45] = onDone;
$[46] = subtitle;
$[47] = t14;
$[48] = t18;
$[49] = title;
$[50] = t19;
} else {
t19 = $[50];
}
let t20;
if ($[51] !== handleKeyDown || $[52] !== t19) {
t20 = <Box flexDirection="column" tabIndex={0} autoFocus={true} onKeyDown={handleKeyDown}>{t19}</Box>;
$[51] = handleKeyDown;
$[52] = t19;
$[53] = t20;
} else {
t20 = $[53];
}
return t20;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZU1lbW8iLCJEZWVwSW1tdXRhYmxlIiwidXNlRWxhcHNlZFRpbWUiLCJLZXlib2FyZEV2ZW50IiwiQm94IiwiVGV4dCIsInVzZVRoZW1lIiwidXNlS2V5YmluZGluZ3MiLCJnZXRFbXB0eVRvb2xQZXJtaXNzaW9uQ29udGV4dCIsIkxvY2FsQWdlbnRUYXNrU3RhdGUiLCJnZXRUb29scyIsImZvcm1hdE51bWJlciIsImV4dHJhY3RUYWciLCJCeWxpbmUiLCJEaWFsb2ciLCJLZXlib2FyZFNob3J0Y3V0SGludCIsIlVzZXJQbGFuTWVzc2FnZSIsInJlbmRlclRvb2xBY3Rpdml0eSIsImdldFRhc2tTdGF0dXNDb2xvciIsImdldFRhc2tTdGF0dXNJY29uIiwiUHJvcHMiLCJhZ2VudCIsIm9uRG9uZSIsIm9uS2lsbEFnZW50Iiwib25CYWNrIiwiQXN5bmNBZ2VudERldGFpbERpYWxvZyIsInQwIiwiJCIsIl9jIiwidGhlbWUiLCJ0MSIsIlN5bWJvbCIsImZvciIsInRvb2xzIiwiZWxhcHNlZFRpbWUiLCJzdGFydFRpbWUiLCJzdGF0dXMiLCJ0b3RhbFBhdXNlZE1zIiwidDIiLCJ0MyIsImNvbnRleHQiLCJ0NCIsImUiLCJrZXkiLCJwcmV2ZW50RGVmYXVsdCIsImhhbmRsZUtleURvd24iLCJ0NSIsInByb21wdCIsInBsYW5Db250ZW50IiwiZGlzcGxheVByb21wdCIsImxlbmd0aCIsInN1YnN0cmluZyIsInRva2VuQ291bnQiLCJyZXN1bHQiLCJ0b3RhbFRva2VucyIsInByb2dyZXNzIiwidG9vbFVzZUNvdW50IiwidG90YWxUb29sVXNlQ291bnQiLCJ0NiIsInNlbGVjdGVkQWdlbnQiLCJhZ2VudFR5cGUiLCJ0NyIsImRlc2NyaXB0aW9uIiwidDgiLCJ0aXRsZSIsInQ5IiwidDEwIiwidW5kZWZpbmVkIiwidDExIiwidDEyIiwidDEzIiwic3VidGl0bGUiLCJ0MTQiLCJleGl0U3RhdGUiLCJwZW5kaW5nIiwia2V5TmFtZSIsInQxNSIsInJlY2VudEFjdGl2aXRpZXMiLCJtYXAiLCJhY3Rpdml0eSIsImkiLCJ0MTYiLCJ0MTciLCJlcnJvciIsInQxOCIsInQxOSIsInQyMCJdLCJzb3VyY2VzIjpbIkFzeW5jQWdlbnREZXRhaWxEaWFsb2cudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCwgeyB1c2VNZW1vIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IERlZXBJbW11dGFibGUgfSBmcm9tICdzcmMvdHlwZXMvdXRpbHMuanMnXG5pbXBvcnQgeyB1c2VFbGFwc2VkVGltZSB9IGZyb20gJy4uLy4uL2hvb2tzL3VzZUVsYXBzZWRUaW1lLmpzJ1xuaW1wb3J0IHR5cGUgeyBLZXlib2FyZEV2ZW50IH0gZnJvbSAnLi4vLi4vaW5rL2V2ZW50cy9rZXlib2FyZC1ldmVudC5qcydcbmltcG9ydCB7IEJveCwgVGV4dCwgdXNlVGhlbWUgfSBmcm9tICcuLi8uLi9pbmsuanMnXG5pbXBvcnQgeyB1c2VLZXliaW5kaW5ncyB9IGZyb20gJy4uLy4uL2tleWJpbmRpbmdzL3VzZUtleWJpbmRpbmcuanMnXG5pbXBvcnQgeyBnZXRFbXB0eVRvb2xQZXJtaXNzaW9uQ29udGV4dCB9IGZyb20gJy4uLy4uL1Rvb2wuanMnXG5pbXBvcnQgdHlwZSB7IExvY2FsQWdlbnRUYXNrU3RhdGUgfSBmcm9tICcuLi8uLi90YXNrcy9Mb2NhbEFnZW50VGFzay9Mb2NhbEFnZW50VGFzay5qcydcbmltcG9ydCB7IGdldFRvb2xzIH0gZnJvbSAnLi4vLi4vdG9vbHMuanMnXG5pbXBvcnQgeyBmb3JtYXROdW1iZXIgfSBmcm9tICcuLi8uLi91dGlscy9mb3JtYXQuanMnXG5pbXBvcnQgeyBleHRyYWN0VGFnIH0gZnJvbSAnLi4vLi4vdXRpbHMvbWVzc2FnZXMuanMnXG5pbXBvcnQgeyBCeWxpbmUgfSBmcm9tICcuLi9kZXNpZ24tc3lzdGVtL0J5bGluZS5qcydcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4uL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuaW1wb3J0IHsgS2V5Ym9hcmRTaG9ydGN1dEhpbnQgfSBmcm9tICcuLi9kZXNpZ24tc3lzdGVtL0tleWJvYXJkU2hvcnRjdXRIaW50LmpzJ1xuaW1wb3J0IHsgVXNlclBsYW5NZXNzYWdlIH0gZnJvbSAnLi4vbWVzc2FnZXMvVXNlclBsYW5NZXNzYWdlLmpzJ1xuaW1wb3J0IHsgcmVuZGVyVG9vbEFjdGl2aXR5IH0gZnJvbSAnLi9yZW5kZXJUb29sQWN0aXZpdHkuanMnXG5pbXBvcnQgeyBnZXRUYXNrU3RhdHVzQ29sb3IsIGdldFRhc2tTdGF0dXNJY29uIH0gZnJvbSAnLi90YXNrU3RhdHVzVXRpbHMuanMnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIGFnZW50OiBEZWVwSW1tdXRhYmxlPExvY2FsQWdlbnRUYXNrU3RhdGU+XG4gIG9uRG9uZTogKCkgPT4gdm9pZFxuICBvbktpbGxBZ2VudD86ICgpID0+IHZvaWRcbiAgb25CYWNrPzogKCkgPT4gdm9pZFxufVxuXG5leHBvcnQgZnVuY3Rpb24gQXN5bmNBZ2VudERldGFpbERpYWxvZyh7XG4gIGFnZW50LFxuICBvbkRvbmUsXG4gIG9uS2lsbEFnZW50LFxuICBvbkJhY2ssXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IFt0aGVtZV0gPSB1c2VUaGVtZSgpXG5cbiAgLy8gR2V0IHRvb2xzIGZvciByZW5kZXJpbmcgYWN0aXZpdHkgbWVzc2FnZXNcbiAgY29uc3QgdG9vbHMgPSB1c2VNZW1vKCgpID0+IGdldFRvb2xzKGdldEVtcHR5VG9vbFBlcm1pc3Npb25Db250ZXh0KCkpLCBbXSlcblxuICBjb25zdCBlbGFwc2VkVGltZSA9IHVzZUVsYXBzZWRUaW1lKFxuICAgIGFnZW50LnN0YXJ0VGltZSxcbiAgICBhZ2VudC5zdGF0dXMgPT09ICdydW5uaW5nJyxcbiAgICAxMDAwLFxuICAgIGFnZW50LnRvdGFsUGF1c2VkTXMgPz8gMCxcbiAgKVxuXG4gIC8vIFJlc3RvcmUgY29uZmlybTp5ZXMgKEVudGVyL3kpIGRpc21pc3NhbCDigJQgRGlhbG9nIGhhbmRsZXMgY29uZmlybTpubyAoRXNjKVxuICAvLyBpbnRlcm5hbGx5IGJ1dCBkb2VzIE5PVCBhdXRvLXdpcmUgY29uZmlybTp5ZXMuXG4gIHVzZUtleWJpbmRpbmdzKFxuICAgIHtcbiAgICAgICdjb25maXJtOnllcyc6IG9uRG9uZSxcbiAgICB9LFxuICAgIHsgY29udGV4dDogJ0NvbmZpcm1hdGlvbicgfSxcbiAgKVxuXG4gIC8vIENvbXBvbmVudC1zcGVjaWZpYyBzaG9ydGN1dHMgc2hvd24gaW4gVUkgaGludHMgKHg9c3RvcCkgYW5kXG4gIC8vIG5hdmlnYXRpb24ga2V5cyAoc3BhY2U9ZGlzbWlzcywgbGVmdD1iYWNrKS4gVGhlc2UgYXJlIGNvbnRleHQtZGV