import { Component, createEffect, createMemo, createSignal, For, Match, onMount, Show, Switch, onCleanup, Index, type JSX, } from "solid-js" import { createStore } from "solid-js/store" import stripAnsi from "strip-ansi" import { Dynamic } from "solid-js/web" import { AgentPart, AssistantMessage, FilePart, Message as MessageType, Part as PartType, ReasoningPart, TextPart, ToolPart, UserMessage, Todo, QuestionAnswer, QuestionInfo, } from "@opencode-ai/sdk/v2" import { useData } from "../context" import { useFileComponent } from "../context/file" import { useDialog } from "../context/dialog" import { type UiI18n, useI18n } from "../context/i18n" import { BasicTool, GenericTool } from "./basic-tool" import { Accordion } from "./accordion" import { StickyAccordionHeader } from "./sticky-accordion-header" import { Card } from "./card" import { Collapsible } from "./collapsible" import { FileIcon } from "./file-icon" import { Icon } from "./icon" import { ToolErrorCard } from "./tool-error-card" import { Checkbox } from "./checkbox" import { DiffChanges } from "./diff-changes" import { Markdown } from "./markdown" import { ImagePreview } from "./image-preview" import { getDirectory as _getDirectory, getFilename } from "@opencode-ai/util/path" import { checksum } from "@opencode-ai/util/encode" import { Tooltip } from "./tooltip" import { IconButton } from "./icon-button" import { TextShimmer } from "./text-shimmer" import { AnimatedCountList } from "./tool-count-summary" import { ToolStatusTitle } from "./tool-status-title" import { animate } from "motion" import { useLocation } from "@solidjs/router" import { attached, inline, kind } from "./message-file" function ShellSubmessage(props: { text: string; animate?: boolean }) { let widthRef: HTMLSpanElement | undefined let valueRef: HTMLSpanElement | undefined onMount(() => { if (!props.animate) return requestAnimationFrame(() => { if (widthRef) { animate(widthRef, { width: "auto" }, { type: "spring", visualDuration: 0.25, bounce: 0 }) } if (valueRef) { animate(valueRef, { opacity: 1, filter: "blur(0px)" }, { duration: 0.32, ease: [0.16, 1, 0.3, 1] }) } }) }) return ( {props.text} ) } interface Diagnostic { range: { start: { line: number; character: number } end: { line: number; character: number } } message: string severity?: number } function getDiagnostics( diagnosticsByFile: Record | undefined, filePath: string | undefined, ): Diagnostic[] { if (!diagnosticsByFile || !filePath) return [] const diagnostics = diagnosticsByFile[filePath] ?? [] return diagnostics.filter((d) => d.severity === 1).slice(0, 3) } function DiagnosticsDisplay(props: { diagnostics: Diagnostic[] }): JSX.Element { const i18n = useI18n() return ( 0}>
{(diagnostic) => (
{i18n.t("ui.messagePart.diagnostic.error")} [{diagnostic.range.start.line + 1}:{diagnostic.range.start.character + 1}] {diagnostic.message}
)}
) } export interface MessageProps { message: MessageType parts: PartType[] actions?: UserActions showAssistantCopyPartID?: string | null showReasoningSummaries?: boolean } export type SessionAction = (input: { sessionID: string; messageID: string }) => Promise | void export type UserActions = { fork?: SessionAction revert?: SessionAction } export interface MessagePartProps { part: PartType message: MessageType hideDetails?: boolean defaultOpen?: boolean showAssistantCopyPartID?: string | null turnDurationMs?: number } export type PartComponent = Component export const PART_MAPPING: Record = {} const TEXT_RENDER_THROTTLE_MS = 100 function createThrottledValue(getValue: () => string) { const [value, setValue] = createSignal(getValue()) let timeout: ReturnType | undefined let last = 0 createEffect(() => { const next = getValue() const now = Date.now() const remaining = TEXT_RENDER_THROTTLE_MS - (now - last) if (remaining <= 0) { if (timeout) { clearTimeout(timeout) timeout = undefined } last = now setValue(next) return } if (timeout) clearTimeout(timeout) timeout = setTimeout(() => { last = Date.now() setValue(next) timeout = undefined }, remaining) }) onCleanup(() => { if (timeout) clearTimeout(timeout) }) return value } function relativizeProjectPath(path: string, directory?: string) { if (!path) return "" if (!directory) return path if (directory === "/") return path if (directory === "\\") return path if (path === directory) return "" const separator = directory.includes("\\") ? "\\" : "/" const prefix = directory.endsWith(separator) ? directory : directory + separator if (!path.startsWith(prefix)) return path return path.slice(directory.length) } function getDirectory(path: string | undefined) { const data = useData() return relativizeProjectPath(_getDirectory(path), data.directory) } import type { IconProps } from "./icon" export type ToolInfo = { icon: IconProps["name"] title: string subtitle?: string } function agentTitle(i18n: UiI18n, type?: string) { if (!type) return i18n.t("ui.tool.agent.default") return i18n.t("ui.tool.agent", { type }) } export function getToolInfo(tool: string, input: any = {}): ToolInfo { const i18n = useI18n() switch (tool) { case "read": return { icon: "glasses", title: i18n.t("ui.tool.read"), subtitle: input.filePath ? getFilename(input.filePath) : undefined, } case "list": return { icon: "bullet-list", title: i18n.t("ui.tool.list"), subtitle: input.path ? getFilename(input.path) : undefined, } case "glob": return { icon: "magnifying-glass-menu", title: i18n.t("ui.tool.glob"), subtitle: input.pattern, } case "grep": return { icon: "magnifying-glass-menu", title: i18n.t("ui.tool.grep"), subtitle: input.pattern, } case "webfetch": return { icon: "window-cursor", title: i18n.t("ui.tool.webfetch"), subtitle: input.url, } case "websearch": return { icon: "window-cursor", title: i18n.t("ui.tool.websearch"), subtitle: input.query, } case "codesearch": return { icon: "code", title: i18n.t("ui.tool.codesearch"), subtitle: input.query, } case "task": { const type = typeof input.subagent_type === "string" && input.subagent_type ? input.subagent_type[0]!.toUpperCase() + input.subagent_type.slice(1) : undefined return { icon: "task", title: agentTitle(i18n, type), subtitle: input.description, } } case "bash": return { icon: "console", title: i18n.t("ui.tool.shell"), subtitle: input.description, } case "edit": return { icon: "code-lines", title: i18n.t("ui.messagePart.title.edit"), subtitle: input.filePath ? getFilename(input.filePath) : undefined, } case "write": return { icon: "code-lines", title: i18n.t("ui.messagePart.title.write"), subtitle: input.filePath ? getFilename(input.filePath) : undefined, } case "apply_patch": return { icon: "code-lines", title: i18n.t("ui.tool.patch"), subtitle: input.files?.length ? `${input.files.length} ${i18n.t(input.files.length > 1 ? "ui.common.file.other" : "ui.common.file.one")}` : undefined, } case "todowrite": return { icon: "checklist", title: i18n.t("ui.tool.todos"), } case "todoread": return { icon: "checklist", title: i18n.t("ui.tool.todos.read"), } case "question": return { icon: "bubble-5", title: i18n.t("ui.tool.questions"), } case "skill": return { icon: "brain", title: input.name || i18n.t("ui.tool.skill"), } default: return { icon: "mcp", title: tool, } } } function urls(text: string | undefined) { if (!text) return [] const seen = new Set() return [...text.matchAll(/https?:\/\/[^\s<>"'`)\]]+/g)] .map((item) => item[0].replace(/[),.;:!?]+$/g, "")) .filter((item) => { if (seen.has(item)) return false seen.add(item) return true }) } function sessionLink(id: string | undefined, path: string, href?: (id: string) => string | undefined) { if (!id) return const direct = href?.(id) if (direct) return direct const idx = path.indexOf("/session") if (idx === -1) return return `${path.slice(0, idx)}/session/${id}` } const CONTEXT_GROUP_TOOLS = new Set(["read", "glob", "grep", "list"]) const HIDDEN_TOOLS = new Set(["todowrite", "todoread"]) function list(value: T[] | undefined | null, fallback: T[]) { if (Array.isArray(value)) return value return fallback } function same(a: readonly T[] | undefined, b: readonly T[] | undefined) { if (a === b) return true if (!a || !b) return false if (a.length !== b.length) return false return a.every((x, i) => x === b[i]) } type PartRef = { messageID: string partID: string } type PartGroup = | { key: string type: "part" ref: PartRef } | { key: string type: "context" refs: PartRef[] } function sameRef(a: PartRef, b: PartRef) { return a.messageID === b.messageID && a.partID === b.partID } function sameGroup(a: PartGroup, b: PartGroup) { if (a === b) return true if (a.key !== b.key) return false if (a.type !== b.type) return false if (a.type === "part") { if (b.type !== "part") return false return sameRef(a.ref, b.ref) } if (b.type !== "context") return false if (a.refs.length !== b.refs.length) return false return a.refs.every((ref, i) => sameRef(ref, b.refs[i]!)) } function sameGroups(a: readonly PartGroup[] | undefined, b: readonly PartGroup[] | undefined) { if (a === b) return true if (!a || !b) return false if (a.length !== b.length) return false return a.every((item, i) => sameGroup(item, b[i]!)) } function groupParts(parts: { messageID: string; part: PartType }[]) { const result: PartGroup[] = [] let start = -1 const flush = (end: number) => { if (start < 0) return const first = parts[start] const last = parts[end] if (!first || !last) { start = -1 return } result.push({ key: `context:${first.part.id}`, type: "context", refs: parts.slice(start, end + 1).map((item) => ({ messageID: item.messageID, partID: item.part.id, })), }) start = -1 } parts.forEach((item, index) => { if (isContextGroupTool(item.part)) { if (start < 0) start = index return } flush(index - 1) result.push({ key: `part:${item.messageID}:${item.part.id}`, type: "part", ref: { messageID: item.messageID, partID: item.part.id, }, }) }) flush(parts.length - 1) return result } function index(items: readonly T[]) { return new Map(items.map((item) => [item.id, item] as const)) } function renderable(part: PartType, showReasoningSummaries = true) { if (part.type === "tool") { if (HIDDEN_TOOLS.has(part.tool)) return false if (part.tool === "question") return part.state.status !== "pending" && part.state.status !== "running" return true } if (part.type === "text") return !!part.text?.trim() if (part.type === "reasoning") return showReasoningSummaries && !!part.text?.trim() return !!PART_MAPPING[part.type] } function toolDefaultOpen(tool: string, shell = false, edit = false) { if (tool === "bash") return shell if (tool === "edit" || tool === "write" || tool === "apply_patch") return edit } function partDefaultOpen(part: PartType, shell = false, edit = false) { if (part.type !== "tool") return return toolDefaultOpen(part.tool, shell, edit) } export function AssistantParts(props: { messages: AssistantMessage[] showAssistantCopyPartID?: string | null turnDurationMs?: number working?: boolean showReasoningSummaries?: boolean shellToolDefaultOpen?: boolean editToolDefaultOpen?: boolean }) { const data = useData() const emptyParts: PartType[] = [] const emptyTools: ToolPart[] = [] const msgs = createMemo(() => index(props.messages)) const part = createMemo( () => new Map( props.messages.map((message) => [message.id, index(list(data.store.part?.[message.id], emptyParts))] as const), ), ) const grouped = createMemo( () => groupParts( props.messages.flatMap((message) => list(data.store.part?.[message.id], emptyParts) .filter((part) => renderable(part, props.showReasoningSummaries ?? true)) .map((part) => ({ messageID: message.id, part, })), ), ), [] as PartGroup[], { equals: sameGroups }, ) const last = createMemo(() => grouped().at(-1)?.key) return ( {(entryAccessor) => { const entryType = createMemo(() => entryAccessor().type) return ( {(() => { const parts = createMemo( () => { const entry = entryAccessor() if (entry.type !== "context") return emptyTools return entry.refs .map((ref) => part().get(ref.messageID)?.get(ref.partID)) .filter((part): part is ToolPart => !!part && isContextGroupTool(part)) }, emptyTools, { equals: same }, ) const busy = createMemo(() => props.working && last() === entryAccessor().key) return ( 0}> ) })()} {(() => { const message = createMemo(() => { const entry = entryAccessor() if (entry.type !== "part") return return msgs().get(entry.ref.messageID) }) const item = createMemo(() => { const entry = entryAccessor() if (entry.type !== "part") return return part().get(entry.ref.messageID)?.get(entry.ref.partID) }) return ( ) })()} ) }} ) } function isContextGroupTool(part: PartType): part is ToolPart { return part.type === "tool" && CONTEXT_GROUP_TOOLS.has(part.tool) } function contextToolDetail(part: ToolPart): string | undefined { const info = getToolInfo(part.tool, part.state.input ?? {}) if (info.subtitle) return info.subtitle if (part.state.status === "error") return part.state.error if ((part.state.status === "running" || part.state.status === "completed") && part.state.title) return part.state.title const description = part.state.input?.description if (typeof description === "string") return description return undefined } function contextToolTrigger(part: ToolPart, i18n: ReturnType) { const input = (part.state.input ?? {}) as Record const path = typeof input.path === "string" ? input.path : "/" const filePath = typeof input.filePath === "string" ? input.filePath : undefined const pattern = typeof input.pattern === "string" ? input.pattern : undefined const include = typeof input.include === "string" ? input.include : undefined const offset = typeof input.offset === "number" ? input.offset : undefined const limit = typeof input.limit === "number" ? input.limit : undefined switch (part.tool) { case "read": { const args: string[] = [] if (offset !== undefined) args.push("offset=" + offset) if (limit !== undefined) args.push("limit=" + limit) return { title: i18n.t("ui.tool.read"), subtitle: filePath ? getFilename(filePath) : "", args, } } case "list": return { title: i18n.t("ui.tool.list"), subtitle: getDirectory(path), } case "glob": return { title: i18n.t("ui.tool.glob"), subtitle: getDirectory(path), args: pattern ? ["pattern=" + pattern] : [], } case "grep": { const args: string[] = [] if (pattern) args.push("pattern=" + pattern) if (include) args.push("include=" + include) return { title: i18n.t("ui.tool.grep"), subtitle: getDirectory(path), args, } } default: { const info = getToolInfo(part.tool, input) return { title: info.title, subtitle: info.subtitle || contextToolDetail(part), args: [], } } } } function contextToolSummary(parts: ToolPart[]) { const read = parts.filter((part) => part.tool === "read").length const search = parts.filter((part) => part.tool === "glob" || part.tool === "grep").length const list = parts.filter((part) => part.tool === "list").length return { read, search, list } } function ExaOutput(props: { output?: string }) { const links = createMemo(() => urls(props.output)) return ( 0}> ) } export function registerPartComponent(type: string, component: PartComponent) { PART_MAPPING[type] = component } export function Message(props: MessageProps) { return ( {(userMessage) => ( )} {(assistantMessage) => ( )} ) } export function AssistantMessageDisplay(props: { message: AssistantMessage parts: PartType[] showAssistantCopyPartID?: string | null showReasoningSummaries?: boolean }) { const emptyTools: ToolPart[] = [] const part = createMemo(() => index(props.parts)) const grouped = createMemo( () => groupParts( props.parts .filter((part) => renderable(part, props.showReasoningSummaries ?? true)) .map((part) => ({ messageID: props.message.id, part, })), ), [] as PartGroup[], { equals: sameGroups }, ) return ( {(entryAccessor) => { const entryType = createMemo(() => entryAccessor().type) return ( {(() => { const parts = createMemo( () => { const entry = entryAccessor() if (entry.type !== "context") return emptyTools return entry.refs .map((ref) => part().get(ref.partID)) .filter((part): part is ToolPart => !!part && isContextGroupTool(part)) }, emptyTools, { equals: same }, ) return ( 0}> ) })()} {(() => { const item = createMemo(() => { const entry = entryAccessor() if (entry.type !== "part") return return part().get(entry.ref.partID) }) return ( ) })()} ) }} ) } function ContextToolGroup(props: { parts: ToolPart[]; busy?: boolean }) { const i18n = useI18n() const [open, setOpen] = createSignal(false) const pending = createMemo( () => !!props.busy || props.parts.some((part) => part.state.status === "pending" || part.state.status === "running"), ) const summary = createMemo(() => contextToolSummary(props.parts)) return (
{(partAccessor) => { const trigger = createMemo(() => contextToolTrigger(partAccessor(), i18n)) const running = createMemo( () => partAccessor().state.status === "pending" || partAccessor().state.status === "running", ) return (
{trigger().subtitle} {(arg) => {arg}}
) }}
) } export function UserMessageDisplay(props: { message: UserMessage; parts: PartType[]; actions?: UserActions }) { const data = useData() const dialog = useDialog() const i18n = useI18n() const [state, setState] = createStore({ copied: false, busy: undefined as "fork" | "revert" | undefined, }) const copied = () => state.copied const busy = () => state.busy const textPart = createMemo( () => props.parts?.find((p) => p.type === "text" && !(p as TextPart).synthetic) as TextPart | undefined, ) const text = createMemo(() => textPart()?.text || "") const files = createMemo(() => (props.parts?.filter((p) => p.type === "file") as FilePart[]) ?? []) const attachments = createMemo(() => files().filter(attached)) const inlineFiles = createMemo(() => files().filter(inline)) const agents = createMemo(() => (props.parts?.filter((p) => p.type === "agent") as AgentPart[]) ?? []) const model = createMemo(() => { const providerID = props.message.model?.providerID const modelID = props.message.model?.modelID if (!providerID || !modelID) return "" const match = data.store.provider?.all?.find((p) => p.id === providerID) return match?.models?.[modelID]?.name ?? modelID }) const timefmt = createMemo(() => new Intl.DateTimeFormat(i18n.locale(), { timeStyle: "short" })) const stamp = createMemo(() => { const created = props.message.time?.created if (typeof created !== "number") return "" return timefmt().format(created) }) const metaHead = createMemo(() => { const agent = props.message.agent const items = [agent ? agent[0]?.toUpperCase() + agent.slice(1) : "", model()] return items.filter((x) => !!x).join("\u00A0\u00B7\u00A0") }) const metaTail = stamp const openImagePreview = (url: string, alt?: string) => { dialog.show(() => ) } const handleCopy = async () => { const content = text() if (!content) return await navigator.clipboard.writeText(content) setState("copied", true) setTimeout(() => setState("copied", false), 2000) } const run = (kind: "fork" | "revert") => { const act = kind === "fork" ? props.actions?.fork : props.actions?.revert if (!act || busy()) return setState("busy", kind) void Promise.resolve() .then(() => act({ sessionID: props.message.sessionID, messageID: props.message.id, }), ) .finally(() => { if (busy() === kind) setState("busy", undefined) }) } return (
0}>
{(file) => { const type = kind(file) const name = file.filename ?? i18n.t("ui.message.attachment.alt") return (
{ if (type === "image") openImagePreview(file.url, name) }} > {name}
} > {name}
) }}
<>
{metaHead()} {"\u00A0\u00B7\u00A0"} {metaTail()} e.preventDefault()} onClick={(event) => { event.stopPropagation() run("fork") }} aria-label={i18n.t("ui.message.forkMessage")} /> e.preventDefault()} onClick={(event) => { event.stopPropagation() run("revert") }} aria-label={i18n.t("ui.message.revertMessage")} /> e.preventDefault()} onClick={(event) => { event.stopPropagation() handleCopy() }} aria-label={copied() ? i18n.t("ui.message.copied") : i18n.t("ui.message.copyMessage")} />
) } type HighlightSegment = { text: string; type?: "file" | "agent" } function HighlightedText(props: { text: string; references: FilePart[]; agents: AgentPart[] }) { const segments = createMemo(() => { const text = props.text const allRefs: { start: number; end: number; type: "file" | "agent" }[] = [ ...props.references .filter((r) => r.source?.text?.start !== undefined && r.source?.text?.end !== undefined) .map((r) => ({ start: r.source!.text!.start, end: r.source!.text!.end, type: "file" as const })), ...props.agents .filter((a) => a.source?.start !== undefined && a.source?.end !== undefined) .map((a) => ({ start: a.source!.start, end: a.source!.end, type: "agent" as const })), ].sort((a, b) => a.start - b.start) const result: HighlightSegment[] = [] let lastIndex = 0 for (const ref of allRefs) { if (ref.start < lastIndex) continue if (ref.start > lastIndex) { result.push({ text: text.slice(lastIndex, ref.start) }) } result.push({ text: text.slice(ref.start, ref.end), type: ref.type }) lastIndex = ref.end } if (lastIndex < text.length) { result.push({ text: text.slice(lastIndex) }) } return result }) return {(segment) => {segment.text}} } export function Part(props: MessagePartProps) { const component = createMemo(() => PART_MAPPING[props.part.type]) return ( ) } export interface ToolProps { input: Record metadata: Record tool: string output?: string status?: string hideDetails?: boolean defaultOpen?: boolean forceOpen?: boolean locked?: boolean } export type ToolComponent = Component const state: Record< string, { name: string render?: ToolComponent } > = {} export function registerTool(input: { name: string; render?: ToolComponent }) { state[input.name] = input return input } export function getTool(name: string) { return state[name]?.render } export const ToolRegistry = { register: registerTool, render: getTool, } function ToolFileAccordion(props: { path: string; actions?: JSX.Element; children: JSX.Element }) { const value = createMemo(() => props.path || "tool-file") return (
{`\u202A${getDirectory(props.path)}\u202C`} {getFilename(props.path)}
{props.actions}
{props.children}
) } PART_MAPPING["tool"] = function ToolPartDisplay(props) { const data = useData() const i18n = useI18n() const part = () => props.part as ToolPart if (part().tool === "todowrite" || part().tool === "todoread") return null const hideQuestion = createMemo( () => part().tool === "question" && (part().state.status === "pending" || part().state.status === "running"), ) const emptyInput: Record = {} const emptyMetadata: Record = {} const input = () => part().state?.input ?? emptyInput // @ts-expect-error const partMetadata = () => part().state?.metadata ?? emptyMetadata const taskId = createMemo(() => { if (part().tool !== "task") return const value = partMetadata().sessionId if (typeof value === "string" && value) return value }) const taskHref = createMemo(() => { if (part().tool !== "task") return return sessionLink(taskId(), useLocation().pathname, data.sessionHref) }) const taskSubtitle = createMemo(() => { if (part().tool !== "task") return undefined const value = input().description if (typeof value === "string" && value) return value return taskId() }) const render = createMemo(() => ToolRegistry.render(part().tool) ?? GenericTool) return (
{(error) => { const cleaned = error().replace("Error: ", "") if (part().tool === "question" && cleaned.includes("dismissed this question")) { return (
{i18n.t("ui.messagePart.questions.dismissed")}
) } return ( ) }}
) } export function MessageDivider(props: { label: string }) { return (
{props.label}
) } PART_MAPPING["compaction"] = function CompactionPartDisplay() { const i18n = useI18n() return } PART_MAPPING["text"] = function TextPartDisplay(props) { const data = useData() const i18n = useI18n() const numfmt = createMemo(() => new Intl.NumberFormat(i18n.locale())) const part = () => props.part as TextPart const interrupted = createMemo( () => props.message.role === "assistant" && (props.message as AssistantMessage).error?.name === "MessageAbortedError", ) const model = createMemo(() => { if (props.message.role !== "assistant") return "" const message = props.message as AssistantMessage const match = data.store.provider?.all?.find((p) => p.id === message.providerID) return match?.models?.[message.modelID]?.name ?? message.modelID }) const duration = createMemo(() => { if (props.message.role !== "assistant") return "" const message = props.message as AssistantMessage const completed = message.time.completed const ms = typeof props.turnDurationMs === "number" ? props.turnDurationMs : typeof completed === "number" ? completed - message.time.created : -1 if (!(ms >= 0)) return "" const total = Math.round(ms / 1000) if (total < 60) return i18n.t("ui.message.duration.seconds", { count: numfmt().format(total) }) const minutes = Math.floor(total / 60) const seconds = total % 60 return i18n.t("ui.message.duration.minutesSeconds", { minutes: numfmt().format(minutes), seconds: numfmt().format(seconds), }) }) const meta = createMemo(() => { if (props.message.role !== "assistant") return "" const agent = (props.message as AssistantMessage).agent const items = [ agent ? agent[0]?.toUpperCase() + agent.slice(1) : "", model(), duration(), interrupted() ? i18n.t("ui.message.interrupted") : "", ] return items.filter((x) => !!x).join(" \u00B7 ") }) const displayText = () => (part().text ?? "").trim() const throttledText = createThrottledValue(displayText) const isLastTextPart = createMemo(() => { const last = (data.store.part?.[props.message.id] ?? []) .filter((item): item is TextPart => item?.type === "text" && !!item.text?.trim()) .at(-1) return last?.id === part().id }) const showCopy = createMemo(() => { if (props.message.role !== "assistant") return isLastTextPart() if (props.showAssistantCopyPartID === null) return false if (typeof props.showAssistantCopyPartID === "string") return props.showAssistantCopyPartID === part().id return isLastTextPart() }) const [copied, setCopied] = createSignal(false) const handleCopy = async () => { const content = displayText() if (!content) return await navigator.clipboard.writeText(content) setCopied(true) setTimeout(() => setCopied(false), 2000) } return (
e.preventDefault()} onClick={handleCopy} aria-label={copied() ? i18n.t("ui.message.copied") : i18n.t("ui.message.copyResponse")} /> {meta()}
) } PART_MAPPING["reasoning"] = function ReasoningPartDisplay(props) { const part = () => props.part as ReasoningPart const text = () => part().text.trim() const throttledText = createThrottledValue(text) return (
) } ToolRegistry.register({ name: "read", render(props) { const data = useData() const i18n = useI18n() const args: string[] = [] if (props.input.offset) args.push("offset=" + props.input.offset) if (props.input.limit) args.push("limit=" + props.input.limit) const loaded = createMemo(() => { if (props.status !== "completed") return [] const value = props.metadata.loaded if (!value || !Array.isArray(value)) return [] return value.filter((p): p is string => typeof p === "string") }) return ( <> {(filepath) => (
{i18n.t("ui.tool.loaded")} {relativizeProjectPath(filepath, data.directory)}
)}
) }, }) ToolRegistry.register({ name: "list", render(props) { const i18n = useI18n() return (
) }, }) ToolRegistry.register({ name: "glob", render(props) { const i18n = useI18n() return ( ) }, }) ToolRegistry.register({ name: "grep", render(props) { const i18n = useI18n() const args: string[] = [] if (props.input.pattern) args.push("pattern=" + props.input.pattern) if (props.input.include) args.push("include=" + props.input.include) return (
) }, }) ToolRegistry.register({ name: "webfetch", render(props) { const i18n = useI18n() const pending = createMemo(() => props.status === "pending" || props.status === "running") const url = createMemo(() => { const value = props.input.url if (typeof value !== "string") return "" return value }) return (
} /> ) }, }) ToolRegistry.register({ name: "websearch", render(props) { const i18n = useI18n() const query = createMemo(() => { const value = props.input.query if (typeof value !== "string") return "" return value }) return ( ) }, }) ToolRegistry.register({ name: "codesearch", render(props) { const i18n = useI18n() const query = createMemo(() => { const value = props.input.query if (typeof value !== "string") return "" return value }) return ( ) }, }) ToolRegistry.register({ name: "task", render(props) { const data = useData() const i18n = useI18n() const location = useLocation() const childSessionId = () => props.metadata.sessionId as string | undefined const type = createMemo(() => { const raw = props.input.subagent_type if (typeof raw !== "string" || !raw) return undefined return raw[0]!.toUpperCase() + raw.slice(1) }) const title = createMemo(() => agentTitle(i18n, type())) const subtitle = createMemo(() => { const value = props.input.description if (typeof value === "string" && value) return value return childSessionId() }) const running = createMemo(() => props.status === "pending" || props.status === "running") const href = createMemo(() => sessionLink(childSessionId(), location.pathname, data.sessionHref)) const titleContent = () => const trigger = () => (
{titleContent()} e.stopPropagation()} > {subtitle()} {subtitle()}
) return }, }) ToolRegistry.register({ name: "bash", render(props) { const i18n = useI18n() const pending = () => props.status === "pending" || props.status === "running" const sawPending = pending() const text = createMemo(() => { const cmd = props.input.command ?? props.metadata.command ?? "" const out = stripAnsi(props.output || props.metadata.output || "") return `$ ${cmd}${out ? "\n\n" + out : ""}` }) const [copied, setCopied] = createSignal(false) const handleCopy = async () => { const content = text() if (!content) return await navigator.clipboard.writeText(content) setCopied(true) setTimeout(() => setCopied(false), 2000) } return (
} >
e.preventDefault()} onClick={handleCopy} aria-label={copied() ? i18n.t("ui.message.copied") : i18n.t("ui.message.copy")} />
              {text()}
            
) }, }) ToolRegistry.register({ name: "edit", render(props) { const i18n = useI18n() const fileComponent = useFileComponent() const diagnostics = createMemo(() => getDiagnostics(props.metadata.diagnostics, props.input.filePath)) const path = createMemo(() => props.metadata?.filediff?.file || props.input.filePath || "") const filename = () => getFilename(props.input.filePath ?? "") const pending = () => props.status === "pending" || props.status === "running" return (
{filename()}
{getDirectory(props.input.filePath!)}
} > } >
) }, }) ToolRegistry.register({ name: "write", render(props) { const i18n = useI18n() const fileComponent = useFileComponent() const diagnostics = createMemo(() => getDiagnostics(props.metadata.diagnostics, props.input.filePath)) const path = createMemo(() => props.input.filePath || "") const filename = () => getFilename(props.input.filePath ?? "") const pending = () => props.status === "pending" || props.status === "running" return (
{filename()}
{getDirectory(props.input.filePath!)}
{/* */}
} >
) }, }) interface ApplyPatchFile { filePath: string relativePath: string type: "add" | "update" | "delete" | "move" diff: string before: string after: string additions: number deletions: number movePath?: string } ToolRegistry.register({ name: "apply_patch", render(props) { const i18n = useI18n() const fileComponent = useFileComponent() const files = createMemo(() => (props.metadata.files ?? []) as ApplyPatchFile[]) const pending = createMemo(() => props.status === "pending" || props.status === "running") const single = createMemo(() => { const list = files() if (list.length !== 1) return return list[0] }) const [expanded, setExpanded] = createSignal([]) let seeded = false createEffect(() => { const list = files() if (list.length === 0) return if (seeded) return seeded = true setExpanded(list.filter((f) => f.type !== "delete").map((f) => f.filePath)) }) const subtitle = createMemo(() => { const count = files().length if (count === 0) return "" return `${count} ${i18n.t(count > 1 ? "ui.common.file.other" : "ui.common.file.one")}` }) return ( 0}> setExpanded(Array.isArray(value) ? value : value ? [value] : [])} > {(file) => { const active = createMemo(() => expanded().includes(file.filePath)) const [visible, setVisible] = createSignal(false) createEffect(() => { if (!active()) { setVisible(false) return } requestAnimationFrame(() => { if (!active()) return setVisible(true) }) }) return (
{`\u202A${getDirectory(file.relativePath)}\u202C`} {getFilename(file.relativePath)}
{i18n.t("ui.patch.action.created")} {i18n.t("ui.patch.action.deleted")} {i18n.t("ui.patch.action.moved")}
) }}
} >
{getFilename(single()!.relativePath)}
{getDirectory(single()!.relativePath)}
} > {i18n.t("ui.patch.action.created")} {i18n.t("ui.patch.action.deleted")} {i18n.t("ui.patch.action.moved")} } >
) }, }) ToolRegistry.register({ name: "todowrite", render(props) { const i18n = useI18n() const todos = createMemo(() => { const meta = props.metadata?.todos if (Array.isArray(meta)) return meta const input = props.input.todos if (Array.isArray(input)) return input return [] }) const subtitle = createMemo(() => { const list = todos() if (list.length === 0) return "" return `${list.filter((t: Todo) => t.status === "completed").length}/${list.length}` }) return (
{(todo: Todo) => ( {todo.content} )}
) }, }) ToolRegistry.register({ name: "question", render(props) { const i18n = useI18n() const questions = createMemo(() => (props.input.questions ?? []) as QuestionInfo[]) const answers = createMemo(() => (props.metadata.answers ?? []) as QuestionAnswer[]) const completed = createMemo(() => answers().length > 0) const subtitle = createMemo(() => { const count = questions().length if (count === 0) return "" if (completed()) return i18n.t("ui.question.subtitle.answered", { count }) return `${count} ${i18n.t(count > 1 ? "ui.common.question.other" : "ui.common.question.one")}` }) return (
{(q, i) => { const answer = () => answers()[i()] ?? [] return (
{q.question}
{answer().join(", ") || i18n.t("ui.question.answer.none")}
) }}
) }, }) ToolRegistry.register({ name: "skill", render(props) { const i18n = useI18n() const title = createMemo(() => props.input.name || i18n.t("ui.tool.skill")) const running = createMemo(() => props.status === "pending" || props.status === "running") const titleContent = () => const trigger = () => (
{titleContent()}
) return }, })