header.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. package model
  2. import (
  3. "fmt"
  4. "strings"
  5. "charm.land/lipgloss/v2"
  6. "github.com/charmbracelet/crush/internal/config"
  7. "github.com/charmbracelet/crush/internal/fsext"
  8. "github.com/charmbracelet/crush/internal/session"
  9. "github.com/charmbracelet/crush/internal/ui/common"
  10. "github.com/charmbracelet/crush/internal/ui/styles"
  11. uv "github.com/charmbracelet/ultraviolet"
  12. "github.com/charmbracelet/x/ansi"
  13. )
  14. const (
  15. headerDiag = "╱"
  16. minHeaderDiags = 3
  17. leftPadding = 1
  18. rightPadding = 1
  19. diagToDetailsSpacing = 1 // space between diagonal pattern and details section
  20. )
  21. type header struct {
  22. // cached logo and compact logo
  23. logo string
  24. compactLogo string
  25. com *common.Common
  26. width int
  27. compact bool
  28. }
  29. // newHeader creates a new header model.
  30. func newHeader(com *common.Common) *header {
  31. h := &header{
  32. com: com,
  33. }
  34. t := com.Styles
  35. h.compactLogo = t.Header.Charm.Render("Charm™") + " " +
  36. styles.ApplyBoldForegroundGrad(t, "CRUSH", t.Secondary, t.Primary) + " "
  37. return h
  38. }
  39. // drawHeader draws the header for the given session.
  40. func (h *header) drawHeader(
  41. scr uv.Screen,
  42. area uv.Rectangle,
  43. session *session.Session,
  44. compact bool,
  45. detailsOpen bool,
  46. width int,
  47. ) {
  48. t := h.com.Styles
  49. if width != h.width || compact != h.compact {
  50. h.logo = renderLogo(h.com.Styles, compact, width)
  51. }
  52. h.width = width
  53. h.compact = compact
  54. if !compact || session == nil {
  55. uv.NewStyledString(h.logo).Draw(scr, area)
  56. return
  57. }
  58. if session.ID == "" {
  59. return
  60. }
  61. var b strings.Builder
  62. b.WriteString(h.compactLogo)
  63. availDetailWidth := width - leftPadding - rightPadding - lipgloss.Width(b.String()) - minHeaderDiags - diagToDetailsSpacing
  64. lspErrorCount := 0
  65. for _, info := range h.com.Workspace.LSPGetStates() {
  66. lspErrorCount += info.DiagnosticCount
  67. }
  68. details := renderHeaderDetails(
  69. h.com,
  70. session,
  71. lspErrorCount,
  72. detailsOpen,
  73. availDetailWidth,
  74. )
  75. remainingWidth := width -
  76. lipgloss.Width(b.String()) -
  77. lipgloss.Width(details) -
  78. leftPadding -
  79. rightPadding -
  80. diagToDetailsSpacing
  81. if remainingWidth > 0 {
  82. b.WriteString(t.Header.Diagonals.Render(
  83. strings.Repeat(headerDiag, max(minHeaderDiags, remainingWidth)),
  84. ))
  85. b.WriteString(" ")
  86. }
  87. b.WriteString(details)
  88. view := uv.NewStyledString(
  89. t.Base.Padding(0, rightPadding, 0, leftPadding).Render(b.String()))
  90. view.Draw(scr, area)
  91. }
  92. // renderHeaderDetails renders the details section of the header.
  93. func renderHeaderDetails(
  94. com *common.Common,
  95. session *session.Session,
  96. lspErrorCount int,
  97. detailsOpen bool,
  98. availWidth int,
  99. ) string {
  100. t := com.Styles
  101. var parts []string
  102. if lspErrorCount > 0 {
  103. parts = append(parts, t.LSP.ErrorDiagnostic.Render(fmt.Sprintf("%s%d", styles.LSPErrorIcon, lspErrorCount)))
  104. }
  105. agentCfg := com.Config().Agents[config.AgentCoder]
  106. model := com.Config().GetModelByType(agentCfg.Model)
  107. if model != nil && model.ContextWindow > 0 {
  108. percentage := (float64(session.CompletionTokens+session.PromptTokens) / float64(model.ContextWindow)) * 100
  109. formattedPercentage := t.Header.Percentage.Render(fmt.Sprintf("%d%%", int(percentage)))
  110. parts = append(parts, formattedPercentage)
  111. }
  112. const keystroke = "ctrl+d"
  113. if detailsOpen {
  114. parts = append(parts, t.Header.Keystroke.Render(keystroke)+t.Header.KeystrokeTip.Render(" close"))
  115. } else {
  116. parts = append(parts, t.Header.Keystroke.Render(keystroke)+t.Header.KeystrokeTip.Render(" open "))
  117. }
  118. dot := t.Header.Separator.Render(" • ")
  119. metadata := strings.Join(parts, dot)
  120. metadata = dot + metadata
  121. const dirTrimLimit = 4
  122. cwd := fsext.DirTrim(fsext.PrettyPath(com.Workspace.WorkingDir()), dirTrimLimit)
  123. cwd = t.Header.WorkingDir.Render(cwd)
  124. result := cwd + metadata
  125. return ansi.Truncate(result, max(0, availWidth), "…")
  126. }