mirror of
https://github.com/sst/opencode.git
synced 2025-07-23 15:55:03 +00:00
711 lines
17 KiB
Go
711 lines
17 KiB
Go
package chat
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"slices"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/charmbracelet/lipgloss/v2"
|
|
"github.com/charmbracelet/lipgloss/v2/compat"
|
|
"github.com/charmbracelet/x/ansi"
|
|
"github.com/muesli/reflow/truncate"
|
|
"github.com/sst/opencode-sdk-go"
|
|
"github.com/sst/opencode/internal/app"
|
|
"github.com/sst/opencode/internal/components/diff"
|
|
"github.com/sst/opencode/internal/styles"
|
|
"github.com/sst/opencode/internal/theme"
|
|
"github.com/sst/opencode/internal/util"
|
|
"golang.org/x/text/cases"
|
|
"golang.org/x/text/language"
|
|
)
|
|
|
|
type blockRenderer struct {
|
|
textColor compat.AdaptiveColor
|
|
border bool
|
|
borderColor *compat.AdaptiveColor
|
|
borderColorRight bool
|
|
paddingTop int
|
|
paddingBottom int
|
|
paddingLeft int
|
|
paddingRight int
|
|
marginTop int
|
|
marginBottom int
|
|
}
|
|
|
|
type renderingOption func(*blockRenderer)
|
|
|
|
func WithTextColor(color compat.AdaptiveColor) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.textColor = color
|
|
}
|
|
}
|
|
|
|
func WithNoBorder() renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.border = false
|
|
}
|
|
}
|
|
|
|
func WithBorderColor(color compat.AdaptiveColor) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.borderColor = &color
|
|
}
|
|
}
|
|
|
|
func WithBorderColorRight(color compat.AdaptiveColor) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.borderColorRight = true
|
|
c.borderColor = &color
|
|
}
|
|
}
|
|
|
|
func WithMarginTop(padding int) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.marginTop = padding
|
|
}
|
|
}
|
|
|
|
func WithMarginBottom(padding int) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.marginBottom = padding
|
|
}
|
|
}
|
|
|
|
func WithPadding(padding int) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.paddingTop = padding
|
|
c.paddingBottom = padding
|
|
c.paddingLeft = padding
|
|
c.paddingRight = padding
|
|
}
|
|
}
|
|
|
|
func WithPaddingLeft(padding int) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.paddingLeft = padding
|
|
}
|
|
}
|
|
|
|
func WithPaddingRight(padding int) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.paddingRight = padding
|
|
}
|
|
}
|
|
|
|
func WithPaddingTop(padding int) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.paddingTop = padding
|
|
}
|
|
}
|
|
|
|
func WithPaddingBottom(padding int) renderingOption {
|
|
return func(c *blockRenderer) {
|
|
c.paddingBottom = padding
|
|
}
|
|
}
|
|
|
|
func renderContentBlock(
|
|
app *app.App,
|
|
content string,
|
|
width int,
|
|
options ...renderingOption,
|
|
) string {
|
|
t := theme.CurrentTheme()
|
|
renderer := &blockRenderer{
|
|
textColor: t.TextMuted(),
|
|
border: true,
|
|
paddingTop: 1,
|
|
paddingBottom: 1,
|
|
paddingLeft: 2,
|
|
paddingRight: 2,
|
|
}
|
|
for _, option := range options {
|
|
option(renderer)
|
|
}
|
|
|
|
borderColor := t.BackgroundPanel()
|
|
if renderer.borderColor != nil {
|
|
borderColor = *renderer.borderColor
|
|
}
|
|
|
|
style := styles.NewStyle().
|
|
Foreground(renderer.textColor).
|
|
Background(t.BackgroundPanel()).
|
|
PaddingTop(renderer.paddingTop).
|
|
PaddingBottom(renderer.paddingBottom).
|
|
PaddingLeft(renderer.paddingLeft).
|
|
PaddingRight(renderer.paddingRight).
|
|
AlignHorizontal(lipgloss.Left)
|
|
|
|
if renderer.border {
|
|
style = style.
|
|
BorderStyle(lipgloss.ThickBorder()).
|
|
BorderLeft(true).
|
|
BorderRight(true).
|
|
BorderLeftForeground(borderColor).
|
|
BorderLeftBackground(t.Background()).
|
|
BorderRightForeground(t.BackgroundPanel()).
|
|
BorderRightBackground(t.Background())
|
|
|
|
if renderer.borderColorRight {
|
|
style = style.
|
|
BorderLeftBackground(t.Background()).
|
|
BorderLeftForeground(t.BackgroundPanel()).
|
|
BorderRightForeground(borderColor).
|
|
BorderRightBackground(t.Background())
|
|
}
|
|
|
|
}
|
|
|
|
content = style.Render(content)
|
|
if renderer.marginTop > 0 {
|
|
for range renderer.marginTop {
|
|
content = "\n" + content
|
|
}
|
|
}
|
|
if renderer.marginBottom > 0 {
|
|
for range renderer.marginBottom {
|
|
content = content + "\n"
|
|
}
|
|
}
|
|
|
|
return content
|
|
}
|
|
|
|
func renderText(
|
|
app *app.App,
|
|
message opencode.MessageUnion,
|
|
text string,
|
|
author string,
|
|
showToolDetails bool,
|
|
width int,
|
|
extra string,
|
|
toolCalls ...opencode.ToolPart,
|
|
) string {
|
|
t := theme.CurrentTheme()
|
|
|
|
var ts time.Time
|
|
backgroundColor := t.BackgroundPanel()
|
|
var content string
|
|
switch casted := message.(type) {
|
|
case opencode.AssistantMessage:
|
|
ts = time.UnixMilli(int64(casted.Time.Created))
|
|
content = util.ToMarkdown(text, width, backgroundColor)
|
|
case opencode.UserMessage:
|
|
ts = time.UnixMilli(int64(casted.Time.Created))
|
|
base := styles.NewStyle().Foreground(t.Text()).Background(backgroundColor)
|
|
words := strings.Fields(text)
|
|
for i, word := range words {
|
|
if strings.HasPrefix(word, "@") {
|
|
words[i] = base.Foreground(t.Secondary()).Render(word + " ")
|
|
} else {
|
|
words[i] = base.Render(word + " ")
|
|
}
|
|
}
|
|
text = strings.Join(words, "")
|
|
text = ansi.WordwrapWc(text, width-6, " -")
|
|
content = base.Width(width - 6).Render(text)
|
|
}
|
|
|
|
timestamp := ts.
|
|
Local().
|
|
Format("02 Jan 2006 03:04 PM")
|
|
if time.Now().Format("02 Jan 2006") == timestamp[:11] {
|
|
// don't show the date if it's today
|
|
timestamp = timestamp[12:]
|
|
}
|
|
info := fmt.Sprintf("%s (%s)", author, timestamp)
|
|
info = styles.NewStyle().Foreground(t.TextMuted()).Render(info)
|
|
|
|
if !showToolDetails && toolCalls != nil && len(toolCalls) > 0 {
|
|
content = content + "\n\n"
|
|
for _, toolCall := range toolCalls {
|
|
title := renderToolTitle(toolCall, width)
|
|
style := styles.NewStyle()
|
|
if toolCall.State.Status == opencode.ToolPartStateStatusError {
|
|
style = style.Foreground(t.Error())
|
|
}
|
|
title = style.Render(title)
|
|
title = "∟ " + title + "\n"
|
|
content = content + title
|
|
}
|
|
}
|
|
|
|
sections := []string{content, info}
|
|
if extra != "" {
|
|
sections = append(sections, "\n"+extra)
|
|
}
|
|
content = strings.Join(sections, "\n")
|
|
|
|
switch message.(type) {
|
|
case opencode.UserMessage:
|
|
return renderContentBlock(
|
|
app,
|
|
content,
|
|
width,
|
|
WithTextColor(t.Text()),
|
|
WithBorderColorRight(t.Secondary()),
|
|
)
|
|
case opencode.AssistantMessage:
|
|
return renderContentBlock(
|
|
app,
|
|
content,
|
|
width,
|
|
WithBorderColor(t.Accent()),
|
|
)
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func renderToolDetails(
|
|
app *app.App,
|
|
toolCall opencode.ToolPart,
|
|
width int,
|
|
) string {
|
|
ignoredTools := []string{"todoread"}
|
|
if slices.Contains(ignoredTools, toolCall.Tool) {
|
|
return ""
|
|
}
|
|
|
|
if toolCall.State.Status == opencode.ToolPartStateStatusPending {
|
|
title := renderToolTitle(toolCall, width)
|
|
return renderContentBlock(app, title, width)
|
|
}
|
|
|
|
var result *string
|
|
if toolCall.State.Output != "" {
|
|
result = &toolCall.State.Output
|
|
}
|
|
|
|
toolInputMap := make(map[string]any)
|
|
if toolCall.State.Input != nil {
|
|
value := toolCall.State.Input
|
|
if m, ok := value.(map[string]any); ok {
|
|
toolInputMap = m
|
|
keys := make([]string, 0, len(toolInputMap))
|
|
for key := range toolInputMap {
|
|
keys = append(keys, key)
|
|
}
|
|
slices.Sort(keys)
|
|
}
|
|
}
|
|
|
|
body := ""
|
|
t := theme.CurrentTheme()
|
|
backgroundColor := t.BackgroundPanel()
|
|
borderColor := t.BackgroundPanel()
|
|
defaultStyle := styles.NewStyle().Background(backgroundColor).Width(width - 6).Render
|
|
|
|
if toolCall.State.Metadata != nil {
|
|
metadata := toolCall.State.Metadata.(map[string]any)
|
|
switch toolCall.Tool {
|
|
case "read":
|
|
var preview any
|
|
if metadata != nil {
|
|
preview = metadata["preview"]
|
|
}
|
|
if preview != nil && toolInputMap["filePath"] != nil {
|
|
filename := toolInputMap["filePath"].(string)
|
|
body = preview.(string)
|
|
body = util.RenderFile(filename, body, width, util.WithTruncate(6))
|
|
}
|
|
case "edit":
|
|
if filename, ok := toolInputMap["filePath"].(string); ok {
|
|
var diffField any
|
|
if metadata != nil {
|
|
diffField = metadata["diff"]
|
|
}
|
|
if diffField != nil {
|
|
patch := diffField.(string)
|
|
var formattedDiff string
|
|
if width < 120 {
|
|
formattedDiff, _ = diff.FormatUnifiedDiff(
|
|
filename,
|
|
patch,
|
|
diff.WithWidth(width-2),
|
|
)
|
|
} else {
|
|
formattedDiff, _ = diff.FormatDiff(
|
|
filename,
|
|
patch,
|
|
diff.WithWidth(width-2),
|
|
)
|
|
}
|
|
body = strings.TrimSpace(formattedDiff)
|
|
style := styles.NewStyle().
|
|
Background(backgroundColor).
|
|
Foreground(t.TextMuted()).
|
|
Padding(1, 2).
|
|
Width(width - 4)
|
|
|
|
if diagnostics := renderDiagnostics(metadata, filename, backgroundColor, width-6); diagnostics != "" {
|
|
diagnostics = style.Render(diagnostics)
|
|
body += "\n" + diagnostics
|
|
}
|
|
|
|
title := renderToolTitle(toolCall, width)
|
|
title = style.Render(title)
|
|
content := title + "\n" + body
|
|
content = renderContentBlock(
|
|
app,
|
|
content,
|
|
width,
|
|
WithPadding(0),
|
|
WithBorderColor(borderColor),
|
|
)
|
|
return content
|
|
}
|
|
}
|
|
case "write":
|
|
if filename, ok := toolInputMap["filePath"].(string); ok {
|
|
if content, ok := toolInputMap["content"].(string); ok {
|
|
body = util.RenderFile(filename, content, width)
|
|
if diagnostics := renderDiagnostics(metadata, filename, backgroundColor, width-4); diagnostics != "" {
|
|
body += "\n\n" + diagnostics
|
|
}
|
|
}
|
|
}
|
|
case "bash":
|
|
stdout := metadata["stdout"]
|
|
if stdout != nil {
|
|
command := toolInputMap["command"].(string)
|
|
body = fmt.Sprintf("```console\n> %s\n%s```", command, stdout)
|
|
body = util.ToMarkdown(body, width, backgroundColor)
|
|
}
|
|
case "webfetch":
|
|
if format, ok := toolInputMap["format"].(string); ok && result != nil {
|
|
body = *result
|
|
body = util.TruncateHeight(body, 10)
|
|
if format == "html" || format == "markdown" {
|
|
body = util.ToMarkdown(body, width, backgroundColor)
|
|
}
|
|
}
|
|
case "todowrite":
|
|
todos := metadata["todos"]
|
|
if todos != nil {
|
|
for _, item := range todos.([]any) {
|
|
todo := item.(map[string]any)
|
|
content := todo["content"].(string)
|
|
switch todo["status"] {
|
|
case "completed":
|
|
body += fmt.Sprintf("- [x] %s\n", content)
|
|
case "cancelled":
|
|
// strike through cancelled todo
|
|
body += fmt.Sprintf("- [~] ~~%s~~\n", content)
|
|
case "in_progress":
|
|
// highlight in progress todo
|
|
body += fmt.Sprintf("- [ ] `%s`\n", content)
|
|
default:
|
|
body += fmt.Sprintf("- [ ] %s\n", content)
|
|
}
|
|
}
|
|
body = util.ToMarkdown(body, width, backgroundColor)
|
|
}
|
|
case "task":
|
|
summary := metadata["summary"]
|
|
if summary != nil {
|
|
toolcalls := summary.([]any)
|
|
steps := []string{}
|
|
for _, item := range toolcalls {
|
|
data, _ := json.Marshal(item)
|
|
var toolCall opencode.ToolPart
|
|
_ = json.Unmarshal(data, &toolCall)
|
|
step := renderToolTitle(toolCall, width)
|
|
step = "∟ " + step
|
|
steps = append(steps, step)
|
|
}
|
|
body = strings.Join(steps, "\n")
|
|
}
|
|
body = defaultStyle(body)
|
|
default:
|
|
if result == nil {
|
|
empty := ""
|
|
result = &empty
|
|
}
|
|
body = *result
|
|
body = util.TruncateHeight(body, 10)
|
|
body = defaultStyle(body)
|
|
}
|
|
}
|
|
|
|
error := ""
|
|
if toolCall.State.Status == opencode.ToolPartStateStatusError {
|
|
error = toolCall.State.Error
|
|
}
|
|
|
|
if error != "" {
|
|
body = styles.NewStyle().
|
|
Width(width - 6).
|
|
Foreground(t.Error()).
|
|
Background(backgroundColor).
|
|
Render(error)
|
|
}
|
|
|
|
if body == "" && error == "" && result != nil {
|
|
body = *result
|
|
body = util.TruncateHeight(body, 10)
|
|
body = defaultStyle(body)
|
|
}
|
|
|
|
if body == "" {
|
|
body = defaultStyle("")
|
|
}
|
|
|
|
title := renderToolTitle(toolCall, width)
|
|
content := title + "\n\n" + body
|
|
return renderContentBlock(app, content, width, WithBorderColor(borderColor))
|
|
}
|
|
|
|
func renderToolName(name string) string {
|
|
switch name {
|
|
case "webfetch":
|
|
return "Fetch"
|
|
default:
|
|
normalizedName := name
|
|
if after, ok := strings.CutPrefix(name, "opencode_"); ok {
|
|
normalizedName = after
|
|
}
|
|
return cases.Title(language.Und).String(normalizedName)
|
|
}
|
|
}
|
|
|
|
func getTodoPhase(metadata map[string]any) string {
|
|
todos, ok := metadata["todos"].([]any)
|
|
if !ok || len(todos) == 0 {
|
|
return "Plan"
|
|
}
|
|
|
|
counts := map[string]int{"pending": 0, "completed": 0}
|
|
for _, item := range todos {
|
|
if todo, ok := item.(map[string]any); ok {
|
|
if status, ok := todo["status"].(string); ok {
|
|
counts[status]++
|
|
}
|
|
}
|
|
}
|
|
|
|
total := len(todos)
|
|
switch {
|
|
case counts["pending"] == total:
|
|
return "Creating plan"
|
|
case counts["completed"] == total:
|
|
return "Completing plan"
|
|
default:
|
|
return "Updating plan"
|
|
}
|
|
}
|
|
|
|
func getTodoTitle(toolCall opencode.ToolPart) string {
|
|
if toolCall.State.Status == opencode.ToolPartStateStatusCompleted {
|
|
if metadata, ok := toolCall.State.Metadata.(map[string]any); ok {
|
|
return getTodoPhase(metadata)
|
|
}
|
|
}
|
|
return "Plan"
|
|
}
|
|
|
|
func renderToolTitle(
|
|
toolCall opencode.ToolPart,
|
|
width int,
|
|
) string {
|
|
if toolCall.State.Status == opencode.ToolPartStateStatusPending {
|
|
title := renderToolAction(toolCall.Tool)
|
|
return styles.NewStyle().Width(width - 6).Render(title)
|
|
}
|
|
|
|
toolArgs := ""
|
|
toolArgsMap := make(map[string]any)
|
|
if toolCall.State.Input != nil {
|
|
value := toolCall.State.Input
|
|
if m, ok := value.(map[string]any); ok {
|
|
toolArgsMap = m
|
|
|
|
keys := make([]string, 0, len(toolArgsMap))
|
|
for key := range toolArgsMap {
|
|
keys = append(keys, key)
|
|
}
|
|
slices.Sort(keys)
|
|
firstKey := ""
|
|
if len(keys) > 0 {
|
|
firstKey = keys[0]
|
|
}
|
|
|
|
toolArgs = renderArgs(&toolArgsMap, firstKey)
|
|
}
|
|
}
|
|
|
|
title := renderToolName(toolCall.Tool)
|
|
switch toolCall.Tool {
|
|
case "read":
|
|
toolArgs = renderArgs(&toolArgsMap, "filePath")
|
|
title = fmt.Sprintf("%s %s", title, toolArgs)
|
|
case "edit", "write":
|
|
if filename, ok := toolArgsMap["filePath"].(string); ok {
|
|
title = fmt.Sprintf("%s %s", title, util.Relative(filename))
|
|
}
|
|
case "bash", "task":
|
|
if description, ok := toolArgsMap["description"].(string); ok {
|
|
title = fmt.Sprintf("%s %s", title, description)
|
|
}
|
|
case "webfetch":
|
|
toolArgs = renderArgs(&toolArgsMap, "url")
|
|
title = fmt.Sprintf("%s %s", title, toolArgs)
|
|
case "todowrite":
|
|
title = getTodoTitle(toolCall)
|
|
case "todoread":
|
|
return "Plan"
|
|
default:
|
|
toolName := renderToolName(toolCall.Tool)
|
|
title = fmt.Sprintf("%s %s", toolName, toolArgs)
|
|
}
|
|
|
|
title = truncate.StringWithTail(title, uint(width-6), "...")
|
|
return title
|
|
}
|
|
|
|
func renderToolAction(name string) string {
|
|
switch name {
|
|
case "task":
|
|
return "Planning..."
|
|
case "bash":
|
|
return "Writing command..."
|
|
case "edit":
|
|
return "Preparing edit..."
|
|
case "webfetch":
|
|
return "Fetching from the web..."
|
|
case "glob":
|
|
return "Finding files..."
|
|
case "grep":
|
|
return "Searching content..."
|
|
case "list":
|
|
return "Listing directory..."
|
|
case "read":
|
|
return "Reading file..."
|
|
case "write":
|
|
return "Preparing write..."
|
|
case "todowrite", "todoread":
|
|
return "Planning..."
|
|
case "patch":
|
|
return "Preparing patch..."
|
|
}
|
|
return "Working..."
|
|
}
|
|
|
|
func renderArgs(args *map[string]any, titleKey string) string {
|
|
if args == nil || len(*args) == 0 {
|
|
return ""
|
|
}
|
|
|
|
keys := make([]string, 0, len(*args))
|
|
for key := range *args {
|
|
keys = append(keys, key)
|
|
}
|
|
slices.Sort(keys)
|
|
|
|
title := ""
|
|
parts := []string{}
|
|
for _, key := range keys {
|
|
value := (*args)[key]
|
|
if value == nil {
|
|
continue
|
|
}
|
|
if key == "filePath" || key == "path" {
|
|
value = util.Relative(value.(string))
|
|
}
|
|
if key == titleKey {
|
|
title = fmt.Sprintf("%s", value)
|
|
continue
|
|
}
|
|
parts = append(parts, fmt.Sprintf("%s=%v", key, value))
|
|
}
|
|
if len(parts) == 0 {
|
|
return title
|
|
}
|
|
return fmt.Sprintf("%s (%s)", title, strings.Join(parts, ", "))
|
|
}
|
|
|
|
// Diagnostic represents an LSP diagnostic
|
|
type Diagnostic struct {
|
|
Range struct {
|
|
Start struct {
|
|
Line int `json:"line"`
|
|
Character int `json:"character"`
|
|
} `json:"start"`
|
|
} `json:"range"`
|
|
Severity int `json:"severity"`
|
|
Message string `json:"message"`
|
|
}
|
|
|
|
// renderDiagnostics formats LSP diagnostics for display in the TUI
|
|
func renderDiagnostics(
|
|
metadata map[string]any,
|
|
filePath string,
|
|
backgroundColor compat.AdaptiveColor,
|
|
width int,
|
|
) string {
|
|
if diagnosticsData, ok := metadata["diagnostics"].(map[string]any); ok {
|
|
if fileDiagnostics, ok := diagnosticsData[filePath].([]any); ok {
|
|
var errorDiagnostics []string
|
|
for _, diagInterface := range fileDiagnostics {
|
|
diagMap, ok := diagInterface.(map[string]any)
|
|
if !ok {
|
|
continue
|
|
}
|
|
// Parse the diagnostic
|
|
var diag Diagnostic
|
|
diagBytes, err := json.Marshal(diagMap)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
if err := json.Unmarshal(diagBytes, &diag); err != nil {
|
|
continue
|
|
}
|
|
// Only show error diagnostics (severity === 1)
|
|
if diag.Severity != 1 {
|
|
continue
|
|
}
|
|
line := diag.Range.Start.Line + 1 // 1-based
|
|
column := diag.Range.Start.Character + 1 // 1-based
|
|
errorDiagnostics = append(
|
|
errorDiagnostics,
|
|
fmt.Sprintf("Error [%d:%d] %s", line, column, diag.Message),
|
|
)
|
|
}
|
|
if len(errorDiagnostics) == 0 {
|
|
return ""
|
|
}
|
|
t := theme.CurrentTheme()
|
|
var result strings.Builder
|
|
for _, diagnostic := range errorDiagnostics {
|
|
if result.Len() > 0 {
|
|
result.WriteString("\n\n")
|
|
}
|
|
diagnostic = ansi.WordwrapWc(diagnostic, width, " -")
|
|
result.WriteString(
|
|
styles.NewStyle().
|
|
Background(backgroundColor).
|
|
Foreground(t.Error()).
|
|
Render(diagnostic),
|
|
)
|
|
}
|
|
return result.String()
|
|
}
|
|
}
|
|
return ""
|
|
|
|
// diagnosticsData should be a map[string][]Diagnostic
|
|
// strDiagnosticsData := diagnosticsData.Raw()
|
|
// diagnosticsMap := gjson.Parse(strDiagnosticsData).Value().(map[string]any)
|
|
// fileDiagnostics, ok := diagnosticsMap[filePath]
|
|
// if !ok {
|
|
// return ""
|
|
// }
|
|
|
|
// diagnosticsList, ok := fileDiagnostics.([]any)
|
|
// if !ok {
|
|
// return ""
|
|
// }
|
|
|
|
}
|