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(_ *toolCallCmp, tool string, width int, params ...string) string {
116 t := styles.CurrentTheme()
117 tool = t.S().Base.Foreground(t.FgHalfMuted).Render(tool) + " "
118 return tool + renderParamList(true, width-lipgloss.Width(tool), params...)
119}
120
121// makeHeader builds "<Tool>: param (key=value)" and truncates as needed.
122func (br baseRenderer) makeHeader(v *toolCallCmp, tool string, width int, params ...string) string {
123 if v.isNested {
124 return br.makeNestedHeader(v, tool, width, params...)
125 }
126 t := styles.CurrentTheme()
127 icon := t.S().Base.Foreground(t.GreenDark).Render(styles.ToolPending)
128 if v.result.ToolCallID != "" {
129 if v.result.IsError {
130 icon = t.S().Base.Foreground(t.RedDark).Render(styles.ToolError)
131 } else {
132 icon = t.S().Base.Foreground(t.Green).Render(styles.ToolSuccess)
133 }
134 } else if v.cancelled {
135 icon = t.S().Muted.Render(styles.ToolPending)
136 }
137 tool = t.S().Base.Foreground(t.Blue).Render(tool)
138 prefix := fmt.Sprintf("%s %s ", icon, tool)
139 return prefix + renderParamList(false, width-lipgloss.Width(prefix), params...)
140}
141
142// renderError provides consistent error rendering
143func (br baseRenderer) renderError(v *toolCallCmp, message string) string {
144 t := styles.CurrentTheme()
145 header := br.makeHeader(v, prettifyToolName(v.call.Name), v.textWidth(), "")
146 errorTag := t.S().Base.Padding(0, 1).Background(t.Red).Foreground(t.White).Render("ERROR")
147 message = t.S().Base.Foreground(t.FgHalfMuted).Render(v.fit(message, v.textWidth()-3-lipgloss.Width(errorTag))) // -2 for padding and space
148 return joinHeaderBody(header, errorTag+" "+message)
149}
150
151// Register tool renderers
152func init() {
153 registry.register(tools.BashToolName, func() renderer { return bashRenderer{} })
154 registry.register(tools.ViewToolName, func() renderer { return viewRenderer{} })
155 registry.register(tools.EditToolName, func() renderer { return editRenderer{} })
156 registry.register(tools.WriteToolName, func() renderer { return writeRenderer{} })
157 registry.register(tools.FetchToolName, func() renderer { return fetchRenderer{} })
158 registry.register(tools.GlobToolName, func() renderer { return globRenderer{} })
159 registry.register(tools.GrepToolName, func() renderer { return grepRenderer{} })
160 registry.register(tools.LSToolName, func() renderer { return lsRenderer{} })
161 registry.register(tools.SourcegraphToolName, func() renderer { return sourcegraphRenderer{} })
162 registry.register(tools.DiagnosticsToolName, func() renderer { return diagnosticsRenderer{} })
163 registry.register(agent.AgentToolName, func() renderer { return agentRenderer{} })
164}
165
166// -----------------------------------------------------------------------------
167// Generic renderer
168// -----------------------------------------------------------------------------
169
170// genericRenderer handles unknown tool types with basic parameter display
171type genericRenderer struct {
172 baseRenderer
173}
174
175// Render displays the tool call with its raw input and plain content output
176func (gr genericRenderer) Render(v *toolCallCmp) string {
177 return gr.renderWithParams(v, prettifyToolName(v.call.Name), []string{v.call.Input}, func() string {
178 return renderPlainContent(v, v.result.Content)
179 })
180}
181
182// -----------------------------------------------------------------------------
183// Bash renderer
184// -----------------------------------------------------------------------------
185
186// bashRenderer handles bash command execution display
187type bashRenderer struct {
188 baseRenderer
189}
190
191// Render displays the bash command with sanitized newlines and plain output
192func (br bashRenderer) Render(v *toolCallCmp) string {
193 var params tools.BashParams
194 if err := br.unmarshalParams(v.call.Input, ¶ms); err != nil {
195 return br.renderError(v, "Invalid bash parameters")
196 }
197
198 cmd := strings.ReplaceAll(params.Command, "\n", " ")
199 args := newParamBuilder().addMain(cmd).build()
200
201 return br.renderWithParams(v, "Bash", args, func() string {
202 if v.result.Content == tools.BashNoOutput {
203 return ""
204 }
205 return renderPlainContent(v, v.result.Content)
206 })
207}
208
209// -----------------------------------------------------------------------------
210// View renderer
211// -----------------------------------------------------------------------------
212
213// viewRenderer handles file viewing with syntax highlighting and line numbers
214type viewRenderer struct {
215 baseRenderer
216}
217
218// Render displays file content with optional limit and offset parameters
219func (vr viewRenderer) Render(v *toolCallCmp) string {
220 var params tools.ViewParams
221 if err := vr.unmarshalParams(v.call.Input, ¶ms); err != nil {
222 return vr.renderError(v, "Invalid view parameters")
223 }
224
225 file := fsext.PrettyPath(params.FilePath)
226 args := newParamBuilder().
227 addMain(file).
228 addKeyValue("limit", formatNonZero(params.Limit)).
229 addKeyValue("offset", formatNonZero(params.Offset)).
230 build()
231
232 return vr.renderWithParams(v, "View", args, func() string {
233 var meta tools.ViewResponseMetadata
234 if err := vr.unmarshalParams(v.result.Metadata, &meta); err != nil {
235 return renderPlainContent(v, v.result.Content)
236 }
237 return renderCodeContent(v, meta.FilePath, meta.Content, params.Offset)
238 })
239}
240
241// formatNonZero returns string representation of non-zero integers, empty string for zero
242func formatNonZero(value int) string {
243 if value == 0 {
244 return ""
245 }
246 return fmt.Sprintf("%d", value)
247}
248
249// -----------------------------------------------------------------------------
250// Edit renderer
251// -----------------------------------------------------------------------------
252
253// editRenderer handles file editing with diff visualization
254type editRenderer struct {
255 baseRenderer
256}
257
258// Render displays the edited file with a formatted diff of changes
259func (er editRenderer) Render(v *toolCallCmp) string {
260 t := styles.CurrentTheme()
261 var params tools.EditParams
262 var args []string
263 if err := er.unmarshalParams(v.call.Input, ¶ms); err == nil {
264 file := fsext.PrettyPath(params.FilePath)
265 args = newParamBuilder().addMain(file).build()
266 }
267
268 return er.renderWithParams(v, "Edit", args, func() string {
269 var meta tools.EditResponseMetadata
270 if err := er.unmarshalParams(v.result.Metadata, &meta); err != nil {
271 return renderPlainContent(v, v.result.Content)
272 }
273
274 formatter := core.DiffFormatter().
275 Before(fsext.PrettyPath(params.FilePath), meta.OldContent).
276 After(fsext.PrettyPath(params.FilePath), meta.NewContent).
277 Width(v.textWidth() - 2) // -2 for padding
278 if v.textWidth() > 120 {
279 formatter = formatter.Split()
280 }
281 // add a message to the bottom if the content was truncated
282 formatted := formatter.String()
283 if lipgloss.Height(formatted) > responseContextHeight {
284 contentLines := strings.Split(formatted, "\n")
285 truncateMessage := t.S().Muted.
286 Background(t.BgBaseLighter).
287 PaddingLeft(2).
288 Width(v.textWidth() - 4).
289 Render(fmt.Sprintf("… (%d lines)", len(contentLines)-responseContextHeight))
290 formatted = strings.Join(contentLines[:responseContextHeight], "\n") + "\n" + truncateMessage
291 }
292 return formatted
293 })
294}
295
296// -----------------------------------------------------------------------------
297// Write renderer
298// -----------------------------------------------------------------------------
299
300// writeRenderer handles file writing with syntax-highlighted content preview
301type writeRenderer struct {
302 baseRenderer
303}
304
305// Render displays the file being written with syntax highlighting
306func (wr writeRenderer) Render(v *toolCallCmp) string {
307 var params tools.WriteParams
308 var args []string
309 var file string
310 if err := wr.unmarshalParams(v.call.Input, ¶ms); err == nil {
311 file = fsext.PrettyPath(params.FilePath)
312 args = newParamBuilder().addMain(file).build()
313 }
314
315 return wr.renderWithParams(v, "Write", args, func() string {
316 return renderCodeContent(v, file, params.Content, 0)
317 })
318}
319
320// -----------------------------------------------------------------------------
321// Fetch renderer
322// -----------------------------------------------------------------------------
323
324// fetchRenderer handles URL fetching with format-specific content display
325type fetchRenderer struct {
326 baseRenderer
327}
328
329// Render displays the fetched URL with format and timeout parameters
330func (fr fetchRenderer) Render(v *toolCallCmp) string {
331 var params tools.FetchParams
332 var args []string
333 if err := fr.unmarshalParams(v.call.Input, ¶ms); err == nil {
334 args = newParamBuilder().
335 addMain(params.URL).
336 addKeyValue("format", params.Format).
337 addKeyValue("timeout", formatTimeout(params.Timeout)).
338 build()
339 }
340
341 return fr.renderWithParams(v, "Fetch", args, func() string {
342 file := fr.getFileExtension(params.Format)
343 return renderCodeContent(v, file, v.result.Content, 0)
344 })
345}
346
347// getFileExtension returns appropriate file extension for syntax highlighting
348func (fr fetchRenderer) getFileExtension(format string) string {
349 switch format {
350 case "text":
351 return "fetch.txt"
352 case "html":
353 return "fetch.html"
354 default:
355 return "fetch.md"
356 }
357}
358
359// formatTimeout converts timeout seconds to duration string
360func formatTimeout(timeout int) string {
361 if timeout == 0 {
362 return ""
363 }
364 return (time.Duration(timeout) * time.Second).String()
365}
366
367// -----------------------------------------------------------------------------
368// Glob renderer
369// -----------------------------------------------------------------------------
370
371// globRenderer handles file pattern matching with path filtering
372type globRenderer struct {
373 baseRenderer
374}
375
376// Render displays the glob pattern with optional path parameter
377func (gr globRenderer) Render(v *toolCallCmp) string {
378 var params tools.GlobParams
379 var args []string
380 if err := gr.unmarshalParams(v.call.Input, ¶ms); err == nil {
381 args = newParamBuilder().
382 addMain(params.Pattern).
383 addKeyValue("path", params.Path).
384 build()
385 }
386
387 return gr.renderWithParams(v, "Glob", args, func() string {
388 return renderPlainContent(v, v.result.Content)
389 })
390}
391
392// -----------------------------------------------------------------------------
393// Grep renderer
394// -----------------------------------------------------------------------------
395
396// grepRenderer handles content searching with pattern matching options
397type grepRenderer struct {
398 baseRenderer
399}
400
401// Render displays the search pattern with path, include, and literal text options
402func (gr grepRenderer) Render(v *toolCallCmp) string {
403 var params tools.GrepParams
404 var args []string
405 if err := gr.unmarshalParams(v.call.Input, ¶ms); err == nil {
406 args = newParamBuilder().
407 addMain(params.Pattern).
408 addKeyValue("path", params.Path).
409 addKeyValue("include", params.Include).
410 addFlag("literal", params.LiteralText).
411 build()
412 }
413
414 return gr.renderWithParams(v, "Grep", args, func() string {
415 return renderPlainContent(v, v.result.Content)
416 })
417}
418
419// -----------------------------------------------------------------------------
420// LS renderer
421// -----------------------------------------------------------------------------
422
423// lsRenderer handles directory listing with default path handling
424type lsRenderer struct {
425 baseRenderer
426}
427
428// Render displays the directory path, defaulting to current directory
429func (lr lsRenderer) Render(v *toolCallCmp) string {
430 var params tools.LSParams
431 var args []string
432 if err := lr.unmarshalParams(v.call.Input, ¶ms); err == nil {
433 path := params.Path
434 if path == "" {
435 path = "."
436 }
437 path = fsext.PrettyPath(path)
438
439 args = newParamBuilder().addMain(path).build()
440 }
441
442 return lr.renderWithParams(v, "List", args, func() string {
443 return renderPlainContent(v, v.result.Content)
444 })
445}
446
447// -----------------------------------------------------------------------------
448// Sourcegraph renderer
449// -----------------------------------------------------------------------------
450
451// sourcegraphRenderer handles code search with count and context options
452type sourcegraphRenderer struct {
453 baseRenderer
454}
455
456// Render displays the search query with optional count and context window parameters
457func (sr sourcegraphRenderer) Render(v *toolCallCmp) string {
458 var params tools.SourcegraphParams
459 var args []string
460 if err := sr.unmarshalParams(v.call.Input, ¶ms); err == nil {
461 args = newParamBuilder().
462 addMain(params.Query).
463 addKeyValue("count", formatNonZero(params.Count)).
464 addKeyValue("context", formatNonZero(params.ContextWindow)).
465 build()
466 }
467
468 return sr.renderWithParams(v, "Sourcegraph", args, func() string {
469 return renderPlainContent(v, v.result.Content)
470 })
471}
472
473// -----------------------------------------------------------------------------
474// Diagnostics renderer
475// -----------------------------------------------------------------------------
476
477// diagnosticsRenderer handles project-wide diagnostic information
478type diagnosticsRenderer struct {
479 baseRenderer
480}
481
482// Render displays project diagnostics with plain content formatting
483func (dr diagnosticsRenderer) Render(v *toolCallCmp) string {
484 args := newParamBuilder().addMain("project").build()
485
486 return dr.renderWithParams(v, "Diagnostics", args, func() string {
487 return renderPlainContent(v, v.result.Content)
488 })
489}
490
491// -----------------------------------------------------------------------------
492// Task renderer
493// -----------------------------------------------------------------------------
494
495// agentRenderer handles project-wide diagnostic information
496type agentRenderer struct {
497 baseRenderer
498}
499
500func RoundedEnumerator(children tree.Children, index int) string {
501 if children.Length()-1 == index {
502 return " ╰──"
503 }
504 return " ├──"
505}
506
507// Render displays agent task parameters and result content
508func (tr agentRenderer) Render(v *toolCallCmp) string {
509 t := styles.CurrentTheme()
510 var params agent.AgentParams
511 tr.unmarshalParams(v.call.Input, ¶ms)
512
513 prompt := params.Prompt
514 prompt = strings.ReplaceAll(prompt, "\n", " ")
515
516 header := tr.makeHeader(v, "Agent", v.textWidth())
517 if res, done := earlyState(header, v); v.cancelled && done {
518 return res
519 }
520 taskTag := t.S().Base.Padding(0, 1).MarginLeft(1).Background(t.BlueLight).Foreground(t.White).Render("Task")
521 remainingWidth := v.textWidth() - lipgloss.Width(header) - lipgloss.Width(taskTag) - 2 // -2 for padding
522 prompt = t.S().Muted.Width(remainingWidth).Render(prompt)
523 header = lipgloss.JoinVertical(
524 lipgloss.Left,
525 header,
526 "",
527 lipgloss.JoinHorizontal(
528 lipgloss.Left,
529 taskTag,
530 " ",
531 prompt,
532 ),
533 )
534 childTools := tree.Root(header)
535
536 for _, call := range v.nestedToolCalls {
537 childTools.Child(call.View())
538 }
539 parts := []string{
540 childTools.Enumerator(RoundedEnumerator).String(),
541 }
542
543 if v.result.ToolCallID == "" {
544 v.spinning = true
545 parts = append(parts, v.anim.View())
546 } else {
547 v.spinning = false
548 }
549
550 header = lipgloss.JoinVertical(
551 lipgloss.Left,
552 parts...,
553 )
554
555 if v.result.ToolCallID == "" {
556 return header
557 }
558
559 body := renderPlainContent(v, v.result.Content)
560 return joinHeaderBody(header, body)
561}
562
563// renderParamList renders params, params[0] (params[1]=params[2] ....)
564func renderParamList(nested bool, paramsWidth int, params ...string) string {
565 t := styles.CurrentTheme()
566 if len(params) == 0 {
567 return ""
568 }
569 mainParam := params[0]
570 if len(mainParam) > paramsWidth {
571 mainParam = mainParam[:paramsWidth-3] + "…"
572 }
573
574 if len(params) == 1 {
575 if nested {
576 return t.S().Muted.Render(mainParam)
577 }
578 return t.S().Subtle.Render(mainParam)
579 }
580 otherParams := params[1:]
581 // create pairs of key/value
582 // if odd number of params, the last one is a key without value
583 if len(otherParams)%2 != 0 {
584 otherParams = append(otherParams, "")
585 }
586 parts := make([]string, 0, len(otherParams)/2)
587 for i := 0; i < len(otherParams); i += 2 {
588 key := otherParams[i]
589 value := otherParams[i+1]
590 if value == "" {
591 continue
592 }
593 parts = append(parts, fmt.Sprintf("%s=%s", key, value))
594 }
595
596 partsRendered := strings.Join(parts, ", ")
597 remainingWidth := paramsWidth - lipgloss.Width(partsRendered) - 3 // count for " ()"
598 if remainingWidth < 30 {
599 if nested {
600 return t.S().Muted.Render(mainParam)
601 }
602 // No space for the params, just show the main
603 return t.S().Subtle.Render(mainParam)
604 }
605
606 if len(parts) > 0 {
607 mainParam = fmt.Sprintf("%s (%s)", mainParam, strings.Join(parts, ", "))
608 }
609
610 if nested {
611 return t.S().Muted.Render(ansi.Truncate(mainParam, paramsWidth, "…"))
612 }
613 return t.S().Subtle.Render(ansi.Truncate(mainParam, paramsWidth, "…"))
614}
615
616// earlyState returns immediately‑rendered error/cancelled/ongoing states.
617func earlyState(header string, v *toolCallCmp) (string, bool) {
618 t := styles.CurrentTheme()
619 message := ""
620 switch {
621 case v.result.IsError:
622 message = v.renderToolError()
623 case v.cancelled:
624 message = t.S().Base.Foreground(t.FgSubtle).Render("Canceled.")
625 case v.result.ToolCallID == "":
626 message = t.S().Base.Foreground(t.FgSubtle).Render("Waiting for tool to start...")
627 default:
628 return "", false
629 }
630
631 message = t.S().Base.PaddingLeft(2).Render(message)
632 return lipgloss.JoinVertical(lipgloss.Left, header, "", message), true
633}
634
635func joinHeaderBody(header, body string) string {
636 t := styles.CurrentTheme()
637 body = t.S().Base.PaddingLeft(2).Render(body)
638 return lipgloss.JoinVertical(lipgloss.Left, header, "", body, "")
639}
640
641func renderPlainContent(v *toolCallCmp, content string) string {
642 t := styles.CurrentTheme()
643 content = strings.TrimSpace(content)
644 lines := strings.Split(content, "\n")
645
646 width := v.textWidth() - 2 // -2 for left padding
647 var out []string
648 for i, ln := range lines {
649 if i >= responseContextHeight {
650 break
651 }
652 ln = " " + ln // left padding
653 if len(ln) > width {
654 ln = v.fit(ln, width)
655 }
656 out = append(out, t.S().Muted.
657 Width(width).
658 Background(t.BgBaseLighter).
659 Render(ln))
660 }
661
662 if len(lines) > responseContextHeight {
663 out = append(out, t.S().Muted.
664 Background(t.BgBaseLighter).
665 Width(width).
666 Render(fmt.Sprintf("… (%d lines)", len(lines)-responseContextHeight)))
667 }
668 return strings.Join(out, "\n")
669}
670
671func pad(v any, width int) string {
672 s := fmt.Sprintf("%v", v)
673 w := ansi.StringWidth(s)
674 if w >= width {
675 return s
676 }
677 return strings.Repeat(" ", width-w) + s
678}
679
680func renderCodeContent(v *toolCallCmp, path, content string, offset int) string {
681 t := styles.CurrentTheme()
682 truncated := truncateHeight(content, responseContextHeight)
683
684 highlighted, _ := highlight.SyntaxHighlight(truncated, path, t.BgBase)
685 lines := strings.Split(highlighted, "\n")
686
687 if len(strings.Split(content, "\n")) > responseContextHeight {
688 lines = append(lines, t.S().Muted.
689 Background(t.BgBase).
690 Render(fmt.Sprintf(" …(%d lines)", len(strings.Split(content, "\n"))-responseContextHeight)))
691 }
692
693 maxLineNumber := len(lines) + offset
694 padding := lipgloss.Width(fmt.Sprintf("%d", maxLineNumber))
695 for i, ln := range lines {
696 num := t.S().Base.
697 Foreground(t.FgMuted).
698 Background(t.BgBase).
699 PaddingRight(1).
700 PaddingLeft(1).
701 Render(pad(i+1+offset, padding))
702 w := v.textWidth() - 10 - lipgloss.Width(num) // -4 for left padding
703 lines[i] = lipgloss.JoinHorizontal(lipgloss.Left,
704 num,
705 t.S().Base.
706 PaddingLeft(1).
707 Render(v.fit(ln, w-1)))
708 }
709 return lipgloss.JoinVertical(lipgloss.Left, lines...)
710}
711
712func (v *toolCallCmp) renderToolError() string {
713 t := styles.CurrentTheme()
714 err := strings.ReplaceAll(v.result.Content, "\n", " ")
715 errTag := t.S().Base.Padding(0, 1).Background(t.Red).Foreground(t.White).Render("ERROR")
716 err = fmt.Sprintf("%s %s", errTag, t.S().Base.Foreground(t.FgHalfMuted).Render(v.fit(err, v.textWidth()-2-lipgloss.Width(errTag))))
717 return err
718}
719
720func truncateHeight(s string, h int) string {
721 lines := strings.Split(s, "\n")
722 if len(lines) > h {
723 return strings.Join(lines[:h], "\n")
724 }
725 return s
726}
727
728func prettifyToolName(name string) string {
729 switch name {
730 case agent.AgentToolName:
731 return "Agent"
732 case tools.BashToolName:
733 return "Bash"
734 case tools.EditToolName:
735 return "Edit"
736 case tools.FetchToolName:
737 return "Fetch"
738 case tools.GlobToolName:
739 return "Glob"
740 case tools.GrepToolName:
741 return "Grep"
742 case tools.LSToolName:
743 return "List"
744 case tools.SourcegraphToolName:
745 return "Sourcegraph"
746 case tools.ViewToolName:
747 return "View"
748 case tools.WriteToolName:
749 return "Write"
750 default:
751 return name
752 }
753}