renderer.go

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