1package messages
  2
  3import (
  4	"encoding/json"
  5	"fmt"
  6	"strings"
  7	"time"
  8
  9	"github.com/charmbracelet/crush/internal/fsext"
 10	"github.com/charmbracelet/crush/internal/llm/agent"
 11	"github.com/charmbracelet/crush/internal/llm/tools"
 12	"github.com/charmbracelet/crush/internal/tui/components/core"
 13	"github.com/charmbracelet/crush/internal/tui/highlight"
 14	"github.com/charmbracelet/crush/internal/tui/styles"
 15	"github.com/charmbracelet/lipgloss/v2"
 16	"github.com/charmbracelet/lipgloss/v2/tree"
 17	"github.com/charmbracelet/x/ansi"
 18)
 19
 20// responseContextHeight limits the number of lines displayed in tool output
 21const responseContextHeight = 10
 22
 23// renderer defines the interface for tool-specific rendering implementations
 24type renderer interface {
 25	// Render returns the complete (already styled) toolโcall view, not
 26	// including the outer border.
 27	Render(v *toolCallCmp) string
 28}
 29
 30// rendererFactory creates new renderer instances
 31type rendererFactory func() renderer
 32
 33// renderRegistry manages the mapping of tool names to their renderers
 34type renderRegistry map[string]rendererFactory
 35
 36// register adds a new renderer factory to the registry
 37func (rr renderRegistry) register(name string, f rendererFactory) { rr[name] = f }
 38
 39// lookup retrieves a renderer for the given tool name, falling back to generic renderer
 40func (rr renderRegistry) lookup(name string) renderer {
 41	if f, ok := rr[name]; ok {
 42		return f()
 43	}
 44	return genericRenderer{} // sensible fallback
 45}
 46
 47// registry holds all registered tool renderers
 48var registry = renderRegistry{}
 49
 50// baseRenderer provides common functionality for all tool renderers
 51type baseRenderer struct{}
 52
 53// paramBuilder helps construct parameter lists for tool headers
 54type paramBuilder struct {
 55	args []string
 56}
 57
 58// newParamBuilder creates a new parameter builder
 59func newParamBuilder() *paramBuilder {
 60	return ¶mBuilder{args: make([]string, 0)}
 61}
 62
 63// addMain adds the main parameter (first argument)
 64func (pb *paramBuilder) addMain(value string) *paramBuilder {
 65	if value != "" {
 66		pb.args = append(pb.args, value)
 67	}
 68	return pb
 69}
 70
 71// addKeyValue adds a key-value pair parameter
 72func (pb *paramBuilder) addKeyValue(key, value string) *paramBuilder {
 73	if value != "" {
 74		pb.args = append(pb.args, key, value)
 75	}
 76	return pb
 77}
 78
 79// addFlag adds a boolean flag parameter
 80func (pb *paramBuilder) addFlag(key string, value bool) *paramBuilder {
 81	if value {
 82		pb.args = append(pb.args, key, "true")
 83	}
 84	return pb
 85}
 86
 87// build returns the final parameter list
 88func (pb *paramBuilder) build() []string {
 89	return pb.args
 90}
 91
 92// renderWithParams provides a common rendering pattern for tools with parameters
 93func (br baseRenderer) renderWithParams(v *toolCallCmp, toolName string, args []string, contentRenderer func() string) string {
 94	width := v.textWidth()
 95	if v.isNested {
 96		width -= 4 // Adjust for nested tool call indentation
 97	}
 98	header := br.makeHeader(v, toolName, width, args...)
 99	if v.isNested {
100		return v.style().Render(header)
101	}
102	if res, done := earlyState(header, v); done {
103		return res
104	}
105	body := contentRenderer()
106	return joinHeaderBody(header, body)
107}
108
109// unmarshalParams safely unmarshal JSON parameters
110func (br baseRenderer) unmarshalParams(input string, target any) error {
111	return json.Unmarshal([]byte(input), target)
112}
113
114// makeHeader builds the tool call header with status icon and parameters for a nested tool call.
115func (br baseRenderer) makeNestedHeader(v *toolCallCmp, tool string, width int, params ...string) string {
116	t := styles.CurrentTheme()
117	icon := t.S().Base.Foreground(t.GreenDark).Render(styles.ToolPending)
118	if v.result.ToolCallID != "" {
119		if v.result.IsError {
120			icon = t.S().Base.Foreground(t.RedDark).Render(styles.ToolError)
121		} else {
122			icon = t.S().Base.Foreground(t.Green).Render(styles.ToolSuccess)
123		}
124	} else if v.cancelled {
125		icon = t.S().Muted.Render(styles.ToolPending)
126	}
127	tool = t.S().Base.Foreground(t.FgHalfMuted).Render(tool) + " "
128	prefix := fmt.Sprintf("%s %s ", icon, tool)
129	return prefix + renderParamList(true, width-lipgloss.Width(tool), params...)
130}
131
132// makeHeader builds "<Tool>: param (key=value)" and truncates as needed.
133func (br baseRenderer) makeHeader(v *toolCallCmp, tool string, width int, params ...string) string {
134	if v.isNested {
135		return br.makeNestedHeader(v, tool, width, params...)
136	}
137	t := styles.CurrentTheme()
138	icon := t.S().Base.Foreground(t.GreenDark).Render(styles.ToolPending)
139	if v.result.ToolCallID != "" {
140		if v.result.IsError {
141			icon = t.S().Base.Foreground(t.RedDark).Render(styles.ToolError)
142		} else {
143			icon = t.S().Base.Foreground(t.Green).Render(styles.ToolSuccess)
144		}
145	} else if v.cancelled {
146		icon = t.S().Muted.Render(styles.ToolPending)
147	}
148	tool = t.S().Base.Foreground(t.Blue).Render(tool)
149	prefix := fmt.Sprintf("%s %s ", icon, tool)
150	return prefix + renderParamList(false, width-lipgloss.Width(prefix), params...)
151}
152
153// renderError provides consistent error rendering
154func (br baseRenderer) renderError(v *toolCallCmp, message string) string {
155	t := styles.CurrentTheme()
156	header := br.makeHeader(v, prettifyToolName(v.call.Name), v.textWidth(), "")
157	errorTag := t.S().Base.Padding(0, 1).Background(t.Red).Foreground(t.White).Render("ERROR")
158	message = t.S().Base.Foreground(t.FgHalfMuted).Render(v.fit(message, v.textWidth()-3-lipgloss.Width(errorTag))) // -2 for padding and space
159	return joinHeaderBody(header, errorTag+" "+message)
160}
161
162// Register tool renderers
163func init() {
164	registry.register(tools.BashToolName, func() renderer { return bashRenderer{} })
165	registry.register(tools.ViewToolName, func() renderer { return viewRenderer{} })
166	registry.register(tools.EditToolName, func() renderer { return editRenderer{} })
167	registry.register(tools.WriteToolName, func() renderer { return writeRenderer{} })
168	registry.register(tools.FetchToolName, func() renderer { return fetchRenderer{} })
169	registry.register(tools.GlobToolName, func() renderer { return globRenderer{} })
170	registry.register(tools.GrepToolName, func() renderer { return grepRenderer{} })
171	registry.register(tools.LSToolName, func() renderer { return lsRenderer{} })
172	registry.register(tools.SourcegraphToolName, func() renderer { return sourcegraphRenderer{} })
173	registry.register(tools.DiagnosticsToolName, func() renderer { return diagnosticsRenderer{} })
174	registry.register(agent.AgentToolName, func() renderer { return agentRenderer{} })
175}
176
177// -----------------------------------------------------------------------------
178//  Generic renderer
179// -----------------------------------------------------------------------------
180
181// genericRenderer handles unknown tool types with basic parameter display
182type genericRenderer struct {
183	baseRenderer
184}
185
186// Render displays the tool call with its raw input and plain content output
187func (gr genericRenderer) Render(v *toolCallCmp) string {
188	return gr.renderWithParams(v, prettifyToolName(v.call.Name), []string{v.call.Input}, func() string {
189		return renderPlainContent(v, v.result.Content)
190	})
191}
192
193// -----------------------------------------------------------------------------
194//  Bash renderer
195// -----------------------------------------------------------------------------
196
197// bashRenderer handles bash command execution display
198type bashRenderer struct {
199	baseRenderer
200}
201
202// Render displays the bash command with sanitized newlines and plain output
203func (br bashRenderer) Render(v *toolCallCmp) string {
204	var params tools.BashParams
205	if err := br.unmarshalParams(v.call.Input, ¶ms); err != nil {
206		return br.renderError(v, "Invalid bash parameters")
207	}
208
209	cmd := strings.ReplaceAll(params.Command, "\n", " ")
210	cmd = strings.ReplaceAll(cmd, "\t", "    ")
211	args := newParamBuilder().addMain(cmd).build()
212
213	return br.renderWithParams(v, "Bash", args, func() string {
214		if v.result.Content == tools.BashNoOutput {
215			return ""
216		}
217		return renderPlainContent(v, v.result.Content)
218	})
219}
220
221// -----------------------------------------------------------------------------
222//  View renderer
223// -----------------------------------------------------------------------------
224
225// viewRenderer handles file viewing with syntax highlighting and line numbers
226type viewRenderer struct {
227	baseRenderer
228}
229
230// Render displays file content with optional limit and offset parameters
231func (vr viewRenderer) Render(v *toolCallCmp) string {
232	var params tools.ViewParams
233	if err := vr.unmarshalParams(v.call.Input, ¶ms); err != nil {
234		return vr.renderError(v, "Invalid view parameters")
235	}
236
237	file := fsext.PrettyPath(params.FilePath)
238	args := newParamBuilder().
239		addMain(file).
240		addKeyValue("limit", formatNonZero(params.Limit)).
241		addKeyValue("offset", formatNonZero(params.Offset)).
242		build()
243
244	return vr.renderWithParams(v, "View", args, func() string {
245		var meta tools.ViewResponseMetadata
246		if err := vr.unmarshalParams(v.result.Metadata, &meta); err != nil {
247			return renderPlainContent(v, v.result.Content)
248		}
249		return renderCodeContent(v, meta.FilePath, meta.Content, params.Offset)
250	})
251}
252
253// formatNonZero returns string representation of non-zero integers, empty string for zero
254func formatNonZero(value int) string {
255	if value == 0 {
256		return ""
257	}
258	return fmt.Sprintf("%d", value)
259}
260
261// -----------------------------------------------------------------------------
262//  Edit renderer
263// -----------------------------------------------------------------------------
264
265// editRenderer handles file editing with diff visualization
266type editRenderer struct {
267	baseRenderer
268}
269
270// Render displays the edited file with a formatted diff of changes
271func (er editRenderer) Render(v *toolCallCmp) string {
272	t := styles.CurrentTheme()
273	var params tools.EditParams
274	var args []string
275	if err := er.unmarshalParams(v.call.Input, ¶ms); err == nil {
276		file := fsext.PrettyPath(params.FilePath)
277		args = newParamBuilder().addMain(file).build()
278	}
279
280	return er.renderWithParams(v, "Edit", args, func() string {
281		var meta tools.EditResponseMetadata
282		if err := er.unmarshalParams(v.result.Metadata, &meta); err != nil {
283			return renderPlainContent(v, v.result.Content)
284		}
285
286		formatter := core.DiffFormatter().
287			Before(fsext.PrettyPath(params.FilePath), meta.OldContent).
288			After(fsext.PrettyPath(params.FilePath), meta.NewContent).
289			Width(v.textWidth() - 2) // -2 for padding
290		if v.textWidth() > 120 {
291			formatter = formatter.Split()
292		}
293		// add a message to the bottom if the content was truncated
294		formatted := formatter.String()
295		if lipgloss.Height(formatted) > responseContextHeight {
296			contentLines := strings.Split(formatted, "\n")
297			truncateMessage := t.S().Muted.
298				Background(t.BgBaseLighter).
299				PaddingLeft(2).
300				Width(v.textWidth() - 4).
301				Render(fmt.Sprintf("โฆ (%d lines)", len(contentLines)-responseContextHeight))
302			formatted = strings.Join(contentLines[:responseContextHeight], "\n") + "\n" + truncateMessage
303		}
304		return formatted
305	})
306}
307
308// -----------------------------------------------------------------------------
309//  Write renderer
310// -----------------------------------------------------------------------------
311
312// writeRenderer handles file writing with syntax-highlighted content preview
313type writeRenderer struct {
314	baseRenderer
315}
316
317// Render displays the file being written with syntax highlighting
318func (wr writeRenderer) Render(v *toolCallCmp) string {
319	var params tools.WriteParams
320	var args []string
321	var file string
322	if err := wr.unmarshalParams(v.call.Input, ¶ms); err == nil {
323		file = fsext.PrettyPath(params.FilePath)
324		args = newParamBuilder().addMain(file).build()
325	}
326
327	return wr.renderWithParams(v, "Write", args, func() string {
328		return renderCodeContent(v, file, params.Content, 0)
329	})
330}
331
332// -----------------------------------------------------------------------------
333//  Fetch renderer
334// -----------------------------------------------------------------------------
335
336// fetchRenderer handles URL fetching with format-specific content display
337type fetchRenderer struct {
338	baseRenderer
339}
340
341// Render displays the fetched URL with format and timeout parameters
342func (fr fetchRenderer) Render(v *toolCallCmp) string {
343	var params tools.FetchParams
344	var args []string
345	if err := fr.unmarshalParams(v.call.Input, ¶ms); err == nil {
346		args = newParamBuilder().
347			addMain(params.URL).
348			addKeyValue("format", params.Format).
349			addKeyValue("timeout", formatTimeout(params.Timeout)).
350			build()
351	}
352
353	return fr.renderWithParams(v, "Fetch", args, func() string {
354		file := fr.getFileExtension(params.Format)
355		return renderCodeContent(v, file, v.result.Content, 0)
356	})
357}
358
359// getFileExtension returns appropriate file extension for syntax highlighting
360func (fr fetchRenderer) getFileExtension(format string) string {
361	switch format {
362	case "text":
363		return "fetch.txt"
364	case "html":
365		return "fetch.html"
366	default:
367		return "fetch.md"
368	}
369}
370
371// formatTimeout converts timeout seconds to duration string
372func formatTimeout(timeout int) string {
373	if timeout == 0 {
374		return ""
375	}
376	return (time.Duration(timeout) * time.Second).String()
377}
378
379// -----------------------------------------------------------------------------
380//  Glob renderer
381// -----------------------------------------------------------------------------
382
383// globRenderer handles file pattern matching with path filtering
384type globRenderer struct {
385	baseRenderer
386}
387
388// Render displays the glob pattern with optional path parameter
389func (gr globRenderer) Render(v *toolCallCmp) string {
390	var params tools.GlobParams
391	var args []string
392	if err := gr.unmarshalParams(v.call.Input, ¶ms); err == nil {
393		args = newParamBuilder().
394			addMain(params.Pattern).
395			addKeyValue("path", params.Path).
396			build()
397	}
398
399	return gr.renderWithParams(v, "Glob", args, func() string {
400		return renderPlainContent(v, v.result.Content)
401	})
402}
403
404// -----------------------------------------------------------------------------
405//  Grep renderer
406// -----------------------------------------------------------------------------
407
408// grepRenderer handles content searching with pattern matching options
409type grepRenderer struct {
410	baseRenderer
411}
412
413// Render displays the search pattern with path, include, and literal text options
414func (gr grepRenderer) Render(v *toolCallCmp) string {
415	var params tools.GrepParams
416	var args []string
417	if err := gr.unmarshalParams(v.call.Input, ¶ms); err == nil {
418		args = newParamBuilder().
419			addMain(params.Pattern).
420			addKeyValue("path", params.Path).
421			addKeyValue("include", params.Include).
422			addFlag("literal", params.LiteralText).
423			build()
424	}
425
426	return gr.renderWithParams(v, "Grep", args, func() string {
427		return renderPlainContent(v, v.result.Content)
428	})
429}
430
431// -----------------------------------------------------------------------------
432//  LS renderer
433// -----------------------------------------------------------------------------
434
435// lsRenderer handles directory listing with default path handling
436type lsRenderer struct {
437	baseRenderer
438}
439
440// Render displays the directory path, defaulting to current directory
441func (lr lsRenderer) Render(v *toolCallCmp) string {
442	var params tools.LSParams
443	var args []string
444	if err := lr.unmarshalParams(v.call.Input, ¶ms); err == nil {
445		path := params.Path
446		if path == "" {
447			path = "."
448		}
449		path = fsext.PrettyPath(path)
450
451		args = newParamBuilder().addMain(path).build()
452	}
453
454	return lr.renderWithParams(v, "List", args, func() string {
455		return renderPlainContent(v, v.result.Content)
456	})
457}
458
459// -----------------------------------------------------------------------------
460//  Sourcegraph renderer
461// -----------------------------------------------------------------------------
462
463// sourcegraphRenderer handles code search with count and context options
464type sourcegraphRenderer struct {
465	baseRenderer
466}
467
468// Render displays the search query with optional count and context window parameters
469func (sr sourcegraphRenderer) Render(v *toolCallCmp) string {
470	var params tools.SourcegraphParams
471	var args []string
472	if err := sr.unmarshalParams(v.call.Input, ¶ms); err == nil {
473		args = newParamBuilder().
474			addMain(params.Query).
475			addKeyValue("count", formatNonZero(params.Count)).
476			addKeyValue("context", formatNonZero(params.ContextWindow)).
477			build()
478	}
479
480	return sr.renderWithParams(v, "Sourcegraph", args, func() string {
481		return renderPlainContent(v, v.result.Content)
482	})
483}
484
485// -----------------------------------------------------------------------------
486//  Diagnostics renderer
487// -----------------------------------------------------------------------------
488
489// diagnosticsRenderer handles project-wide diagnostic information
490type diagnosticsRenderer struct {
491	baseRenderer
492}
493
494// Render displays project diagnostics with plain content formatting
495func (dr diagnosticsRenderer) Render(v *toolCallCmp) string {
496	args := newParamBuilder().addMain("project").build()
497
498	return dr.renderWithParams(v, "Diagnostics", args, func() string {
499		return renderPlainContent(v, v.result.Content)
500	})
501}
502
503// -----------------------------------------------------------------------------
504//  Task renderer
505// -----------------------------------------------------------------------------
506
507// agentRenderer handles project-wide diagnostic information
508type agentRenderer struct {
509	baseRenderer
510}
511
512func RoundedEnumerator(children tree.Children, index int) string {
513	if children.Length()-1 == index {
514		return " โฐโโ"
515	}
516	return " โโโ"
517}
518
519// Render displays agent task parameters and result content
520func (tr agentRenderer) Render(v *toolCallCmp) string {
521	t := styles.CurrentTheme()
522	var params agent.AgentParams
523	tr.unmarshalParams(v.call.Input, ¶ms)
524
525	prompt := params.Prompt
526	prompt = strings.ReplaceAll(prompt, "\n", " ")
527
528	header := tr.makeHeader(v, "Agent", v.textWidth())
529	if res, done := earlyState(header, v); v.cancelled && done {
530		return res
531	}
532	taskTag := t.S().Base.Padding(0, 1).MarginLeft(1).Background(t.BlueLight).Foreground(t.White).Render("Task")
533	remainingWidth := v.textWidth() - lipgloss.Width(header) - lipgloss.Width(taskTag) - 2 // -2 for padding
534	prompt = t.S().Muted.Width(remainingWidth).Render(prompt)
535	header = lipgloss.JoinVertical(
536		lipgloss.Left,
537		header,
538		"",
539		lipgloss.JoinHorizontal(
540			lipgloss.Left,
541			taskTag,
542			" ",
543			prompt,
544		),
545	)
546	childTools := tree.Root(header)
547
548	for _, call := range v.nestedToolCalls {
549		childTools.Child(call.View())
550	}
551	parts := []string{
552		childTools.Enumerator(RoundedEnumerator).String(),
553	}
554
555	if v.result.ToolCallID == "" {
556		v.spinning = true
557		parts = append(parts, "", v.anim.View())
558	} else {
559		v.spinning = false
560	}
561
562	header = lipgloss.JoinVertical(
563		lipgloss.Left,
564		parts...,
565	)
566
567	if v.result.ToolCallID == "" {
568		return header
569	}
570
571	body := renderPlainContent(v, v.result.Content)
572	return joinHeaderBody(header, body)
573}
574
575// renderParamList renders params, params[0] (params[1]=params[2] ....)
576func renderParamList(nested bool, paramsWidth int, params ...string) string {
577	t := styles.CurrentTheme()
578	if len(params) == 0 {
579		return ""
580	}
581	mainParam := params[0]
582	if paramsWidth >= 0 && lipgloss.Width(mainParam) > paramsWidth {
583		mainParam = ansi.Truncate(mainParam, paramsWidth, "โฆ")
584	}
585
586	if len(params) == 1 {
587		if nested {
588			return t.S().Muted.Render(mainParam)
589		}
590		return t.S().Subtle.Render(mainParam)
591	}
592	otherParams := params[1:]
593	// create pairs of key/value
594	// if odd number of params, the last one is a key without value
595	if len(otherParams)%2 != 0 {
596		otherParams = append(otherParams, "")
597	}
598	parts := make([]string, 0, len(otherParams)/2)
599	for i := 0; i < len(otherParams); i += 2 {
600		key := otherParams[i]
601		value := otherParams[i+1]
602		if value == "" {
603			continue
604		}
605		parts = append(parts, fmt.Sprintf("%s=%s", key, value))
606	}
607
608	partsRendered := strings.Join(parts, ", ")
609	remainingWidth := paramsWidth - lipgloss.Width(partsRendered) - 3 // count for " ()"
610	if remainingWidth < 30 {
611		if nested {
612			return t.S().Muted.Render(mainParam)
613		}
614		// No space for the params, just show the main
615		return t.S().Subtle.Render(mainParam)
616	}
617
618	if len(parts) > 0 {
619		mainParam = fmt.Sprintf("%s (%s)", mainParam, strings.Join(parts, ", "))
620	}
621
622	if nested {
623		return t.S().Muted.Render(ansi.Truncate(mainParam, paramsWidth, "โฆ"))
624	}
625	return t.S().Subtle.Render(ansi.Truncate(mainParam, paramsWidth, "โฆ"))
626}
627
628// earlyState returns immediatelyโrendered error/cancelled/ongoing states.
629func earlyState(header string, v *toolCallCmp) (string, bool) {
630	t := styles.CurrentTheme()
631	message := ""
632	switch {
633	case v.result.IsError:
634		message = v.renderToolError()
635	case v.cancelled:
636		message = t.S().Base.Foreground(t.FgSubtle).Render("Canceled.")
637	case v.result.ToolCallID == "":
638		message = t.S().Base.Foreground(t.FgSubtle).Render("Waiting for tool to start...")
639	default:
640		return "", false
641	}
642
643	message = t.S().Base.PaddingLeft(2).Render(message)
644	return lipgloss.JoinVertical(lipgloss.Left, header, "", message), true
645}
646
647func joinHeaderBody(header, body string) string {
648	t := styles.CurrentTheme()
649	if body == "" {
650		return header
651	}
652	body = t.S().Base.PaddingLeft(2).Render(body)
653	return lipgloss.JoinVertical(lipgloss.Left, header, "", body)
654}
655
656func renderPlainContent(v *toolCallCmp, content string) string {
657	t := styles.CurrentTheme()
658	content = strings.ReplaceAll(content, "\r\n", "\n") // Normalize line endings
659	content = strings.ReplaceAll(content, "\t", "    ") // Replace tabs with spaces
660	content = strings.TrimSpace(content)
661	lines := strings.Split(content, "\n")
662
663	width := v.textWidth() - 2 // -2 for left padding
664	var out []string
665	for i, ln := range lines {
666		if i >= responseContextHeight {
667			break
668		}
669		ln = escapeContent(ln)
670		ln = " " + ln // left padding
671		if len(ln) > width {
672			ln = v.fit(ln, width)
673		}
674		out = append(out, t.S().Muted.
675			Width(width).
676			Background(t.BgBaseLighter).
677			Render(ln))
678	}
679
680	if len(lines) > responseContextHeight {
681		out = append(out, t.S().Muted.
682			Background(t.BgBaseLighter).
683			Width(width).
684			Render(fmt.Sprintf("โฆ (%d lines)", len(lines)-responseContextHeight)))
685	}
686
687	return strings.Join(out, "\n")
688}
689
690func pad(v any, width int) string {
691	s := fmt.Sprintf("%v", v)
692	w := ansi.StringWidth(s)
693	if w >= width {
694		return s
695	}
696	return strings.Repeat(" ", width-w) + s
697}
698
699func renderCodeContent(v *toolCallCmp, path, content string, offset int) string {
700	t := styles.CurrentTheme()
701	content = strings.ReplaceAll(content, "\r\n", "\n") // Normalize line endings
702	content = strings.ReplaceAll(content, "\t", "    ") // Replace tabs with spaces
703	truncated := truncateHeight(content, responseContextHeight)
704
705	lines := strings.Split(truncated, "\n")
706	for i, ln := range lines {
707		lines[i] = escapeContent(ln)
708	}
709
710	highlighted, _ := highlight.SyntaxHighlight(strings.Join(lines, "\n"), path, t.BgBase)
711	lines = strings.Split(highlighted, "\n")
712
713	if len(strings.Split(content, "\n")) > responseContextHeight {
714		lines = append(lines, t.S().Muted.
715			Background(t.BgBase).
716			Render(fmt.Sprintf(" โฆ(%d lines)", len(strings.Split(content, "\n"))-responseContextHeight)))
717	}
718
719	maxLineNumber := len(lines) + offset
720	padding := lipgloss.Width(fmt.Sprintf("%d", maxLineNumber))
721	for i, ln := range lines {
722		num := t.S().Base.
723			Foreground(t.FgMuted).
724			Background(t.BgBase).
725			PaddingRight(1).
726			PaddingLeft(1).
727			Render(pad(i+1+offset, padding))
728		w := v.textWidth() - 10 - lipgloss.Width(num) // -4 for left padding
729		lines[i] = lipgloss.JoinHorizontal(lipgloss.Left,
730			num,
731			t.S().Base.
732				PaddingLeft(1).
733				Render(v.fit(ln, w-1)))
734	}
735
736	return lipgloss.JoinVertical(lipgloss.Left, lines...)
737}
738
739func (v *toolCallCmp) renderToolError() string {
740	t := styles.CurrentTheme()
741	err := strings.ReplaceAll(v.result.Content, "\n", " ")
742	errTag := t.S().Base.Padding(0, 1).Background(t.Red).Foreground(t.White).Render("ERROR")
743	err = fmt.Sprintf("%s %s", errTag, t.S().Base.Foreground(t.FgHalfMuted).Render(v.fit(err, v.textWidth()-2-lipgloss.Width(errTag))))
744	return err
745}
746
747func truncateHeight(s string, h int) string {
748	lines := strings.Split(s, "\n")
749	if len(lines) > h {
750		return strings.Join(lines[:h], "\n")
751	}
752	return s
753}
754
755func prettifyToolName(name string) string {
756	switch name {
757	case agent.AgentToolName:
758		return "Agent"
759	case tools.BashToolName:
760		return "Bash"
761	case tools.EditToolName:
762		return "Edit"
763	case tools.FetchToolName:
764		return "Fetch"
765	case tools.GlobToolName:
766		return "Glob"
767	case tools.GrepToolName:
768		return "Grep"
769	case tools.LSToolName:
770		return "List"
771	case tools.SourcegraphToolName:
772		return "Sourcegraph"
773	case tools.ViewToolName:
774		return "View"
775	case tools.WriteToolName:
776		return "Write"
777	default:
778		return name
779	}
780}
781
782// escapeContent replaces control characters with their Unicode Control Picture
783// representations to ensure they are displayed correctly in the UI.
784func escapeContent(content string) string {
785	var sb strings.Builder
786	for _, r := range content {
787		switch {
788		case r >= 0 && r <= 0x1f: // Control characters 0x00-0x1F
789			sb.WriteRune('\u2400' + r)
790		case r == ansi.DEL:
791			sb.WriteRune('\u2421')
792		default:
793			sb.WriteRune(r)
794		}
795	}
796	return sb.String()
797}